PageRenderTime 54ms CodeModel.GetById 23ms RepoModel.GetById 0ms app.codeStats 0ms

/libjava/gnu/java/net/natPlainDatagramSocketImplWin32.cc

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