1 /*
2 * Copyright (c) 2001, 2017, Oracle and/or its affiliates. All rights reserved.
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4 *
5 * This code is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License version 2 only, as
7 * published by the Free Software Foundation. Oracle designates this
8 * particular file as subject to the "Classpath" exception as provided
9 * by Oracle in the LICENSE file that accompanied this code.
10 *
11 * This code is distributed in the hope that it will be useful, but WITHOUT
12 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 * version 2 for more details (a copy is included in the LICENSE file that
15 * accompanied this code).
16 *
17 * You should have received a copy of the GNU General Public License version
18 * 2 along with this work; if not, write to the Free Software Foundation,
19 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20 *
21 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22 * or visit www.oracle.com if you need additional information or have any
23 * questions.
24 */
25
26 #include <windows.h>
27 #include <winsock2.h>
28 #include "jni.h"
29 #include "jni_util.h"
30 #include "jvm.h"
31 #include "jlong.h"
32
33 #include "nio.h"
34 #include "nio_util.h"
35 #include "net_util.h"
36
37 #include "sun_nio_ch_Net.h"
38 #include "sun_nio_ch_PollArrayWrapper.h"
39
40 /**
41 * Definitions to allow for building with older SDK include files.
42 */
43
44 #ifndef MCAST_BLOCK_SOURCE
45
46 #define MCAST_BLOCK_SOURCE 43
47 #define MCAST_UNBLOCK_SOURCE 44
48 #define MCAST_JOIN_SOURCE_GROUP 45
49 #define MCAST_LEAVE_SOURCE_GROUP 46
50
51 #endif /* MCAST_BLOCK_SOURCE */
52
53 struct my_ip_mreq_source {
54 IN_ADDR imr_multiaddr;
55 IN_ADDR imr_sourceaddr;
56 IN_ADDR imr_interface;
57 };
58
59 struct my_group_source_req {
60 ULONG gsr_interface;
61 SOCKADDR_STORAGE gsr_group;
62 SOCKADDR_STORAGE gsr_source;
63 };
64
65 /**
66 * Copy IPv6 address as jbytearray to target
67 */
68 #define COPY_INET6_ADDRESS(env, source, target) \
69 (*env)->GetByteArrayRegion(env, source, 0, 16, target)
70
71 /**
72 * Enable or disable receipt of WSAECONNRESET errors.
73 */
setConnectionReset(SOCKET s,BOOL enable)74 static void setConnectionReset(SOCKET s, BOOL enable) {
75 DWORD bytesReturned = 0;
76 WSAIoctl(s, SIO_UDP_CONNRESET, &enable, sizeof(enable),
77 NULL, 0, &bytesReturned, NULL, NULL);
78 }
79
80
81 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_initIDs(JNIEnv * env,jclass clazz)82 Java_sun_nio_ch_Net_initIDs(JNIEnv *env, jclass clazz)
83 {
84 initInetAddressIDs(env);
85 }
86
87 JNIEXPORT jboolean JNICALL
Java_sun_nio_ch_Net_isIPv6Available0(JNIEnv * env,jclass cl)88 Java_sun_nio_ch_Net_isIPv6Available0(JNIEnv* env, jclass cl)
89 {
90 /*
91 * Return true if IPv6 is configured
92 */
93 return ipv6_available() ? JNI_TRUE : JNI_FALSE;
94 }
95
96 JNIEXPORT jboolean JNICALL
Java_sun_nio_ch_Net_isReusePortAvailable0(JNIEnv * env,jclass c1)97 Java_sun_nio_ch_Net_isReusePortAvailable0(JNIEnv* env, jclass c1)
98 {
99 // SO_REUSEPORT is not supported on Windows
100 return JNI_FALSE;
101 }
102
103 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_isExclusiveBindAvailable(JNIEnv * env,jclass clazz)104 Java_sun_nio_ch_Net_isExclusiveBindAvailable(JNIEnv *env, jclass clazz) {
105 return 1;
106 }
107
108
109 JNIEXPORT jboolean JNICALL
Java_sun_nio_ch_Net_canIPv6SocketJoinIPv4Group0(JNIEnv * env,jclass cl)110 Java_sun_nio_ch_Net_canIPv6SocketJoinIPv4Group0(JNIEnv* env, jclass cl)
111 {
112 return JNI_FALSE;
113 }
114
115 JNIEXPORT jboolean JNICALL
Java_sun_nio_ch_Net_canJoin6WithIPv4Group0(JNIEnv * env,jclass cl)116 Java_sun_nio_ch_Net_canJoin6WithIPv4Group0(JNIEnv* env, jclass cl)
117 {
118 return JNI_FALSE;
119 }
120
121 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_socket0(JNIEnv * env,jclass cl,jboolean preferIPv6,jboolean stream,jboolean reuse,jboolean fastLoopback)122 Java_sun_nio_ch_Net_socket0(JNIEnv *env, jclass cl, jboolean preferIPv6,
123 jboolean stream, jboolean reuse, jboolean fastLoopback)
124 {
125 SOCKET s;
126 int domain = (preferIPv6) ? AF_INET6 : AF_INET;
127
128 s = socket(domain, (stream ? SOCK_STREAM : SOCK_DGRAM), 0);
129 if (s != INVALID_SOCKET) {
130 SetHandleInformation((HANDLE)s, HANDLE_FLAG_INHERIT, 0);
131
132 /* IPV6_V6ONLY is true by default */
133 if (domain == AF_INET6) {
134 int opt = 0;
135 setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY,
136 (const char *)&opt, sizeof(opt));
137 }
138
139 /* Disable WSAECONNRESET errors for initially unconnected UDP sockets */
140 if (!stream) {
141 setConnectionReset(s, FALSE);
142 }
143
144 } else {
145 NET_ThrowNew(env, WSAGetLastError(), "socket");
146 }
147
148 if (stream && fastLoopback) {
149 static int loopback_available = 1;
150 if (loopback_available) {
151 int rv = NET_EnableFastTcpLoopback((jint)s);
152 if (rv) {
153 if (rv == WSAEOPNOTSUPP) {
154 loopback_available = 0;
155 } else {
156 NET_ThrowNew(env, rv, "fastLoopback");
157 }
158 }
159 }
160 }
161
162 return (jint)s;
163 }
164
165 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_bind0(JNIEnv * env,jclass clazz,jobject fdo,jboolean preferIPv6,jboolean isExclBind,jobject iao,jint port)166 Java_sun_nio_ch_Net_bind0(JNIEnv *env, jclass clazz, jobject fdo, jboolean preferIPv6,
167 jboolean isExclBind, jobject iao, jint port)
168 {
169 SOCKETADDRESS sa;
170 int rv;
171 int sa_len = 0;
172
173 if (NET_InetAddressToSockaddr(env, iao, port, &sa, &sa_len, preferIPv6) != 0) {
174 return;
175 }
176
177 rv = NET_WinBind(fdval(env, fdo), &sa, sa_len, isExclBind);
178 if (rv == SOCKET_ERROR)
179 NET_ThrowNew(env, WSAGetLastError(), "bind");
180 }
181
182 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_listen(JNIEnv * env,jclass cl,jobject fdo,jint backlog)183 Java_sun_nio_ch_Net_listen(JNIEnv *env, jclass cl, jobject fdo, jint backlog)
184 {
185 if (listen(fdval(env,fdo), backlog) == SOCKET_ERROR) {
186 NET_ThrowNew(env, WSAGetLastError(), "listen");
187 }
188 }
189
190
191 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_connect0(JNIEnv * env,jclass clazz,jboolean preferIPv6,jobject fdo,jobject iao,jint port)192 Java_sun_nio_ch_Net_connect0(JNIEnv *env, jclass clazz, jboolean preferIPv6, jobject fdo,
193 jobject iao, jint port)
194 {
195 SOCKETADDRESS sa;
196 int rv;
197 int sa_len = 0;
198 SOCKET s = (SOCKET)fdval(env, fdo);
199
200 if (NET_InetAddressToSockaddr(env, iao, port, &sa, &sa_len, preferIPv6) != 0) {
201 return IOS_THROWN;
202 }
203
204 rv = connect(s, &sa.sa, sa_len);
205 if (rv != 0) {
206 int err = WSAGetLastError();
207 if (err == WSAEINPROGRESS || err == WSAEWOULDBLOCK) {
208 return IOS_UNAVAILABLE;
209 }
210 NET_ThrowNew(env, err, "connect");
211 return IOS_THROWN;
212 } else {
213 /* Enable WSAECONNRESET errors when a UDP socket is connected */
214 int type = 0, optlen = sizeof(type);
215 rv = getsockopt(s, SOL_SOCKET, SO_TYPE, (char*)&type, &optlen);
216 if (rv == 0 && type == SOCK_DGRAM) {
217 setConnectionReset(s, TRUE);
218 }
219 }
220 return 1;
221 }
222
223 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_localPort(JNIEnv * env,jclass clazz,jobject fdo)224 Java_sun_nio_ch_Net_localPort(JNIEnv *env, jclass clazz, jobject fdo)
225 {
226 SOCKETADDRESS sa;
227 int sa_len = sizeof(sa);
228
229 if (getsockname(fdval(env, fdo), &sa.sa, &sa_len) < 0) {
230 int error = WSAGetLastError();
231 if (error == WSAEINVAL) {
232 return 0;
233 }
234 NET_ThrowNew(env, error, "getsockname");
235 return IOS_THROWN;
236 }
237 return NET_GetPortFromSockaddr(&sa);
238 }
239
240 JNIEXPORT jobject JNICALL
Java_sun_nio_ch_Net_localInetAddress(JNIEnv * env,jclass clazz,jobject fdo)241 Java_sun_nio_ch_Net_localInetAddress(JNIEnv *env, jclass clazz, jobject fdo)
242 {
243 SOCKETADDRESS sa;
244 int sa_len = sizeof(sa);
245 int port;
246
247 if (getsockname(fdval(env, fdo), &sa.sa, &sa_len) < 0) {
248 NET_ThrowNew(env, WSAGetLastError(), "getsockname");
249 return NULL;
250 }
251 return NET_SockaddrToInetAddress(env, &sa, &port);
252 }
253
254 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_remotePort(JNIEnv * env,jclass clazz,jobject fdo)255 Java_sun_nio_ch_Net_remotePort(JNIEnv *env, jclass clazz, jobject fdo)
256 {
257 SOCKETADDRESS sa;
258 int sa_len = sizeof(sa);
259
260 if (getpeername(fdval(env, fdo), &sa.sa, &sa_len) < 0) {
261 int error = WSAGetLastError();
262 if (error == WSAEINVAL) {
263 return 0;
264 }
265 NET_ThrowNew(env, error, "getsockname");
266 return IOS_THROWN;
267 }
268 return NET_GetPortFromSockaddr(&sa);
269 }
270
271 JNIEXPORT jobject JNICALL
Java_sun_nio_ch_Net_remoteInetAddress(JNIEnv * env,jclass clazz,jobject fdo)272 Java_sun_nio_ch_Net_remoteInetAddress(JNIEnv *env, jclass clazz, jobject fdo)
273 {
274 SOCKETADDRESS sa;
275 int sa_len = sizeof(sa);
276 int port;
277
278 if (getpeername(fdval(env, fdo), &sa.sa, &sa_len) < 0) {
279 NET_ThrowNew(env, WSAGetLastError(), "getsockname");
280 return NULL;
281 }
282 return NET_SockaddrToInetAddress(env, &sa, &port);
283 }
284
285 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_getIntOption0(JNIEnv * env,jclass clazz,jobject fdo,jboolean mayNeedConversion,jint level,jint opt)286 Java_sun_nio_ch_Net_getIntOption0(JNIEnv *env, jclass clazz, jobject fdo,
287 jboolean mayNeedConversion, jint level, jint opt)
288 {
289 int result = 0;
290 struct linger linger;
291 char *arg;
292 int arglen, n;
293
294 if (level == SOL_SOCKET && opt == SO_LINGER) {
295 arg = (char *)&linger;
296 arglen = sizeof(linger);
297 } else {
298 arg = (char *)&result;
299 arglen = sizeof(result);
300 }
301
302 /**
303 * HACK: IP_TOS is deprecated on Windows and querying the option
304 * returns a protocol error. NET_GetSockOpt handles this and uses
305 * a fallback mechanism. Same applies to IPV6_TCLASS
306 */
307 if ((level == IPPROTO_IP && opt == IP_TOS) || (level == IPPROTO_IPV6 && opt == IPV6_TCLASS)) {
308 mayNeedConversion = JNI_TRUE;
309 }
310
311 if (mayNeedConversion) {
312 n = NET_GetSockOpt(fdval(env, fdo), level, opt, arg, &arglen);
313 } else {
314 n = getsockopt(fdval(env, fdo), level, opt, arg, &arglen);
315 }
316 if (n < 0) {
317 handleSocketError(env, WSAGetLastError());
318 return IOS_THROWN;
319 }
320
321 if (level == SOL_SOCKET && opt == SO_LINGER)
322 return linger.l_onoff ? linger.l_linger : -1;
323 else
324 return result;
325 }
326
327 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_setIntOption0(JNIEnv * env,jclass clazz,jobject fdo,jboolean mayNeedConversion,jint level,jint opt,jint arg,jboolean ipv6)328 Java_sun_nio_ch_Net_setIntOption0(JNIEnv *env, jclass clazz, jobject fdo,
329 jboolean mayNeedConversion, jint level, jint opt, jint arg, jboolean ipv6)
330 {
331 struct linger linger;
332 char *parg;
333 int arglen, n;
334
335 if (level == SOL_SOCKET && opt == SO_LINGER) {
336 parg = (char *)&linger;
337 arglen = sizeof(linger);
338 if (arg >= 0) {
339 linger.l_onoff = 1;
340 linger.l_linger = (unsigned short)arg;
341 } else {
342 linger.l_onoff = 0;
343 linger.l_linger = 0;
344 }
345 } else {
346 parg = (char *)&arg;
347 arglen = sizeof(arg);
348 }
349
350 if (level == IPPROTO_IPV6 && opt == IPV6_TCLASS) {
351 /* No op */
352 return;
353 }
354
355 if (mayNeedConversion) {
356 n = NET_SetSockOpt(fdval(env, fdo), level, opt, parg, arglen);
357 } else {
358 n = setsockopt(fdval(env, fdo), level, opt, parg, arglen);
359 }
360 if (n < 0)
361 handleSocketError(env, WSAGetLastError());
362 }
363
364 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_joinOrDrop4(JNIEnv * env,jobject this,jboolean join,jobject fdo,jint group,jint interf,jint source)365 Java_sun_nio_ch_Net_joinOrDrop4(JNIEnv *env, jobject this, jboolean join, jobject fdo,
366 jint group, jint interf, jint source)
367 {
368 struct ip_mreq mreq;
369 struct my_ip_mreq_source mreq_source;
370 int opt, n, optlen;
371 void* optval;
372
373 if (source == 0) {
374 mreq.imr_multiaddr.s_addr = htonl(group);
375 mreq.imr_interface.s_addr = htonl(interf);
376 opt = (join) ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP;
377 optval = (void*)&mreq;
378 optlen = sizeof(mreq);
379 } else {
380 mreq_source.imr_multiaddr.s_addr = htonl(group);
381 mreq_source.imr_sourceaddr.s_addr = htonl(source);
382 mreq_source.imr_interface.s_addr = htonl(interf);
383 opt = (join) ? IP_ADD_SOURCE_MEMBERSHIP : IP_DROP_SOURCE_MEMBERSHIP;
384 optval = (void*)&mreq_source;
385 optlen = sizeof(mreq_source);
386 }
387
388 n = setsockopt(fdval(env,fdo), IPPROTO_IP, opt, optval, optlen);
389 if (n < 0) {
390 if (join && (WSAGetLastError() == WSAENOPROTOOPT))
391 return IOS_UNAVAILABLE;
392 handleSocketError(env, WSAGetLastError());
393 }
394 return 0;
395 }
396
397 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_blockOrUnblock4(JNIEnv * env,jobject this,jboolean block,jobject fdo,jint group,jint interf,jint source)398 Java_sun_nio_ch_Net_blockOrUnblock4(JNIEnv *env, jobject this, jboolean block, jobject fdo,
399 jint group, jint interf, jint source)
400 {
401 struct my_ip_mreq_source mreq_source;
402 int n;
403 int opt = (block) ? IP_BLOCK_SOURCE : IP_UNBLOCK_SOURCE;
404
405 mreq_source.imr_multiaddr.s_addr = htonl(group);
406 mreq_source.imr_sourceaddr.s_addr = htonl(source);
407 mreq_source.imr_interface.s_addr = htonl(interf);
408
409 n = setsockopt(fdval(env,fdo), IPPROTO_IP, opt,
410 (void*)&mreq_source, sizeof(mreq_source));
411 if (n < 0) {
412 if (block && (WSAGetLastError() == WSAENOPROTOOPT))
413 return IOS_UNAVAILABLE;
414 handleSocketError(env, WSAGetLastError());
415 }
416 return 0;
417 }
418
419 /**
420 * Call setsockopt with a IPPROTO_IPV6 level socket option
421 * and a group_source_req structure as the option value. The
422 * given IPv6 group, interface index, and IPv6 source address
423 * are copied into the structure.
424 */
setGroupSourceReqOption(JNIEnv * env,jobject fdo,int opt,jbyteArray group,jint index,jbyteArray source)425 static int setGroupSourceReqOption(JNIEnv* env,
426 jobject fdo,
427 int opt,
428 jbyteArray group,
429 jint index,
430 jbyteArray source)
431 {
432 struct my_group_source_req req;
433 struct sockaddr_in6* sin6;
434
435 req.gsr_interface = (ULONG)index;
436
437 sin6 = (struct sockaddr_in6*)&(req.gsr_group);
438 sin6->sin6_family = AF_INET6;
439 COPY_INET6_ADDRESS(env, group, (jbyte*)&(sin6->sin6_addr));
440
441 sin6 = (struct sockaddr_in6*)&(req.gsr_source);
442 sin6->sin6_family = AF_INET6;
443 COPY_INET6_ADDRESS(env, source, (jbyte*)&(sin6->sin6_addr));
444
445 return setsockopt(fdval(env,fdo), IPPROTO_IPV6, opt, (void*)&req, sizeof(req));
446 }
447
448 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_joinOrDrop6(JNIEnv * env,jobject this,jboolean join,jobject fdo,jbyteArray group,jint index,jbyteArray source)449 Java_sun_nio_ch_Net_joinOrDrop6(JNIEnv *env, jobject this, jboolean join, jobject fdo,
450 jbyteArray group, jint index, jbyteArray source)
451 {
452 struct ipv6_mreq mreq6;
453 int n;
454
455 if (source == NULL) {
456 int opt = (join) ? IPV6_ADD_MEMBERSHIP : IPV6_DROP_MEMBERSHIP;
457 COPY_INET6_ADDRESS(env, group, (jbyte*)&(mreq6.ipv6mr_multiaddr));
458 mreq6.ipv6mr_interface = (int)index;
459 n = setsockopt(fdval(env,fdo), IPPROTO_IPV6, opt,
460 (void*)&mreq6, sizeof(mreq6));
461 } else {
462 int opt = (join) ? MCAST_JOIN_SOURCE_GROUP : MCAST_LEAVE_SOURCE_GROUP;
463 n = setGroupSourceReqOption(env, fdo, opt, group, index, source);
464 }
465
466 if (n < 0) {
467 handleSocketError(env, errno);
468 }
469 return 0;
470 }
471
472 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_blockOrUnblock6(JNIEnv * env,jobject this,jboolean block,jobject fdo,jbyteArray group,jint index,jbyteArray source)473 Java_sun_nio_ch_Net_blockOrUnblock6(JNIEnv *env, jobject this, jboolean block, jobject fdo,
474 jbyteArray group, jint index, jbyteArray source)
475 {
476 int opt = (block) ? MCAST_BLOCK_SOURCE : MCAST_UNBLOCK_SOURCE;
477 int n = setGroupSourceReqOption(env, fdo, opt, group, index, source);
478 if (n < 0) {
479 handleSocketError(env, errno);
480 }
481 return 0;
482 }
483
484 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_setInterface4(JNIEnv * env,jobject this,jobject fdo,jint interf)485 Java_sun_nio_ch_Net_setInterface4(JNIEnv* env, jobject this, jobject fdo, jint interf)
486 {
487 struct in_addr in;
488 int arglen = sizeof(struct in_addr);
489 int n;
490
491 in.s_addr = htonl(interf);
492
493 n = setsockopt(fdval(env, fdo), IPPROTO_IP, IP_MULTICAST_IF,
494 (void*)&(in.s_addr), arglen);
495 if (n < 0) {
496 handleSocketError(env, WSAGetLastError());
497 }
498 }
499
500 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_getInterface4(JNIEnv * env,jobject this,jobject fdo)501 Java_sun_nio_ch_Net_getInterface4(JNIEnv* env, jobject this, jobject fdo)
502 {
503 struct in_addr in;
504 int arglen = sizeof(struct in_addr);
505 int n;
506
507 n = getsockopt(fdval(env, fdo), IPPROTO_IP, IP_MULTICAST_IF, (void*)&in, &arglen);
508 if (n < 0) {
509 handleSocketError(env, WSAGetLastError());
510 return IOS_THROWN;
511 }
512 return ntohl(in.s_addr);
513 }
514
515 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_setInterface6(JNIEnv * env,jobject this,jobject fdo,jint index)516 Java_sun_nio_ch_Net_setInterface6(JNIEnv* env, jobject this, jobject fdo, jint index)
517 {
518 int value = (jint)index;
519 int arglen = sizeof(value);
520 int n;
521
522 n = setsockopt(fdval(env, fdo), IPPROTO_IPV6, IPV6_MULTICAST_IF,
523 (void*)&(index), arglen);
524 if (n < 0) {
525 handleSocketError(env, errno);
526 }
527 }
528
529 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_getInterface6(JNIEnv * env,jobject this,jobject fdo)530 Java_sun_nio_ch_Net_getInterface6(JNIEnv* env, jobject this, jobject fdo)
531 {
532 int index;
533 int arglen = sizeof(index);
534 int n;
535
536 n = getsockopt(fdval(env, fdo), IPPROTO_IPV6, IPV6_MULTICAST_IF, (void*)&index, &arglen);
537 if (n < 0) {
538 handleSocketError(env, errno);
539 return -1;
540 }
541 return (jint)index;
542 }
543
544 JNIEXPORT void JNICALL
Java_sun_nio_ch_Net_shutdown(JNIEnv * env,jclass cl,jobject fdo,jint jhow)545 Java_sun_nio_ch_Net_shutdown(JNIEnv *env, jclass cl, jobject fdo, jint jhow) {
546 int how = (jhow == sun_nio_ch_Net_SHUT_RD) ? SD_RECEIVE :
547 (jhow == sun_nio_ch_Net_SHUT_WR) ? SD_SEND : SD_BOTH;
548 if (shutdown(fdval(env, fdo), how) == SOCKET_ERROR) {
549 NET_ThrowNew(env, WSAGetLastError(), "shutdown");
550 }
551 }
552
553 JNIEXPORT jint JNICALL
Java_sun_nio_ch_Net_poll(JNIEnv * env,jclass this,jobject fdo,jint events,jlong timeout)554 Java_sun_nio_ch_Net_poll(JNIEnv* env, jclass this, jobject fdo, jint events, jlong timeout)
555 {
556 int rv;
557 int revents = 0;
558 struct timeval t;
559 int lastError = 0;
560 fd_set rd, wr, ex;
561 jint fd = fdval(env, fdo);
562
563 t.tv_sec = (long)(timeout / 1000);
564 t.tv_usec = (timeout % 1000) * 1000;
565
566 FD_ZERO(&rd);
567 FD_ZERO(&wr);
568 FD_ZERO(&ex);
569 if (events & POLLIN) {
570 FD_SET(fd, &rd);
571 }
572 if (events & POLLOUT ||
573 events & POLLCONN) {
574 FD_SET(fd, &wr);
575 }
576 FD_SET(fd, &ex);
577
578 rv = select(fd+1, &rd, &wr, &ex, &t);
579
580 /* save last winsock error */
581 if (rv == SOCKET_ERROR) {
582 handleSocketError(env, lastError);
583 return IOS_THROWN;
584 } else if (rv >= 0) {
585 rv = 0;
586 if (FD_ISSET(fd, &rd)) {
587 rv |= POLLIN;
588 }
589 if (FD_ISSET(fd, &wr)) {
590 rv |= POLLOUT;
591 }
592 if (FD_ISSET(fd, &ex)) {
593 rv |= POLLERR;
594 }
595 }
596 return rv;
597 }
598
599 JNIEXPORT jshort JNICALL
Java_sun_nio_ch_Net_pollinValue(JNIEnv * env,jclass this)600 Java_sun_nio_ch_Net_pollinValue(JNIEnv *env, jclass this)
601 {
602 return (jshort)POLLIN;
603 }
604
605 JNIEXPORT jshort JNICALL
Java_sun_nio_ch_Net_polloutValue(JNIEnv * env,jclass this)606 Java_sun_nio_ch_Net_polloutValue(JNIEnv *env, jclass this)
607 {
608 return (jshort)POLLOUT;
609 }
610
611 JNIEXPORT jshort JNICALL
Java_sun_nio_ch_Net_pollerrValue(JNIEnv * env,jclass this)612 Java_sun_nio_ch_Net_pollerrValue(JNIEnv *env, jclass this)
613 {
614 return (jshort)POLLERR;
615 }
616
617 JNIEXPORT jshort JNICALL
Java_sun_nio_ch_Net_pollhupValue(JNIEnv * env,jclass this)618 Java_sun_nio_ch_Net_pollhupValue(JNIEnv *env, jclass this)
619 {
620 return (jshort)POLLHUP;
621 }
622
623 JNIEXPORT jshort JNICALL
Java_sun_nio_ch_Net_pollnvalValue(JNIEnv * env,jclass this)624 Java_sun_nio_ch_Net_pollnvalValue(JNIEnv *env, jclass this)
625 {
626 return (jshort)POLLNVAL;
627 }
628
629 JNIEXPORT jshort JNICALL
Java_sun_nio_ch_Net_pollconnValue(JNIEnv * env,jclass this)630 Java_sun_nio_ch_Net_pollconnValue(JNIEnv *env, jclass this)
631 {
632 return (jshort)POLLCONN;
633 }
634