PageRenderTime 49ms CodeModel.GetById 20ms RepoModel.GetById 0ms app.codeStats 0ms

/libjava/gnu/java/net/natPlainDatagramSocketImplPosix.cc

https://gitlab.com/lobl.pavel/gcc-6.2.0
C++ | 852 lines | 713 code | 96 blank | 43 comment | 153 complexity | bae3e6c8bad7d2189bc811b5065637b0 MD5 | raw file
  1. /* Copyright (C) 2003, 2005, 2006, 2012 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 HAVE_NETINET_IN_H
  9. #include <netinet/in.h>
  10. #endif
  11. #ifdef HAVE_ARPA_INET_H
  12. #include <arpa/inet.h>
  13. #endif
  14. #include <errno.h>
  15. #include <string.h>
  16. #include <gcj/cni.h>
  17. #include <gnu/java/net/PlainDatagramSocketImpl.h>
  18. #include <java/io/IOException.h>
  19. #include <java/io/InterruptedIOException.h>
  20. #include <java/net/BindException.h>
  21. #include <java/net/SocketException.h>
  22. #include <java/net/SocketTimeoutException.h>
  23. #include <java/net/InetAddress.h>
  24. #include <java/net/NetworkInterface.h>
  25. #include <java/net/DatagramPacket.h>
  26. #include <java/net/PortUnreachableException.h>
  27. #include <java/lang/InternalError.h>
  28. #include <java/lang/Object.h>
  29. #include <java/lang/Boolean.h>
  30. #include <java/lang/Integer.h>
  31. #include <java/net/UnknownHostException.h>
  32. #include <java/net/ConnectException.h>
  33. #include <java/lang/NullPointerException.h>
  34. union SockAddr
  35. {
  36. struct sockaddr_in address;
  37. #ifdef HAVE_INET6
  38. struct sockaddr_in6 address6;
  39. #endif
  40. };
  41. union McastReq
  42. {
  43. #if HAVE_STRUCT_IP_MREQ
  44. struct ip_mreq mreq;
  45. #endif
  46. #if HAVE_STRUCT_IPV6_MREQ
  47. struct ipv6_mreq mreq6;
  48. #endif
  49. };
  50. union InAddr
  51. {
  52. struct in_addr addr;
  53. #ifdef HAVE_INET6
  54. struct in6_addr addr6;
  55. #endif
  56. };
  57. // FIXME: routines here and/or in natPlainSocketImpl.cc could throw
  58. // NoRouteToHostException; also consider UnknownHostException, ConnectException.
  59. void
  60. gnu::java::net::PlainDatagramSocketImpl::create ()
  61. {
  62. int sock = _Jv_socket (AF_INET, SOCK_DGRAM, 0);
  63. if (sock < 0)
  64. {
  65. char* strerr = strerror (errno);
  66. throw new ::java::net::SocketException (JvNewStringUTF (strerr));
  67. }
  68. // We use native_fd in place of fd here. From leaving fd null we avoid
  69. // the double close problem in FileDescriptor.finalize.
  70. native_fd = sock;
  71. }
  72. void
  73. gnu::java::net::PlainDatagramSocketImpl::bind (jint lport,
  74. ::java::net::InetAddress *host)
  75. {
  76. union SockAddr u;
  77. struct sockaddr *ptr = (struct sockaddr *) &u.address;
  78. // FIXME: Use getaddrinfo() to get actual protocol instead of assuming ipv4.
  79. jbyteArray haddress = host->addr;
  80. jbyte *bytes = elements (haddress);
  81. int len = haddress->length;
  82. if (len == 4)
  83. {
  84. u.address.sin_family = AF_INET;
  85. if (host != NULL)
  86. memcpy (&u.address.sin_addr, bytes, len);
  87. else
  88. u.address.sin_addr.s_addr = htonl (INADDR_ANY);
  89. len = sizeof (struct sockaddr_in);
  90. u.address.sin_port = htons (lport);
  91. }
  92. #ifdef HAVE_INET6
  93. else if (len == 16)
  94. {
  95. u.address6.sin6_family = AF_INET6;
  96. memcpy (&u.address6.sin6_addr, bytes, len);
  97. len = sizeof (struct sockaddr_in6);
  98. u.address6.sin6_port = htons (lport);
  99. }
  100. #endif
  101. else
  102. throw new ::java::net::SocketException (JvNewStringUTF ("invalid length"));
  103. if (_Jv_bind (native_fd, ptr, len) == 0)
  104. {
  105. socklen_t addrlen = sizeof(u);
  106. if (lport != 0)
  107. localPort = lport;
  108. else if (::getsockname (native_fd, (sockaddr*) &u, &addrlen) == 0)
  109. localPort = ntohs (u.address.sin_port);
  110. else
  111. goto error;
  112. /* Allow broadcast by default. */
  113. int broadcast = 1;
  114. if (::setsockopt (native_fd, SOL_SOCKET, SO_BROADCAST, (char *) &broadcast,
  115. sizeof (broadcast)) != 0)
  116. goto error;
  117. return;
  118. }
  119. error:
  120. char* strerr = strerror (errno);
  121. throw new ::java::net::BindException (JvNewStringUTF (strerr));
  122. }
  123. void
  124. gnu::java::net::PlainDatagramSocketImpl::connect (::java::net::InetAddress *host,
  125. jint rport)
  126. {
  127. if (! host)
  128. throw new ::java::lang::NullPointerException;
  129. union SockAddr u;
  130. jbyteArray haddress = host->addr;
  131. jbyte *bytes = elements (haddress);
  132. int len = haddress->length;
  133. struct sockaddr *ptr = (struct sockaddr *) &u.address;
  134. if (len == 4)
  135. {
  136. u.address.sin_family = AF_INET;
  137. memcpy (&u.address.sin_addr, bytes, len);
  138. len = sizeof (struct sockaddr_in);
  139. u.address.sin_port = htons (rport);
  140. }
  141. #ifdef HAVE_INET6
  142. else if (len == 16)
  143. {
  144. u.address6.sin6_family = AF_INET6;
  145. memcpy (&u.address6.sin6_addr, bytes, len);
  146. len = sizeof (struct sockaddr_in6);
  147. u.address6.sin6_port = htons (rport);
  148. }
  149. #endif
  150. else
  151. throw new ::java::net::SocketException (JvNewStringUTF ("invalid length"));
  152. if (_Jv_connect (native_fd, ptr, len) == 0)
  153. return;
  154. char* strerr = strerror (errno);
  155. throw new ::java::net::ConnectException (JvNewStringUTF (strerr));
  156. }
  157. void
  158. gnu::java::net::PlainDatagramSocketImpl::disconnect ()
  159. {
  160. struct sockaddr addr;
  161. addr.sa_family = AF_UNSPEC;
  162. // Ignore errors. This is lame but apparently required.
  163. _Jv_connect (native_fd, &addr, sizeof (addr));
  164. }
  165. jint
  166. gnu::java::net::PlainDatagramSocketImpl::peek (::java::net::InetAddress *i)
  167. {
  168. // FIXME: Deal with Multicast and if the socket is connected.
  169. union SockAddr u;
  170. socklen_t addrlen = sizeof(u);
  171. ssize_t retlen =
  172. ::recvfrom (native_fd, (char *) NULL, 0, MSG_PEEK, (sockaddr*) &u,
  173. &addrlen);
  174. if (retlen < 0)
  175. goto error;
  176. // FIXME: Deal with Multicast addressing and if the socket is connected.
  177. jbyteArray raddr;
  178. jint rport;
  179. if (u.address.sin_family == AF_INET)
  180. {
  181. raddr = JvNewByteArray (4);
  182. memcpy (elements (raddr), &u.address.sin_addr, 4);
  183. rport = ntohs (u.address.sin_port);
  184. }
  185. #ifdef HAVE_INET6
  186. else if (u.address.sin_family == AF_INET6)
  187. {
  188. raddr = JvNewByteArray (16);
  189. memcpy (elements (raddr), &u.address6.sin6_addr, 16);
  190. rport = ntohs (u.address6.sin6_port);
  191. }
  192. #endif
  193. else
  194. throw new ::java::net::SocketException (JvNewStringUTF ("invalid family"));
  195. i->addr = raddr;
  196. return rport;
  197. error:
  198. char* strerr = strerror (errno);
  199. if (errno == ECONNREFUSED)
  200. throw new ::java::net::PortUnreachableException (JvNewStringUTF (strerr));
  201. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  202. }
  203. jint
  204. gnu::java::net::PlainDatagramSocketImpl::peekData (::java::net::DatagramPacket *p)
  205. {
  206. // FIXME: Deal with Multicast and if the socket is connected.
  207. union SockAddr u;
  208. socklen_t addrlen = sizeof(u);
  209. jbyte *dbytes = elements (p->getData()) + p->getOffset();
  210. jint maxlen = p->maxlen - p->getOffset();
  211. ssize_t retlen = 0;
  212. // Do timeouts via select since SO_RCVTIMEO is not always available.
  213. if (timeout > 0 && native_fd >= 0 && native_fd < FD_SETSIZE)
  214. {
  215. fd_set rset;
  216. struct timeval tv;
  217. FD_ZERO(&rset);
  218. FD_SET(native_fd, &rset);
  219. tv.tv_sec = timeout / 1000;
  220. tv.tv_usec = (timeout % 1000) * 1000;
  221. int retval;
  222. if ((retval = _Jv_select (native_fd + 1, &rset, NULL, NULL, &tv)) < 0)
  223. goto error;
  224. else if (retval == 0)
  225. throw new ::java::net::SocketTimeoutException
  226. (JvNewStringUTF ("PeekData timed out") );
  227. }
  228. retlen =
  229. ::recvfrom (native_fd, (char *) dbytes, maxlen, MSG_PEEK, (sockaddr*) &u,
  230. &addrlen);
  231. if (retlen < 0)
  232. goto error;
  233. // FIXME: Deal with Multicast addressing and if the socket is connected.
  234. jbyteArray raddr;
  235. jint rport;
  236. if (u.address.sin_family == AF_INET)
  237. {
  238. raddr = JvNewByteArray (4);
  239. memcpy (elements (raddr), &u.address.sin_addr, 4);
  240. rport = ntohs (u.address.sin_port);
  241. }
  242. #ifdef HAVE_INET6
  243. else if (u.address.sin_family == AF_INET6)
  244. {
  245. raddr = JvNewByteArray (16);
  246. memcpy (elements (raddr), &u.address6.sin6_addr, 16);
  247. rport = ntohs (u.address6.sin6_port);
  248. }
  249. #endif
  250. else
  251. throw new ::java::net::SocketException (JvNewStringUTF ("invalid family"));
  252. p->setAddress (::java::net::InetAddress::getByAddress (raddr));
  253. p->setPort (rport);
  254. p->length = (int) retlen;
  255. return rport;
  256. error:
  257. char* strerr = strerror (errno);
  258. if (errno == ECONNREFUSED)
  259. throw new ::java::net::PortUnreachableException (JvNewStringUTF (strerr));
  260. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  261. }
  262. // Close(shutdown) the socket.
  263. void
  264. gnu::java::net::PlainDatagramSocketImpl::close ()
  265. {
  266. // Avoid races from asynchronous finalization.
  267. JvSynchronize sync (this);
  268. // The method isn't declared to throw anything, so we disregard
  269. // the return value.
  270. _Jv_close (native_fd);
  271. native_fd = -1;
  272. timeout = 0;
  273. }
  274. void
  275. gnu::java::net::PlainDatagramSocketImpl::send (::java::net::DatagramPacket *p)
  276. {
  277. JvSynchronize lock (SEND_LOCK);
  278. // FIXME: Deal with Multicast.
  279. ::java::net::InetAddress *host = p->getAddress();
  280. if (host == NULL)
  281. {
  282. // If there is no host, maybe this socket was connected, in
  283. // which case we try a plain send().
  284. jbyte *dbytes = elements (p->getData()) + p->getOffset();
  285. if (::send (native_fd, (char *) dbytes, p->getLength(), 0) >= 0)
  286. return;
  287. }
  288. else
  289. {
  290. jint rport = p->getPort();
  291. union SockAddr u;
  292. jbyteArray haddress = host->addr;
  293. jbyte *bytes = elements (haddress);
  294. int len = haddress->length;
  295. struct sockaddr *ptr = (struct sockaddr *) &u.address;
  296. jbyte *dbytes = elements (p->getData()) + p->getOffset();
  297. if (len == 4)
  298. {
  299. u.address.sin_family = AF_INET;
  300. memcpy (&u.address.sin_addr, bytes, len);
  301. len = sizeof (struct sockaddr_in);
  302. u.address.sin_port = htons (rport);
  303. }
  304. #ifdef HAVE_INET6
  305. else if (len == 16)
  306. {
  307. u.address6.sin6_family = AF_INET6;
  308. memcpy (&u.address6.sin6_addr, bytes, len);
  309. len = sizeof (struct sockaddr_in6);
  310. u.address6.sin6_port = htons (rport);
  311. }
  312. #endif
  313. else
  314. throw new ::java::net::SocketException (JvNewStringUTF ("invalid length"));
  315. if (::sendto (native_fd, (char *) dbytes, p->getLength(), 0, ptr, len)
  316. >= 0)
  317. return;
  318. }
  319. char* strerr = strerror (errno);
  320. if (errno == ECONNREFUSED)
  321. throw new ::java::net::PortUnreachableException (JvNewStringUTF (strerr));
  322. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  323. }
  324. void
  325. gnu::java::net::PlainDatagramSocketImpl::receive (::java::net::DatagramPacket *p)
  326. {
  327. JvSynchronize lock (RECEIVE_LOCK);
  328. // FIXME: Deal with Multicast and if the socket is connected.
  329. union SockAddr u;
  330. socklen_t addrlen = sizeof(u);
  331. jbyte *dbytes = elements (p->getData()) + p->getOffset();
  332. jint maxlen = p->maxlen - p->getOffset();
  333. ssize_t retlen = 0;
  334. // Do timeouts via select since SO_RCVTIMEO is not always available.
  335. if (timeout > 0 && native_fd >= 0 && native_fd < FD_SETSIZE)
  336. {
  337. fd_set rset;
  338. struct timeval tv;
  339. FD_ZERO(&rset);
  340. FD_SET(native_fd, &rset);
  341. tv.tv_sec = timeout / 1000;
  342. tv.tv_usec = (timeout % 1000) * 1000;
  343. int retval;
  344. if ((retval = _Jv_select (native_fd + 1, &rset, NULL, NULL, &tv)) < 0)
  345. goto error;
  346. else if (retval == 0)
  347. throw new ::java::net::SocketTimeoutException
  348. (JvNewStringUTF ("Receive timed out") );
  349. }
  350. retlen =
  351. ::recvfrom (native_fd, (char *) dbytes, maxlen, 0, (sockaddr*) &u,
  352. &addrlen);
  353. if (retlen < 0)
  354. goto error;
  355. // FIXME: Deal with Multicast addressing and if the socket is connected.
  356. jbyteArray raddr;
  357. jint rport;
  358. if (u.address.sin_family == AF_INET)
  359. {
  360. raddr = JvNewByteArray (4);
  361. memcpy (elements (raddr), &u.address.sin_addr, 4);
  362. rport = ntohs (u.address.sin_port);
  363. }
  364. #ifdef HAVE_INET6
  365. else if (u.address.sin_family == AF_INET6)
  366. {
  367. raddr = JvNewByteArray (16);
  368. memcpy (elements (raddr), &u.address6.sin6_addr, 16);
  369. rport = ntohs (u.address6.sin6_port);
  370. }
  371. #endif
  372. else
  373. throw new ::java::net::SocketException (JvNewStringUTF ("invalid family"));
  374. p->setAddress (::java::net::InetAddress::getByAddress (raddr));
  375. p->setPort (rport);
  376. p->length = (jint) retlen;
  377. return;
  378. error:
  379. char* strerr = strerror (errno);
  380. if (errno == ECONNREFUSED)
  381. throw new ::java::net::PortUnreachableException (JvNewStringUTF (strerr));
  382. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  383. }
  384. void
  385. gnu::java::net::PlainDatagramSocketImpl::setTimeToLive (jint ttl)
  386. {
  387. // Assumes IPPROTO_IP rather than IPPROTO_IPV6 since socket created is IPv4.
  388. char val = (char) ttl;
  389. socklen_t val_len = sizeof(val);
  390. if (::setsockopt (native_fd, IPPROTO_IP, IP_MULTICAST_TTL, &val, val_len) == 0)
  391. return;
  392. char* strerr = strerror (errno);
  393. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  394. }
  395. jint
  396. gnu::java::net::PlainDatagramSocketImpl::getTimeToLive ()
  397. {
  398. // Assumes IPPROTO_IP rather than IPPROTO_IPV6 since socket created is IPv4.
  399. char val;
  400. socklen_t val_len = sizeof(val);
  401. if (::getsockopt (native_fd, IPPROTO_IP, IP_MULTICAST_TTL, &val, &val_len) == 0)
  402. return ((int) val) & 0xFF;
  403. char* strerr = strerror (errno);
  404. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  405. }
  406. void
  407. gnu::java::net::PlainDatagramSocketImpl::mcastGrp (::java::net::InetAddress *inetaddr,
  408. ::java::net::NetworkInterface *,
  409. jboolean join)
  410. {
  411. // FIXME: implement use of NetworkInterface
  412. jbyteArray haddress = inetaddr->addr;
  413. #if HAVE_STRUCT_IP_MREQ || HAVE_STRUCT_IPV6_MREQ
  414. union McastReq u;
  415. jbyte *bytes = elements (haddress);
  416. #endif
  417. int len = haddress->length;
  418. int level, opname;
  419. const char *ptr;
  420. if (0)
  421. ;
  422. #if HAVE_STRUCT_IP_MREQ
  423. else if (len == 4)
  424. {
  425. level = IPPROTO_IP;
  426. opname = join ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP;
  427. memcpy (&u.mreq.imr_multiaddr, bytes, len);
  428. // FIXME: If a non-default interface is set, use it; see Stevens p. 501.
  429. // Maybe not, see note in last paragraph at bottom of Stevens p. 497.
  430. u.mreq.imr_interface.s_addr = htonl (INADDR_ANY);
  431. len = sizeof (struct ip_mreq);
  432. ptr = (const char *) &u.mreq;
  433. }
  434. #endif
  435. #if HAVE_STRUCT_IPV6_MREQ
  436. else if (len == 16)
  437. {
  438. level = IPPROTO_IPV6;
  439. /* Prefer new RFC 2553 names. */
  440. #ifndef IPV6_JOIN_GROUP
  441. #define IPV6_JOIN_GROUP IPV6_ADD_MEMBERSHIP
  442. #endif
  443. #ifndef IPV6_LEAVE_GROUP
  444. #define IPV6_LEAVE_GROUP IPV6_DROP_MEMBERSHIP
  445. #endif
  446. opname = join ? IPV6_JOIN_GROUP : IPV6_LEAVE_GROUP;
  447. memcpy (&u.mreq6.ipv6mr_multiaddr, bytes, len);
  448. // FIXME: If a non-default interface is set, use it; see Stevens p. 501.
  449. // Maybe not, see note in last paragraph at bottom of Stevens p. 497.
  450. u.mreq6.ipv6mr_interface = 0;
  451. len = sizeof (struct ipv6_mreq);
  452. ptr = (const char *) &u.mreq6;
  453. }
  454. #endif
  455. else
  456. throw new ::java::net::SocketException (JvNewStringUTF ("invalid length"));
  457. if (::setsockopt (native_fd, level, opname, ptr, len) == 0)
  458. return;
  459. char* strerr = strerror (errno);
  460. throw new ::java::io::IOException (JvNewStringUTF (strerr));
  461. }
  462. // Helper function to get the InetAddress for a given socket (file
  463. // descriptor).
  464. static ::java::net::InetAddress *
  465. getLocalAddress (int native_fd)
  466. {
  467. jbyteArray laddr;
  468. union SockAddr u;
  469. socklen_t addrlen = sizeof(u);
  470. if (::getsockname (native_fd, (sockaddr*) &u, &addrlen) != 0)
  471. {
  472. char* strerr = strerror (errno);
  473. throw new ::java::net::SocketException (JvNewStringUTF (strerr));
  474. }
  475. if (u.address.sin_family == AF_INET)
  476. {
  477. laddr = JvNewByteArray (4);
  478. memcpy (elements (laddr), &u.address.sin_addr, 4);
  479. }
  480. #ifdef HAVE_INET6
  481. else if (u.address.sin_family == AF_INET6)
  482. {
  483. laddr = JvNewByteArray (16);
  484. memcpy (elements (laddr), &u.address6.sin6_addr, 16);
  485. }
  486. #endif
  487. else
  488. throw new ::java::net::SocketException (JvNewStringUTF ("invalid family"));
  489. return ::java::net::InetAddress::getByAddress (laddr);
  490. }
  491. void
  492. gnu::java::net::PlainDatagramSocketImpl::setOption (jint optID,
  493. ::java::lang::Object *value)
  494. {
  495. int val;
  496. socklen_t val_len = sizeof (val);
  497. if (native_fd < 0)
  498. throw new ::java::net::SocketException (JvNewStringUTF ("Socket closed"));
  499. if (_Jv_IsInstanceOf (value, &::java::lang::Boolean::class$))
  500. {
  501. ::java::lang::Boolean *boolobj =
  502. static_cast< ::java::lang::Boolean *> (value);
  503. val = boolobj->booleanValue() ? 1 : 0;
  504. }
  505. else if (_Jv_IsInstanceOf (value, &::java::lang::Integer::class$))
  506. {
  507. ::java::lang::Integer *intobj =
  508. static_cast< ::java::lang::Integer *> (value);
  509. val = (int) intobj->intValue();
  510. }
  511. // Else assume value to be an InetAddress for use with IP_MULTICAST_IF.
  512. switch (optID)
  513. {
  514. case _Jv_TCP_NODELAY_ :
  515. throw new ::java::net::SocketException (
  516. JvNewStringUTF ("TCP_NODELAY not valid for UDP"));
  517. return;
  518. case _Jv_SO_LINGER_ :
  519. throw new ::java::net::SocketException (
  520. JvNewStringUTF ("SO_LINGER not valid for UDP"));
  521. return;
  522. case _Jv_SO_KEEPALIVE_ :
  523. throw new ::java::net::SocketException (
  524. JvNewStringUTF ("SO_KEEPALIVE not valid for UDP"));
  525. return;
  526. case _Jv_SO_BROADCAST_ :
  527. if (::setsockopt (native_fd, SOL_SOCKET, SO_BROADCAST, (char *) &val,
  528. val_len) != 0)
  529. goto error;
  530. return;
  531. case _Jv_SO_OOBINLINE_ :
  532. throw new ::java::net::SocketException (
  533. JvNewStringUTF ("SO_OOBINLINE: not valid for UDP"));
  534. return;
  535. case _Jv_SO_SNDBUF_ :
  536. case _Jv_SO_RCVBUF_ :
  537. #if defined(SO_SNDBUF) && defined(SO_RCVBUF)
  538. int opt;
  539. optID == _Jv_SO_SNDBUF_ ? opt = SO_SNDBUF : opt = SO_RCVBUF;
  540. if (::setsockopt (native_fd, SOL_SOCKET, opt, (char *) &val, val_len) != 0)
  541. goto error;
  542. #else
  543. throw new ::java::lang::InternalError (
  544. JvNewStringUTF ("SO_RCVBUF/SO_SNDBUF not supported"));
  545. #endif
  546. return;
  547. case _Jv_SO_REUSEADDR_ :
  548. #if defined(SO_REUSEADDR)
  549. if (::setsockopt (native_fd, SOL_SOCKET, SO_REUSEADDR, (char *) &val,
  550. val_len) != 0)
  551. goto error;
  552. #else
  553. throw new ::java::lang::InternalError (
  554. JvNewStringUTF ("SO_REUSEADDR not supported"));
  555. #endif
  556. return;
  557. case _Jv_SO_BINDADDR_ :
  558. throw new ::java::net::SocketException (
  559. JvNewStringUTF ("SO_BINDADDR: read only option"));
  560. return;
  561. case _Jv_IP_MULTICAST_IF_ :
  562. union InAddr u;
  563. jbyteArray haddress;
  564. jbyte *bytes;
  565. int len;
  566. int level, opname;
  567. const char *ptr;
  568. haddress = ((::java::net::InetAddress *) value)->addr;
  569. bytes = elements (haddress);
  570. len = haddress->length;
  571. if (len == 4)
  572. {
  573. level = IPPROTO_IP;
  574. opname = IP_MULTICAST_IF;
  575. memcpy (&u.addr, bytes, len);
  576. len = sizeof (struct in_addr);
  577. ptr = (const char *) &u.addr;
  578. }
  579. #ifdef HAVE_INET6
  580. else if (len == 16)
  581. {
  582. level = IPPROTO_IPV6;
  583. opname = IPV6_MULTICAST_IF;
  584. memcpy (&u.addr6, bytes, len);
  585. len = sizeof (struct in6_addr);
  586. ptr = (const char *) &u.addr6;
  587. }
  588. #endif
  589. else
  590. throw
  591. new ::java::net::SocketException (JvNewStringUTF ("invalid length"));
  592. if (::setsockopt (native_fd, level, opname, ptr, len) != 0)
  593. goto error;
  594. return;
  595. case _Jv_IP_MULTICAST_IF2_ :
  596. throw new ::java::net::SocketException (
  597. JvNewStringUTF ("IP_MULTICAST_IF2: not yet implemented"));
  598. return;
  599. case _Jv_IP_MULTICAST_LOOP_ :
  600. // cache the local address
  601. if (localAddress == NULL)
  602. localAddress = getLocalAddress (native_fd);
  603. len = localAddress->addr->length;
  604. if (len == 4)
  605. {
  606. level = IPPROTO_IP;
  607. opname = IP_MULTICAST_LOOP;
  608. }
  609. #if defined (HAVE_INET6) && defined (IPV6_MULTICAST_LOOP)
  610. else if (len == 16)
  611. {
  612. level = IPPROTO_IPV6;
  613. opname = IPV6_MULTICAST_LOOP;
  614. }
  615. #endif
  616. else
  617. throw
  618. new ::java::net::SocketException (JvNewStringUTF ("invalid address length"));
  619. if (::setsockopt (native_fd, level, opname, (char *) &val,
  620. val_len) != 0)
  621. goto error;
  622. return;
  623. case _Jv_IP_TOS_ :
  624. if (::setsockopt (native_fd, SOL_SOCKET, IP_TOS, (char *) &val,
  625. val_len) != 0)
  626. goto error;
  627. return;
  628. case _Jv_SO_TIMEOUT_ :
  629. timeout = val;
  630. return;
  631. default :
  632. errno = ENOPROTOOPT;
  633. }
  634. error:
  635. char* strerr = strerror (errno);
  636. throw new ::java::net::SocketException (JvNewStringUTF (strerr));
  637. }
  638. ::java::lang::Object *
  639. gnu::java::net::PlainDatagramSocketImpl::getOption (jint optID)
  640. {
  641. int val;
  642. socklen_t val_len = sizeof(val);
  643. int level, opname;
  644. switch (optID)
  645. {
  646. case _Jv_TCP_NODELAY_ :
  647. throw new ::java::net::SocketException (
  648. JvNewStringUTF ("TCP_NODELAY not valid for UDP"));
  649. break;
  650. case _Jv_SO_LINGER_ :
  651. throw new ::java::net::SocketException (
  652. JvNewStringUTF ("SO_LINGER not valid for UDP"));
  653. break;
  654. case _Jv_SO_KEEPALIVE_ :
  655. throw new ::java::net::SocketException (
  656. JvNewStringUTF ("SO_KEEPALIVE not valid for UDP"));
  657. break;
  658. case _Jv_SO_BROADCAST_ :
  659. if (::getsockopt (native_fd, SOL_SOCKET, SO_BROADCAST, (char *) &val,
  660. &val_len) != 0)
  661. goto error;
  662. return new ::java::lang::Boolean (val != 0);
  663. case _Jv_SO_OOBINLINE_ :
  664. throw new ::java::net::SocketException (
  665. JvNewStringUTF ("SO_OOBINLINE not valid for UDP"));
  666. break;
  667. case _Jv_SO_RCVBUF_ :
  668. case _Jv_SO_SNDBUF_ :
  669. #if defined(SO_SNDBUF) && defined(SO_RCVBUF)
  670. int opt;
  671. optID == _Jv_SO_SNDBUF_ ? opt = SO_SNDBUF : opt = SO_RCVBUF;
  672. if (::getsockopt (native_fd, SOL_SOCKET, opt, (char *) &val, &val_len) != 0)
  673. goto error;
  674. else
  675. return new ::java::lang::Integer (val);
  676. #else
  677. throw new ::java::lang::InternalError (
  678. JvNewStringUTF ("SO_RCVBUF/SO_SNDBUF not supported"));
  679. #endif
  680. break;
  681. case _Jv_SO_BINDADDR_:
  682. // cache the local address
  683. if (localAddress == NULL)
  684. localAddress = getLocalAddress (native_fd);
  685. return localAddress;
  686. break;
  687. case _Jv_SO_REUSEADDR_ :
  688. #if defined(SO_REUSEADDR)
  689. if (::getsockopt (native_fd, SOL_SOCKET, SO_REUSEADDR, (char *) &val,
  690. &val_len) != 0)
  691. goto error;
  692. return new ::java::lang::Boolean (val != 0);
  693. #else
  694. throw new ::java::lang::InternalError (
  695. JvNewStringUTF ("SO_REUSEADDR not supported"));
  696. #endif
  697. break;
  698. case _Jv_IP_MULTICAST_IF_ :
  699. #ifdef HAVE_INET_NTOA
  700. struct in_addr inaddr;
  701. socklen_t inaddr_len;
  702. char *bytes;
  703. inaddr_len = sizeof(inaddr);
  704. if (::getsockopt (native_fd, IPPROTO_IP, IP_MULTICAST_IF, (char *) &inaddr,
  705. &inaddr_len) != 0)
  706. goto error;
  707. bytes = inet_ntoa (inaddr);
  708. return ::java::net::InetAddress::getByName (JvNewStringLatin1 (bytes));
  709. #else
  710. throw new ::java::net::SocketException (
  711. JvNewStringUTF ("IP_MULTICAST_IF: not available - no inet_ntoa()"));
  712. #endif
  713. break;
  714. case _Jv_SO_TIMEOUT_ :
  715. return new ::java::lang::Integer (timeout);
  716. break;
  717. case _Jv_IP_MULTICAST_IF2_ :
  718. throw new ::java::net::SocketException (
  719. JvNewStringUTF ("IP_MULTICAST_IF2: not yet implemented"));
  720. break;
  721. case _Jv_IP_MULTICAST_LOOP_ :
  722. // cache the local address
  723. localAddress = getLocalAddress (native_fd);
  724. if (localAddress->addr->length == 4)
  725. {
  726. level = IPPROTO_IP;
  727. opname = IP_MULTICAST_LOOP;
  728. }
  729. #if defined (HAVE_INET6) && defined (IPV6_MULTICAST_LOOP)
  730. else if (localAddress->addr->length == 16)
  731. {
  732. level = IPPROTO_IPV6;
  733. opname = IPV6_MULTICAST_LOOP;
  734. }
  735. #endif
  736. else
  737. throw
  738. new ::java::net::SocketException (JvNewStringUTF ("invalid address length"));
  739. if (::getsockopt (native_fd, level, opname, (char *) &val,
  740. &val_len) != 0)
  741. goto error;
  742. return new ::java::lang::Boolean (val != 0);
  743. case _Jv_IP_TOS_ :
  744. if (::getsockopt (native_fd, SOL_SOCKET, IP_TOS, (char *) &val,
  745. &val_len) != 0)
  746. goto error;
  747. return new ::java::lang::Integer (val);
  748. default :
  749. errno = ENOPROTOOPT;
  750. }
  751. error:
  752. char* strerr = strerror (errno);
  753. throw new ::java::net::SocketException (JvNewStringUTF (strerr));
  754. }