cpnet.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819
  1. /* cpnet.c -
  2. Copyright (C) 2003, 2004, 2005, 2006 Free Software Foundation, Inc.
  3. This file is part of GNU Classpath.
  4. GNU Classpath is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation; either version 2, or (at your option)
  7. any later version.
  8. GNU Classpath is distributed in the hope that it will be useful, but
  9. WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with GNU Classpath; see the file COPYING. If not, write to the
  14. Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  15. 02110-1301 USA.
  16. Linking this library statically or dynamically with other modules is
  17. making a combined work based on this library. Thus, the terms and
  18. conditions of the GNU General Public License cover the whole
  19. combination.
  20. As a special exception, the copyright holders of this library give you
  21. permission to link this library with independent modules to produce an
  22. executable, regardless of the license terms of these independent
  23. modules, and to copy and distribute the resulting executable under
  24. terms of your choice, provided that you also meet, for each linked
  25. independent module, the terms and conditions of the license of that
  26. module. An independent module is a module which is not derived from
  27. or based on this library. If you modify this library, you may extend
  28. this exception to your version of the library, but you are not
  29. obligated to do so. If you do not wish to do so, delete this
  30. exception statement from your version. */
  31. #include "config.h"
  32. #include <jni.h>
  33. #include <assert.h>
  34. #include <sys/socket.h>
  35. #include <sys/types.h>
  36. #include <unistd.h>
  37. #include <errno.h>
  38. #include <fcntl.h>
  39. #include <netinet/in.h>
  40. #include <netinet/tcp.h>
  41. #include <netdb.h>
  42. #include <sys/time.h>
  43. #include <unistd.h>
  44. #include <arpa/inet.h>
  45. #if defined(HAVE_SYS_IOCTL_H)
  46. #define BSD_COMP /* Get FIONREAD on Solaris2 */
  47. #include <sys/ioctl.h>
  48. #endif
  49. #if defined(HAVE_SYS_FILIO_H) /* Get FIONREAD on Solaris 2.5 */
  50. #include <sys/filio.h>
  51. #endif
  52. #include "cpnet.h"
  53. #define SOCKET_DEFAULT_TIMEOUT -1 /* milliseconds */
  54. static int socketTimeouts[FD_SETSIZE];
  55. static jint waitForWritable(jint fd)
  56. {
  57. struct timeval tv;
  58. fd_set writeset;
  59. int ret;
  60. FD_ZERO(&writeset);
  61. FD_SET(fd, &writeset);
  62. if (socketTimeouts[fd] > 0)
  63. {
  64. tv.tv_sec = socketTimeouts[fd] / 1000;
  65. tv.tv_usec = (socketTimeouts[fd] % 1000) * 1000;
  66. ret = select(fd+1, NULL, &writeset, NULL, &tv);
  67. }
  68. else
  69. ret = select(fd+1, NULL, &writeset, NULL, NULL);
  70. return (ret <= 0) ? -1 : 0;
  71. }
  72. static jint waitForReadable(jint fd)
  73. {
  74. struct timeval tv;
  75. fd_set readset;
  76. int ret;
  77. FD_ZERO(&readset);
  78. FD_SET(fd, &readset);
  79. if (socketTimeouts[fd] > 0)
  80. {
  81. tv.tv_sec = socketTimeouts[fd] / 1000;
  82. tv.tv_usec = (socketTimeouts[fd] % 1000) * 1000;
  83. ret = select(fd+1, &readset, NULL, NULL, &tv);
  84. }
  85. else
  86. ret = select(fd+1, &readset, NULL, NULL, NULL);
  87. return (ret <= 0) ? -1 : 0;
  88. }
  89. jint cpnet_openSocketStream(JNIEnv *env UNUSED, jint *fd, jint family)
  90. {
  91. *fd = socket(family, SOCK_STREAM, 0);
  92. if (*fd == -1)
  93. return errno;
  94. fcntl(*fd, F_SETFD, FD_CLOEXEC);
  95. assert(*fd < FD_SETSIZE);
  96. socketTimeouts[*fd] = SOCKET_DEFAULT_TIMEOUT;
  97. return 0;
  98. }
  99. jint cpnet_openSocketDatagram(JNIEnv *env UNUSED, jint *fd, jint family)
  100. {
  101. *fd = socket(family, SOCK_DGRAM, 0);
  102. if (*fd == -1)
  103. return errno;
  104. fcntl(*fd, F_SETFD, FD_CLOEXEC);
  105. assert(*fd < FD_SETSIZE);
  106. socketTimeouts[*fd] = SOCKET_DEFAULT_TIMEOUT;
  107. return 0;
  108. }
  109. jint cpnet_shutdown (JNIEnv *env UNUSED, jint fd, jbyte flag)
  110. {
  111. int ret;
  112. int shut_flag = 0;
  113. if (flag == CPNET_SHUTDOWN_READ)
  114. shut_flag = SHUT_RD;
  115. else if (flag == CPNET_SHUTDOWN_WRITE)
  116. shut_flag = SHUT_WR;
  117. ret = shutdown (fd, shut_flag);
  118. if (ret != 0)
  119. return errno;
  120. return 0;
  121. }
  122. jint cpnet_close(JNIEnv *env UNUSED, jint fd)
  123. {
  124. if (close (fd) != 0)
  125. return errno;
  126. return 0;
  127. }
  128. jint cpnet_listen(JNIEnv *env UNUSED, jint fd, jint queuelen)
  129. {
  130. if (listen (fd, queuelen) != 0)
  131. return errno;
  132. return 0;
  133. }
  134. jint cpnet_accept(JNIEnv *env UNUSED, jint fd, jint *newfd)
  135. {
  136. if (waitForReadable (fd) < 0)
  137. return ETIMEDOUT;
  138. *newfd = accept(fd, NULL, 0);
  139. if (*newfd != 0)
  140. return errno;
  141. return 0;
  142. }
  143. jint cpnet_bind(JNIEnv *env UNUSED, jint fd, cpnet_address *addr)
  144. {
  145. int ret;
  146. ret = bind(fd, (struct sockaddr *)addr->data, addr->len);
  147. if (ret != 0)
  148. return errno;
  149. return 0;
  150. }
  151. jint cpnet_connect(JNIEnv *env UNUSED, jint fd, cpnet_address *addr)
  152. {
  153. int ret;
  154. /* TODO: implement socket time out */
  155. ret = connect(fd, (struct sockaddr *)addr->data, addr->len);
  156. if (ret != 0)
  157. return errno;
  158. return 0;
  159. }
  160. jint cpnet_getLocalAddr(JNIEnv *env, jint fd, cpnet_address **addr)
  161. {
  162. socklen_t slen = 1024;
  163. int ret;
  164. *addr = JCL_malloc(env, slen);
  165. slen -= sizeof(jint);
  166. ret = getsockname(fd, (struct sockaddr *)(*addr)->data, &slen );
  167. if (ret != 0)
  168. {
  169. int err = errno;
  170. JCL_free(env, *addr);
  171. return err;
  172. }
  173. (*addr)->len = slen;
  174. return 0;
  175. }
  176. jint cpnet_getRemoteAddr(JNIEnv *env, jint fd, cpnet_address **addr)
  177. {
  178. socklen_t slen = 1024;
  179. int ret;
  180. *addr = JCL_malloc(env, slen);
  181. slen -= sizeof(jint);
  182. ret = getpeername(fd, (struct sockaddr *)(*addr)->data, &slen );
  183. if (ret != 0)
  184. {
  185. int err = errno;
  186. JCL_free(env, *addr);
  187. return err;
  188. }
  189. (*addr)->len = slen;
  190. return 0;
  191. }
  192. jint cpnet_setBroadcast(JNIEnv *env UNUSED, jint fd, jint flag)
  193. {
  194. int ret;
  195. ret = setsockopt(fd, SOL_SOCKET, SO_BROADCAST, &flag, sizeof(flag));
  196. if (ret != 0)
  197. return errno;
  198. return 0;
  199. }
  200. #if defined (HAVE_MSG_NOSIGNAL)
  201. #elif defined (HAVE_SO_NOSIGPIPE)
  202. static int setsockopt_NOSIGPIPE (int fd)
  203. {
  204. int setToTrue = 1;
  205. return setsockopt(fd, SOL_SOCKET, SO_NOSIGPIPE, &setToTrue, sizeof(setToTrue));
  206. }
  207. #endif
  208. jint cpnet_send (JNIEnv *env UNUSED, jint fd, jbyte *data, jint len, jint *bytes_sent)
  209. {
  210. ssize_t ret;
  211. if (waitForWritable(fd) < 0)
  212. return ETIMEDOUT;
  213. #if defined (HAVE_MSG_NOSIGNAL)
  214. ret = send(fd, data, len, MSG_NOSIGNAL);
  215. #elif defined (HAVE_SO_NOSIGPIPE)
  216. ret = setsockopt_NOSIGPIPE(fd);
  217. if (ret == 0) ret = send(fd, data, len, 0);
  218. #else
  219. /* We want SIGPIPE to be omitted. But this configuration does not have an
  220. * option for that.
  221. */
  222. ret = send(fd, data, len, 0);
  223. #endif
  224. if (ret < 0)
  225. return errno;
  226. *bytes_sent = ret;
  227. return 0;
  228. }
  229. jint cpnet_sendTo (JNIEnv *env UNUSED, jint fd, jbyte *data, jint len, cpnet_address *addr, jint *bytes_sent)
  230. {
  231. ssize_t ret;
  232. if (waitForWritable(fd) < 0)
  233. return ETIMEDOUT;
  234. #if defined (HAVE_MSG_NOSIGNAL)
  235. ret = sendto(fd, data, len, MSG_NOSIGNAL, (struct sockaddr *)addr->data,
  236. addr->len);
  237. #elif defined (HAVE_SO_NOSIGPIPE)
  238. ret = setsockopt_NOSIGPIPE(fd);
  239. if (ret == 0)
  240. {
  241. ret = sendto(fd, data, len, 0, (struct sockaddr *)addr->data,
  242. addr->len);
  243. }
  244. #else
  245. /* We want SIGPIPE to be omitted. But this configuration does not have an
  246. * option for that.
  247. */
  248. ret = sendto(fd, data, len, 0, (struct sockaddr *)addr->data,
  249. addr->len);
  250. #endif
  251. if (ret < 0)
  252. return errno;
  253. *bytes_sent = ret;
  254. return 0;
  255. }
  256. jint cpnet_recv (JNIEnv *env UNUSED, jint fd, jbyte *data, jint len, jint *bytes_recv)
  257. {
  258. ssize_t ret;
  259. if (waitForReadable(fd) < 0)
  260. return ETIMEDOUT;
  261. ret = recv(fd, data, len, 0);
  262. if (ret < 0)
  263. return errno;
  264. *bytes_recv = ret;
  265. return 0;
  266. }
  267. jint cpnet_recvFrom (JNIEnv *env, jint fd, jbyte *data, jint len, cpnet_address **addr, jint *bytes_recv)
  268. {
  269. socklen_t slen = 1024;
  270. ssize_t ret;
  271. if (waitForReadable(fd) < 0)
  272. return ETIMEDOUT;
  273. *addr = JCL_malloc(env, slen);
  274. slen -= sizeof(jint);
  275. ret = recvfrom(fd, data, len, 0, (struct sockaddr *) (*addr)->data, &slen);
  276. if (ret < 0)
  277. {
  278. int err = errno;
  279. JCL_free(env, *addr);
  280. return err;
  281. }
  282. (*addr)->len = slen;
  283. *bytes_recv = ret;
  284. return 0;
  285. }
  286. jint cpnet_setSocketTCPNoDelay (JNIEnv *env UNUSED, jint fd, jint nodelay)
  287. {
  288. socklen_t len = sizeof(jint);
  289. int ret;
  290. ret = setsockopt(fd, IPPROTO_TCP, TCP_NODELAY, &nodelay, len);
  291. if (ret < 0)
  292. return errno;
  293. return 0;
  294. }
  295. jint cpnet_getSocketTCPNoDelay (JNIEnv *env UNUSED, jint fd, jint *nodelay)
  296. {
  297. socklen_t len = sizeof(jint);
  298. int ret;
  299. ret = getsockopt(fd, IPPROTO_TCP, TCP_NODELAY, nodelay, &len);
  300. if (ret < 0)
  301. return errno;
  302. return 0;
  303. }
  304. jint cpnet_setLinger (JNIEnv *env UNUSED, jint fd, jint flag, jint value)
  305. {
  306. socklen_t len = sizeof(struct linger);
  307. int ret;
  308. struct linger __linger;
  309. if (flag)
  310. {
  311. __linger.l_onoff = 0;
  312. }
  313. else
  314. {
  315. __linger.l_linger = value;
  316. __linger.l_onoff = 1;
  317. }
  318. ret = setsockopt(fd, SOL_SOCKET, SO_LINGER, &__linger, len);
  319. if (ret < 0)
  320. return errno;
  321. return 0;
  322. }
  323. jint cpnet_getLinger (JNIEnv *env UNUSED, jint fd, jint *flag, jint *value)
  324. {
  325. socklen_t slen = sizeof(struct linger);
  326. struct linger __linger;
  327. int ret;
  328. ret = getsockopt(fd, SOL_SOCKET, SO_LINGER, &__linger, &slen);
  329. if (ret != 0)
  330. return errno;
  331. *flag = __linger.l_onoff;
  332. *value = __linger.l_linger;
  333. return ret;
  334. }
  335. jint cpnet_setSocketTimeout (JNIEnv *env UNUSED, jint fd, jint value)
  336. {
  337. socketTimeouts[fd] = value;
  338. return 0;
  339. }
  340. jint cpnet_getSocketTimeout (JNIEnv *env UNUSED, jint fd, jint *value)
  341. {
  342. *value = socketTimeouts[fd];
  343. return 0;
  344. }
  345. jint cpnet_setSendBuf (JNIEnv *env UNUSED, jint fd, jint value)
  346. {
  347. int ret;
  348. ret = setsockopt(fd, SOL_SOCKET, SO_SNDBUF, &value, sizeof(value));
  349. if (ret != 0)
  350. return errno;
  351. return 0;
  352. }
  353. jint cpnet_getSendBuf (JNIEnv *env UNUSED, jint fd, jint *value)
  354. {
  355. int ret;
  356. socklen_t slen = sizeof(*value);
  357. ret = getsockopt(fd, SOL_SOCKET, SO_SNDBUF, value, &slen);
  358. if (ret != 0)
  359. return errno;
  360. return 0;
  361. }
  362. jint cpnet_setRecvBuf (JNIEnv *env UNUSED, jint fd, jint value)
  363. {
  364. int ret;
  365. ret = setsockopt(fd, SOL_SOCKET, SO_RCVBUF, &value, sizeof(value));
  366. if (ret != 0)
  367. return errno;
  368. return 0;
  369. }
  370. jint cpnet_getRecvBuf (JNIEnv *env UNUSED, jint fd, jint *value)
  371. {
  372. int ret;
  373. socklen_t slen = sizeof(*value);
  374. ret = getsockopt(fd, SOL_SOCKET, SO_RCVBUF, value, &slen);
  375. if (ret != 0)
  376. return errno;
  377. return 0;
  378. }
  379. jint cpnet_setTTL (JNIEnv *env UNUSED, jint fd, jint value)
  380. {
  381. int ret;
  382. ret = setsockopt(fd, IPPROTO_IP, IP_TTL, &value, sizeof(value));
  383. if (ret != 0)
  384. return errno;
  385. return 0;
  386. }
  387. jint cpnet_getTTL (JNIEnv *env UNUSED, jint fd, jint *value)
  388. {
  389. int ret;
  390. socklen_t slen = sizeof(*value);
  391. ret = getsockopt(fd, IPPROTO_IP, IP_TTL, value, &slen);
  392. if (ret != 0)
  393. return errno;
  394. return 0;
  395. }
  396. jint cpnet_setMulticastIF (JNIEnv *env UNUSED, jint fd, cpnet_address *addr)
  397. {
  398. int ret;
  399. ret = setsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, (struct sockaddr *)addr->data, addr->len);
  400. if (ret != 0)
  401. return errno;
  402. return 0;
  403. }
  404. jint cpnet_getMulticastIF (JNIEnv *env, jint fd, cpnet_address **addr)
  405. {
  406. socklen_t slen = 1024;
  407. int ret;
  408. *addr = JCL_malloc(env, slen);
  409. slen -= sizeof(jint);
  410. ret = getsockopt(fd, IPPROTO_IP, IP_MULTICAST_IF, (struct sockaddr *)(*addr)->data, &slen);
  411. (*addr)->len = slen;
  412. if (ret != 0)
  413. return errno;
  414. return 0;
  415. }
  416. jint cpnet_setReuseAddress (JNIEnv *env UNUSED, jint fd, jint reuse)
  417. {
  418. int ret;
  419. ret = setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse));
  420. if (ret != 0)
  421. return errno;
  422. return 0;
  423. }
  424. jint cpnet_getReuseAddress (JNIEnv *env UNUSED, jint fd, jint *reuse)
  425. {
  426. int ret;
  427. socklen_t slen = sizeof(*reuse);
  428. ret = getsockopt(fd, SOL_SOCKET, SO_REUSEADDR, reuse, &slen);
  429. if (ret != 0)
  430. return errno;
  431. return 0;
  432. }
  433. jint cpnet_setKeepAlive (JNIEnv *env UNUSED, jint fd, jint keep)
  434. {
  435. int ret;
  436. ret = setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, &keep, sizeof(keep));
  437. if (ret != 0)
  438. return errno;
  439. return 0;
  440. }
  441. jint cpnet_getKeepAlive (JNIEnv *env UNUSED, jint fd, jint *keep)
  442. {
  443. int ret;
  444. socklen_t slen = sizeof(*keep);
  445. ret = getsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, keep, &slen);
  446. if (ret != 0)
  447. return errno;
  448. return 0;
  449. }
  450. jint cpnet_addMembership (JNIEnv *env UNUSED, jint fd, cpnet_address *addr)
  451. {
  452. struct ip_mreq req;
  453. int ret;
  454. struct sockaddr_in *sockaddr = (struct sockaddr_in *)addr->data;
  455. memset(&req, 0, sizeof(req));
  456. req.imr_multiaddr = sockaddr->sin_addr;
  457. req.imr_interface.s_addr = INADDR_ANY;
  458. ret = setsockopt(fd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &req, sizeof(req));
  459. if (ret != 0)
  460. return errno;
  461. return 0;
  462. }
  463. jint cpnet_dropMembership (JNIEnv *env UNUSED, jint fd, cpnet_address *addr)
  464. {
  465. struct ip_mreq req;
  466. int ret;
  467. struct sockaddr_in *sockaddr = (struct sockaddr_in *)addr->data;
  468. memset(&req, 0, sizeof(req));
  469. req.imr_multiaddr = sockaddr->sin_addr;
  470. req.imr_interface.s_addr = INADDR_ANY;
  471. ret = setsockopt(fd, IPPROTO_IP, IP_DROP_MEMBERSHIP, &req, sizeof(req));
  472. if (ret != 0)
  473. return errno;
  474. return 0;
  475. }
  476. jint cpnet_getAvailableBytes (JNIEnv *env UNUSED, jint fd, jint *availableBytes)
  477. {
  478. int ret;
  479. ret = ioctl(fd, FIONREAD, availableBytes);
  480. if (ret != 0)
  481. return errno;
  482. return 0;
  483. }
  484. jint cpnet_getHostname (JNIEnv *env UNUSED, char *hostname, jint hostname_len)
  485. {
  486. int ret;
  487. ret = gethostname(hostname, hostname_len);
  488. if (ret != 0)
  489. return errno;
  490. hostname[hostname_len-1] = 0;
  491. return 0;
  492. }
  493. jint cpnet_getHostByName (JNIEnv *env, const char *hostname, cpnet_address ***addresses, jint *addresses_count)
  494. {
  495. struct hostent hret;
  496. struct hostent *result;
  497. jint buflen = 1024;
  498. int herr = 0;
  499. int ret = 0;
  500. int counter = 0;
  501. cpnet_address **addr_arr;
  502. int i;
  503. char *buf;
  504. do
  505. {
  506. buf = (char *)JCL_malloc(env, buflen);
  507. #ifdef HAVE_GETHOSTBYNAME_R
  508. # if defined(HAVE_FUNC_GETHOSTBYNAME_R_6)
  509. ret = gethostbyname_r (hostname, &hret, buf, buflen, &result, &herr);
  510. # elif defined(HAVE_FUNC_GETHOSTBYNAME_R_5)
  511. result = gethostbyname_r(hostname, &hret, buf, buflen, &herr);
  512. # elif defined(HAVE_FUNC_GETHOSTBYNAME_R_3)
  513. # error IMPLEMENT ME!
  514. # else
  515. # error unknown number of arguments for gethostbyname_r
  516. # endif
  517. #else
  518. hret.h_addr_list = NULL;
  519. hret.h_addrtype = 0;
  520. result = gethostbyname (hostname);
  521. if (result == NULL)
  522. return -errno;
  523. memcpy (&hret, result, sizeof (struct hostent));
  524. #endif
  525. if (ret != 0 || result == NULL)
  526. {
  527. if (herr == ERANGE)
  528. {
  529. buflen *= 2;
  530. JCL_free(env, buf);
  531. continue;
  532. }
  533. JCL_free(env, buf);
  534. return -herr;
  535. }
  536. break;
  537. }
  538. while (1);
  539. while (hret.h_addr_list[counter] != NULL)
  540. counter++;
  541. *addresses_count = counter;
  542. addr_arr = *addresses = JCL_malloc(env, sizeof(cpnet_address *) * counter);
  543. switch (hret.h_addrtype)
  544. {
  545. case AF_INET:
  546. for (i = 0; i < counter; i++)
  547. {
  548. addr_arr[i] = cpnet_newIPV4Address(env);
  549. cpnet_bytesToIPV4Address(addr_arr[i], (jbyte *)hret.h_addr_list[i]);
  550. }
  551. break;
  552. #ifdef HAVE_INET6
  553. case AF_INET6:
  554. for (i = 0; i < counter; i++)
  555. {
  556. addr_arr[i] = cpnet_newIPV6Address(env);
  557. cpnet_bytesToIPV6Address(addr_arr[i], (jbyte *)hret.h_addr_list[i]);
  558. }
  559. break;
  560. #endif
  561. default:
  562. *addresses_count = 0;
  563. JCL_free(env, addr_arr);
  564. break;
  565. }
  566. JCL_free(env, buf);
  567. return 0;
  568. }
  569. jint cpnet_getHostByAddr (JNIEnv *env UNUSED, cpnet_address *addr, char *hostname, jint hostname_len)
  570. {
  571. union
  572. {
  573. struct sockaddr_in *addr_v4;
  574. struct sockaddr_in6 *addr_v6;
  575. char *data;
  576. } haddr;
  577. void *raw_addr;
  578. int addr_type;
  579. struct hostent *ret;
  580. int addr_len;
  581. haddr.data = addr->data;
  582. if (haddr.addr_v4->sin_family == AF_INET)
  583. {
  584. raw_addr = &haddr.addr_v4->sin_addr;
  585. addr_len = sizeof(haddr.addr_v4->sin_addr);
  586. addr_type = AF_INET;
  587. }
  588. #ifdef HAVE_INET6
  589. else if (haddr.addr_v6->sin6_family == AF_INET6)
  590. {
  591. raw_addr = &haddr.addr_v6->sin6_addr;
  592. addr_type = AF_INET6;
  593. addr_len = sizeof(haddr.addr_v6->sin6_addr);
  594. }
  595. #endif
  596. else
  597. return EINVAL;
  598. /* Here we do not have any thread safe call. VM implementors will have to
  599. * do a big lock. Or it should be put on the Classpath VM interface.
  600. */
  601. ret = gethostbyaddr(raw_addr, addr_len, addr_type);
  602. if (ret == NULL)
  603. {
  604. /* The trouble here is how to distinguish the two cases ? */
  605. if (h_errno != 0)
  606. return h_errno;
  607. else
  608. return errno;
  609. }
  610. strncpy(hostname, ret->h_name, hostname_len);
  611. return 0;
  612. }
  613. jint cpnet_aton (JNIEnv *env, const char *hostname, cpnet_address **addr)
  614. {
  615. jbyte *bytes = NULL;
  616. #if defined(HAVE_INET_PTON) && defined(HAVE_INET6)
  617. jbyte inet6_addr[16];
  618. #endif
  619. #ifdef HAVE_INET_ATON
  620. struct in_addr laddr;
  621. if (inet_aton (hostname, &laddr))
  622. {
  623. bytes = (jbyte *) &laddr;
  624. }
  625. #elif defined(HAVE_INET_ADDR)
  626. #if ! HAVE_IN_ADDR_T
  627. typedef jint in_addr_t;
  628. #endif
  629. in_addr_t laddr = inet_addr (hostname);
  630. if (laddr != (in_addr_t)(-1))
  631. {
  632. bytes = (jbyte *) &laddr;
  633. }
  634. #endif
  635. if (bytes)
  636. {
  637. *addr = cpnet_newIPV4Address(env);
  638. cpnet_bytesToIPV4Address(*addr, bytes);
  639. return 0;
  640. }
  641. #if defined(HAVE_INET_PTON) && defined(HAVE_INET6)
  642. if (inet_pton (AF_INET6, hostname, inet6_addr) > 0)
  643. {
  644. *addr = cpnet_newIPV6Address(env);
  645. cpnet_bytesToIPV6Address(*addr, inet6_addr);
  646. return 0;
  647. }
  648. #endif
  649. *addr = NULL;
  650. return 0;
  651. }
  652. void cpnet_freeAddresses(JNIEnv * env, cpnet_address **addr, jint addresses_count)
  653. {
  654. jint i;
  655. for (i = 0; i < addresses_count; i++)
  656. cpnet_freeAddress(env, addr[i]);
  657. }