PageRenderTime 47ms CodeModel.GetById 16ms RepoModel.GetById 0ms app.codeStats 0ms

/gcc-3.2.3-20040701/libjava/java/net/natPlainDatagramSocketImpl.cc

#
C++ | 703 lines | 596 code | 69 blank | 38 comment | 90 complexity | bf85773da4610cdb5e793d1a10941391 MD5 | raw file
Possible License(s): GPL-2.0, LGPL-2.1, LGPL-2.0, CC-BY-SA-3.0
  1. /* Copyright (C) 1999, 2000, 2002 Free Software Foundation
  2. This file is part of libgcj.
  3. This software is copyrighted work licensed under the terms of the
  4. Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
  5. details. */
  6. #include <config.h>
  7. #include <platform.h>
  8. #ifdef WIN32
  9. #include <errno.h>
  10. #include <string.h>
  11. #ifndef ENOPROTOOPT
  12. #define ENOPROTOOPT 109
  13. #endif
  14. static inline int
  15. close(int s)
  16. {
  17. return closesocket(s);
  18. }
  19. #else /* WIN32 */
  20. #ifdef HAVE_SYS_SOCKET_H
  21. #include <sys/socket.h>
  22. #endif
  23. #ifdef HAVE_NETINET_IN_H
  24. #include <netinet/in.h>
  25. #endif
  26. #ifdef HAVE_ARPA_INET_H
  27. #include <arpa/inet.h>
  28. #endif
  29. #include <errno.h>
  30. #include <string.h>
  31. #endif /* WIN32 */
  32. #if HAVE_BSTRING_H
  33. // Needed for bzero, implicitly used by FD_ZERO on IRIX 5.2
  34. #include <bstring.h>
  35. #endif
  36. #ifndef DISABLE_JAVA_NET
  37. // Avoid macro definitions of bind from system headers, e.g. on
  38. // Solaris 7 with _XOPEN_SOURCE. FIXME
  39. static inline int
  40. _Jv_bind (int fd, struct sockaddr *addr, int addrlen)
  41. {
  42. return ::bind (fd, addr, addrlen);
  43. }
  44. #endif /* DISABLE_JAVA_NET */
  45. #ifdef bind
  46. #undef bind
  47. #endif
  48. #include <gcj/cni.h>
  49. #include <java/io/IOException.h>
  50. #include <java/io/InterruptedIOException.h>
  51. #include <java/net/BindException.h>
  52. #include <java/net/SocketException.h>
  53. #include <java/net/PlainDatagramSocketImpl.h>
  54. #include <java/net/InetAddress.h>
  55. #include <java/net/DatagramPacket.h>
  56. #include <java/lang/InternalError.h>
  57. #include <java/lang/Object.h>
  58. #include <java/lang/Boolean.h>
  59. #include <java/lang/Integer.h>
  60. // FIXME: remove these
  61. #define BooleanClass java::lang::Boolean::class$
  62. #define IntegerClass java::lang::Integer::class$
  63. #ifdef DISABLE_JAVA_NET
  64. void
  65. java::net::PlainDatagramSocketImpl::create ()
  66. {
  67. throw new SocketException (
  68. JvNewStringLatin1 ("DatagramSocketImpl.create: unimplemented"));
  69. }
  70. void
  71. java::net::PlainDatagramSocketImpl::bind (jint, java::net::InetAddress *)
  72. {
  73. throw new BindException (
  74. JvNewStringLatin1 ("DatagramSocketImpl.bind: unimplemented"));
  75. }
  76. jint
  77. java::net::PlainDatagramSocketImpl::peek (java::net::InetAddress *)
  78. {
  79. throw new java::io::IOException (
  80. JvNewStringLatin1 ("DatagramSocketImpl.peek: unimplemented"));
  81. }
  82. void
  83. java::net::PlainDatagramSocketImpl::close ()
  84. {
  85. throw new java::io::IOException (
  86. JvNewStringLatin1 ("DatagramSocketImpl.close: unimplemented"));
  87. }
  88. void
  89. java::net::PlainDatagramSocketImpl::send (java::net::DatagramPacket *)
  90. {
  91. throw new java::io::IOException (
  92. JvNewStringLatin1 ("DatagramSocketImpl.send: unimplemented"));
  93. }
  94. void
  95. java::net::PlainDatagramSocketImpl::receive (java::net::DatagramPacket *)
  96. {
  97. throw new java::io::IOException (
  98. JvNewStringLatin1 ("DatagramSocketImpl.receive: unimplemented"));
  99. }
  100. void
  101. java::net::PlainDatagramSocketImpl::setTimeToLive (jint)
  102. {
  103. throw new java::io::IOException (
  104. JvNewStringLatin1 ("DatagramSocketImpl.setTimeToLive: unimplemented"));
  105. }
  106. jint
  107. java::net::PlainDatagramSocketImpl::getTimeToLive ()
  108. {
  109. throw new java::io::IOException (
  110. JvNewStringLatin1 ("DatagramSocketImpl.getTimeToLive: unimplemented"));
  111. }
  112. void
  113. java::net::PlainDatagramSocketImpl::mcastGrp (java::net::InetAddress *,
  114. jboolean)
  115. {
  116. throw new java::io::IOException (
  117. JvNewStringLatin1 ("DatagramSocketImpl.mcastGrp: unimplemented"));
  118. }
  119. void
  120. java::net::PlainDatagramSocketImpl::setOption (jint, java::lang::Object *)
  121. {
  122. throw new SocketException (
  123. JvNewStringLatin1 ("DatagramSocketImpl.setOption: unimplemented"));
  124. }
  125. java::lang::Object *
  126. java::net::PlainDatagramSocketImpl::getOption (jint)
  127. {
  128. throw new SocketException (
  129. JvNewStringLatin1 ("DatagramSocketImpl.getOption: unimplemented"));
  130. }
  131. #else /* DISABLE_JAVA_NET */
  132. #ifndef HAVE_SOCKLEN_T
  133. typedef int socklen_t;
  134. #endif
  135. union SockAddr
  136. {
  137. struct sockaddr_in address;
  138. #ifdef HAVE_INET6
  139. struct sockaddr_in6 address6;
  140. #endif
  141. };
  142. union McastReq
  143. {
  144. #if HAVE_STRUCT_IP_MREQ
  145. struct ip_mreq mreq;
  146. #endif
  147. #if HAVE_STRUCT_IPV6_MREQ
  148. struct ipv6_mreq mreq6;
  149. #endif
  150. };
  151. union InAddr
  152. {
  153. struct in_addr addr;
  154. #ifdef HAVE_INET6
  155. struct in6_addr addr6;
  156. #endif
  157. };
  158. // FIXME: routines here and/or in natPlainSocketImpl.cc could throw
  159. // NoRouteToHostException; also consider UnknownHostException, ConnectException.
  160. void
  161. java::net::PlainDatagramSocketImpl::create ()
  162. {
  163. int sock = ::socket (AF_INET, SOCK_DGRAM, 0);
  164. if (sock < 0)
  165. {
  166. char* strerr = strerror (errno);
  167. throw new java::net::SocketException (JvNewStringUTF (strerr));
  168. }
  169. _Jv_platform_close_on_exec (sock);
  170. // We use fnum in place of fd here. From leaving fd null we avoid
  171. // the double close problem in FileDescriptor.finalize.
  172. fnum = sock;
  173. }
  174. void
  175. java::net::PlainDatagramSocketImpl::bind (jint lport,
  176. java::net::InetAddress *host)
  177. {
  178. union SockAddr u;
  179. struct sockaddr *ptr = (struct sockaddr *) &u.address;
  180. // FIXME: Use getaddrinfo() to get actual protocol instead of assuming ipv4.
  181. jbyteArray haddress = host->addr;
  182. jbyte *bytes = elements (haddress);
  183. int len = haddress->length;
  184. if (len == 4)
  185. {
  186. u.address.sin_family = AF_INET;
  187. if (host != NULL)
  188. memcpy (&u.address.sin_addr, bytes, len);
  189. else
  190. u.address.sin_addr.s_addr = htonl (INADDR_ANY);
  191. len = sizeof (struct sockaddr_in);
  192. u.address.sin_port = htons (lport);
  193. }
  194. #ifdef HAVE_INET6
  195. else if (len == 16)
  196. {
  197. u.address6.sin6_family = AF_INET6;
  198. memcpy (&u.address6.sin6_addr, bytes, len);
  199. len = sizeof (struct sockaddr_in6);
  200. u.address6.sin6_port = htons (lport);
  201. }
  202. #endif
  203. else
  204. throw new java::net::SocketException (JvNewStringUTF ("invalid length"));
  205. if (_Jv_bind (fnum, ptr, len) == 0)
  206. {
  207. socklen_t addrlen = sizeof(u);
  208. if (lport != 0)
  209. localPort = lport;
  210. else if (::getsockname (fnum, (sockaddr*) &u, &addrlen) == 0)
  211. localPort = ntohs (u.address.sin_port);
  212. else
  213. goto error;
  214. /* Allow broadcast by default. */
  215. int broadcast = 1;
  216. if (::setsockopt (fnum, SOL_SOCKET, SO_BROADCAST, (char *) &broadcast,
  217. sizeof (broadcast)) != 0)
  218. goto error;
  219. return;
  220. }
  221. error:
  222. char* strerr = strerror (errno);
  223. throw new java::net::BindException (JvNewStringUTF (strerr));
  224. }
  225. jint
  226. java::net::PlainDatagramSocketImpl::peek (java::net::InetAddress *i)
  227. {
  228. // FIXME: Deal with Multicast and if the socket is connected.
  229. union SockAddr u;
  230. socklen_t addrlen = sizeof(u);
  231. ssize_t retlen =
  232. ::recvfrom (fnum, (char *) NULL, 0, MSG_PEEK, (sockaddr*) &u,
  233. &addrlen);
  234. if (retlen < 0)
  235. goto error;
  236. // FIXME: Deal with Multicast addressing and if the socket is connected.
  237. jbyteArray raddr;
  238. jint rport;
  239. if (u.address.sin_family == AF_INET)
  240. {
  241. raddr = JvNewByteArray (4);
  242. memcpy (elements (raddr), &u.address.sin_addr, 4);
  243. rport = ntohs (u.address.sin_port);
  244. }
  245. #ifdef HAVE_INET6
  246. else if (u.address.sin_family == AF_INET6)
  247. {
  248. raddr = JvNewByteArray (16);
  249. memcpy (elements (raddr), &u.address6.sin6_addr, 16);
  250. rport = ntohs (u.address6.sin6_port);
  251. }
  252. #endif
  253. else
  254. throw new java::net::SocketException (JvNewStringUTF ("invalid family"));
  255. i->addr = raddr;
  256. return rport;
  257. error:
  258. char* strerr = strerror (errno);
  259. throw new java::io::IOException (JvNewStringUTF (strerr));
  260. }
  261. // Close(shutdown) the socket.
  262. void
  263. java::net::PlainDatagramSocketImpl::close ()
  264. {
  265. // Avoid races from asynchronous finalization.
  266. JvSynchronize sync (this);
  267. // The method isn't declared to throw anything, so we disregard
  268. // the return value.
  269. ::close (fnum);
  270. fnum = -1;
  271. timeout = 0;
  272. }
  273. void
  274. java::net::PlainDatagramSocketImpl::send (java::net::DatagramPacket *p)
  275. {
  276. // FIXME: Deal with Multicast and if the socket is connected.
  277. jint rport = p->getPort();
  278. union SockAddr u;
  279. jbyteArray haddress = p->getAddress()->addr;
  280. jbyte *bytes = elements (haddress);
  281. int len = haddress->length;
  282. struct sockaddr *ptr = (struct sockaddr *) &u.address;
  283. jbyte *dbytes = elements (p->getData());
  284. if (len == 4)
  285. {
  286. u.address.sin_family = AF_INET;
  287. memcpy (&u.address.sin_addr, bytes, len);
  288. len = sizeof (struct sockaddr_in);
  289. u.address.sin_port = htons (rport);
  290. }
  291. #ifdef HAVE_INET6
  292. else if (len == 16)
  293. {
  294. u.address6.sin6_family = AF_INET6;
  295. memcpy (&u.address6.sin6_addr, bytes, len);
  296. len = sizeof (struct sockaddr_in6);
  297. u.address6.sin6_port = htons (rport);
  298. }
  299. #endif
  300. else
  301. throw new java::net::SocketException (JvNewStringUTF ("invalid length"));
  302. if (::sendto (fnum, (char *) dbytes, p->getLength(), 0, ptr, len) >= 0)
  303. return;
  304. char* strerr = strerror (errno);
  305. throw new java::io::IOException (JvNewStringUTF (strerr));
  306. }
  307. void
  308. java::net::PlainDatagramSocketImpl::receive (java::net::DatagramPacket *p)
  309. {
  310. // FIXME: Deal with Multicast and if the socket is connected.
  311. union SockAddr u;
  312. socklen_t addrlen = sizeof(u);
  313. jbyte *dbytes = elements (p->getData());
  314. ssize_t retlen = 0;
  315. // FIXME: implement timeout support for Win32
  316. #ifndef WIN32
  317. // Do timeouts via select since SO_RCVTIMEO is not always available.
  318. if (timeout > 0)
  319. {
  320. fd_set rset;
  321. struct timeval tv;
  322. FD_ZERO(&rset);
  323. FD_SET(fnum, &rset);
  324. tv.tv_sec = timeout / 1000;
  325. tv.tv_usec = (timeout % 1000) * 1000;
  326. int retval;
  327. if ((retval = _Jv_select (fnum + 1, &rset, NULL, NULL, &tv)) < 0)
  328. goto error;
  329. else if (retval == 0)
  330. throw new java::io::InterruptedIOException ();
  331. }
  332. #endif /* WIN32 */
  333. retlen =
  334. ::recvfrom (fnum, (char *) dbytes, p->getLength(), 0, (sockaddr*) &u,
  335. &addrlen);
  336. if (retlen < 0)
  337. goto error;
  338. // FIXME: Deal with Multicast addressing and if the socket is connected.
  339. jbyteArray raddr;
  340. jint rport;
  341. if (u.address.sin_family == AF_INET)
  342. {
  343. raddr = JvNewByteArray (4);
  344. memcpy (elements (raddr), &u.address.sin_addr, 4);
  345. rport = ntohs (u.address.sin_port);
  346. }
  347. #ifdef HAVE_INET6
  348. else if (u.address.sin_family == AF_INET6)
  349. {
  350. raddr = JvNewByteArray (16);
  351. memcpy (elements (raddr), &u.address6.sin6_addr, 16);
  352. rport = ntohs (u.address6.sin6_port);
  353. }
  354. #endif
  355. else
  356. throw new java::net::SocketException (JvNewStringUTF ("invalid family"));
  357. p->setAddress (new InetAddress (raddr, NULL));
  358. p->setPort (rport);
  359. p->setLength ((jint) retlen);
  360. return;
  361. error:
  362. char* strerr = strerror (errno);
  363. throw new java::io::IOException (JvNewStringUTF (strerr));
  364. }
  365. void
  366. java::net::PlainDatagramSocketImpl::setTimeToLive (jint ttl)
  367. {
  368. // Assumes IPPROTO_IP rather than IPPROTO_IPV6 since socket created is IPv4.
  369. char val = (char) ttl;
  370. socklen_t val_len = sizeof(val);
  371. if (::setsockopt (fnum, IPPROTO_IP, IP_MULTICAST_TTL, &val, val_len) == 0)
  372. return;
  373. char* strerr = strerror (errno);
  374. throw new java::io::IOException (JvNewStringUTF (strerr));
  375. }
  376. jint
  377. java::net::PlainDatagramSocketImpl::getTimeToLive ()
  378. {
  379. // Assumes IPPROTO_IP rather than IPPROTO_IPV6 since socket created is IPv4.
  380. char val;
  381. socklen_t val_len = sizeof(val);
  382. if (::getsockopt (fnum, IPPROTO_IP, IP_MULTICAST_TTL, &val, &val_len) == 0)
  383. return ((int) val) & 0xFF;
  384. char* strerr = strerror (errno);
  385. throw new java::io::IOException (JvNewStringUTF (strerr));
  386. }
  387. void
  388. java::net::PlainDatagramSocketImpl::mcastGrp (java::net::InetAddress *inetaddr,
  389. jboolean join)
  390. {
  391. union McastReq u;
  392. jbyteArray haddress = inetaddr->addr;
  393. jbyte *bytes = elements (haddress);
  394. int len = haddress->length;
  395. int level, opname;
  396. const char *ptr;
  397. if (0)
  398. ;
  399. #if HAVE_STRUCT_IP_MREQ
  400. else if (len == 4)
  401. {
  402. level = IPPROTO_IP;
  403. opname = join ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP;
  404. memcpy (&u.mreq.imr_multiaddr, bytes, len);
  405. // FIXME: If a non-default interface is set, use it; see Stevens p. 501.
  406. // Maybe not, see note in last paragraph at bottom of Stevens p. 497.
  407. u.mreq.imr_interface.s_addr = htonl (INADDR_ANY);
  408. len = sizeof (struct ip_mreq);
  409. ptr = (const char *) &u.mreq;
  410. }
  411. #endif
  412. #if HAVE_STRUCT_IPV6_MREQ
  413. else if (len == 16)
  414. {
  415. level = IPPROTO_IPV6;
  416. /* Prefer new RFC 2553 names. */
  417. #ifndef IPV6_JOIN_GROUP
  418. #define IPV6_JOIN_GROUP IPV6_ADD_MEMBERSHIP
  419. #endif
  420. #ifndef IPV6_LEAVE_GROUP
  421. #define IPV6_LEAVE_GROUP IPV6_DROP_MEMBERSHIP
  422. #endif
  423. opname = join ? IPV6_JOIN_GROUP : IPV6_LEAVE_GROUP;
  424. memcpy (&u.mreq6.ipv6mr_multiaddr, bytes, len);
  425. // FIXME: If a non-default interface is set, use it; see Stevens p. 501.
  426. // Maybe not, see note in last paragraph at bottom of Stevens p. 497.
  427. u.mreq6.ipv6mr_interface = 0;
  428. len = sizeof (struct ipv6_mreq);
  429. ptr = (const char *) &u.mreq6;
  430. }
  431. #endif
  432. else
  433. throw new java::net::SocketException (JvNewStringUTF ("invalid length"));
  434. if (::setsockopt (fnum, level, opname, ptr, len) == 0)
  435. return;
  436. char* strerr = strerror (errno);
  437. throw new java::io::IOException (JvNewStringUTF (strerr));
  438. }
  439. void
  440. java::net::PlainDatagramSocketImpl::setOption (jint optID,
  441. java::lang::Object *value)
  442. {
  443. int val;
  444. socklen_t val_len = sizeof (val);
  445. if (_Jv_IsInstanceOf (value, &BooleanClass))
  446. {
  447. java::lang::Boolean *boolobj =
  448. static_cast<java::lang::Boolean *> (value);
  449. val = boolobj->booleanValue() ? 1 : 0;
  450. }
  451. else if (_Jv_IsInstanceOf (value, &IntegerClass))
  452. {
  453. java::lang::Integer *intobj =
  454. static_cast<java::lang::Integer *> (value);
  455. val = (int) intobj->intValue();
  456. }
  457. // Else assume value to be an InetAddress for use with IP_MULTICAST_IF.
  458. switch (optID)
  459. {
  460. case _Jv_TCP_NODELAY_ :
  461. throw new java::net::SocketException (
  462. JvNewStringUTF ("TCP_NODELAY not valid for UDP"));
  463. return;
  464. case _Jv_SO_LINGER_ :
  465. throw new java::net::SocketException (
  466. JvNewStringUTF ("SO_LINGER not valid for UDP"));
  467. return;
  468. case _Jv_SO_SNDBUF_ :
  469. case _Jv_SO_RCVBUF_ :
  470. #if defined(SO_SNDBUF) && defined(SO_RCVBUF)
  471. int opt;
  472. optID == _Jv_SO_SNDBUF_ ? opt = SO_SNDBUF : opt = SO_RCVBUF;
  473. if (::setsockopt (fnum, SOL_SOCKET, opt, (char *) &val, val_len) != 0)
  474. goto error;
  475. #else
  476. throw new java::lang::InternalError (
  477. JvNewStringUTF ("SO_RCVBUF/SO_SNDBUF not supported"));
  478. #endif
  479. return;
  480. case _Jv_SO_REUSEADDR_ :
  481. #if defined(SO_REUSEADDR)
  482. if (::setsockopt (fnum, SOL_SOCKET, SO_REUSEADDR, (char *) &val,
  483. val_len) != 0)
  484. goto error;
  485. #else
  486. throw new java::lang::InternalError (
  487. JvNewStringUTF ("SO_REUSEADDR not supported"));
  488. #endif
  489. return;
  490. case _Jv_SO_BINDADDR_ :
  491. throw new java::net::SocketException (
  492. JvNewStringUTF ("SO_BINDADDR: read only option"));
  493. return;
  494. case _Jv_IP_MULTICAST_IF_ :
  495. union InAddr u;
  496. jbyteArray haddress;
  497. jbyte *bytes;
  498. int len;
  499. int level, opname;
  500. const char *ptr;
  501. haddress = ((java::net::InetAddress *) value)->addr;
  502. bytes = elements (haddress);
  503. len = haddress->length;
  504. if (len == 4)
  505. {
  506. level = IPPROTO_IP;
  507. opname = IP_MULTICAST_IF;
  508. memcpy (&u.addr, bytes, len);
  509. len = sizeof (struct in_addr);
  510. ptr = (const char *) &u.addr;
  511. }
  512. // Tru64 UNIX V5.0 has struct sockaddr_in6, but no IPV6_MULTICAST_IF
  513. #if defined (HAVE_INET6) && defined (IPV6_MULTICAST_IF)
  514. else if (len == 16)
  515. {
  516. level = IPPROTO_IPV6;
  517. opname = IPV6_MULTICAST_IF;
  518. memcpy (&u.addr6, bytes, len);
  519. len = sizeof (struct in6_addr);
  520. ptr = (const char *) &u.addr6;
  521. }
  522. #endif
  523. else
  524. throw
  525. new java::net::SocketException (JvNewStringUTF ("invalid length"));
  526. if (::setsockopt (fnum, level, opname, ptr, len) != 0)
  527. goto error;
  528. return;
  529. case _Jv_SO_TIMEOUT_ :
  530. timeout = val;
  531. return;
  532. default :
  533. errno = ENOPROTOOPT;
  534. }
  535. error:
  536. char* strerr = strerror (errno);
  537. throw new java::net::SocketException (JvNewStringUTF (strerr));
  538. }
  539. java::lang::Object *
  540. java::net::PlainDatagramSocketImpl::getOption (jint optID)
  541. {
  542. int val;
  543. socklen_t val_len = sizeof(val);
  544. union SockAddr u;
  545. socklen_t addrlen = sizeof(u);
  546. switch (optID)
  547. {
  548. case _Jv_TCP_NODELAY_ :
  549. throw new java::net::SocketException (
  550. JvNewStringUTF ("TCP_NODELAY not valid for UDP"));
  551. break;
  552. case _Jv_SO_LINGER_ :
  553. throw new java::net::SocketException (
  554. JvNewStringUTF ("SO_LINGER not valid for UDP"));
  555. break;
  556. case _Jv_SO_RCVBUF_ :
  557. case _Jv_SO_SNDBUF_ :
  558. #if defined(SO_SNDBUF) && defined(SO_RCVBUF)
  559. int opt;
  560. optID == _Jv_SO_SNDBUF_ ? opt = SO_SNDBUF : opt = SO_RCVBUF;
  561. if (::getsockopt (fnum, SOL_SOCKET, opt, (char *) &val, &val_len) != 0)
  562. goto error;
  563. else
  564. return new java::lang::Integer (val);
  565. #else
  566. throw new java::lang::InternalError (
  567. JvNewStringUTF ("SO_RCVBUF/SO_SNDBUF not supported"));
  568. #endif
  569. break;
  570. case _Jv_SO_BINDADDR_:
  571. // cache the local address
  572. if (localAddress == NULL)
  573. {
  574. jbyteArray laddr;
  575. if (::getsockname (fnum, (sockaddr*) &u, &addrlen) != 0)
  576. goto error;
  577. if (u.address.sin_family == AF_INET)
  578. {
  579. laddr = JvNewByteArray (4);
  580. memcpy (elements (laddr), &u.address.sin_addr, 4);
  581. }
  582. #ifdef HAVE_INET6
  583. else if (u.address.sin_family == AF_INET6)
  584. {
  585. laddr = JvNewByteArray (16);
  586. memcpy (elements (laddr), &u.address6.sin6_addr, 16);
  587. }
  588. #endif
  589. else
  590. throw new java::net::SocketException (JvNewStringUTF ("invalid family"));
  591. localAddress = new java::net::InetAddress (laddr, NULL);
  592. }
  593. return localAddress;
  594. break;
  595. case _Jv_SO_REUSEADDR_ :
  596. #if defined(SO_REUSEADDR)
  597. if (::getsockopt (fnum, SOL_SOCKET, SO_REUSEADDR, (char *) &val,
  598. &val_len) != 0)
  599. goto error;
  600. return new java::lang::Boolean (val != 0);
  601. #else
  602. throw new java::lang::InternalError (
  603. JvNewStringUTF ("SO_REUSEADDR not supported"));
  604. #endif
  605. break;
  606. case _Jv_IP_MULTICAST_IF_ :
  607. #ifdef HAVE_INET_NTOA
  608. struct in_addr inaddr;
  609. socklen_t inaddr_len;
  610. char *bytes;
  611. inaddr_len = sizeof(inaddr);
  612. if (::getsockopt (fnum, IPPROTO_IP, IP_MULTICAST_IF, (char *) &inaddr,
  613. &inaddr_len) != 0)
  614. goto error;
  615. bytes = inet_ntoa (inaddr);
  616. return java::net::InetAddress::getByName (JvNewStringLatin1 (bytes));
  617. #else
  618. throw new java::net::SocketException (
  619. JvNewStringUTF ("IP_MULTICAST_IF: not available - no inet_ntoa()"));
  620. #endif
  621. break;
  622. case _Jv_SO_TIMEOUT_ :
  623. return new java::lang::Integer (timeout);
  624. break;
  625. default :
  626. errno = ENOPROTOOPT;
  627. }
  628. error:
  629. char* strerr = strerror (errno);
  630. throw new java::net::SocketException (JvNewStringUTF (strerr));
  631. }
  632. #endif /* DISABLE_JAVA_NET */