Path: blob/master/src/java.base/windows/native/libnio/ch/Net.c
41134 views
/*1* Copyright (c) 2001, 2021, Oracle and/or its affiliates. All rights reserved.2* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.3*4* This code is free software; you can redistribute it and/or modify it5* under the terms of the GNU General Public License version 2 only, as6* published by the Free Software Foundation. Oracle designates this7* particular file as subject to the "Classpath" exception as provided8* by Oracle in the LICENSE file that accompanied this code.9*10* This code is distributed in the hope that it will be useful, but WITHOUT11* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or12* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License13* version 2 for more details (a copy is included in the LICENSE file that14* accompanied this code).15*16* You should have received a copy of the GNU General Public License version17* 2 along with this work; if not, write to the Free Software Foundation,18* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.19*20* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA21* or visit www.oracle.com if you need additional information or have any22* questions.23*/2425#include <windows.h>26#include <winsock2.h>2728#include "jni.h"29#include "jni_util.h"30#include "jvm.h"31#include "jlong.h"32#include "nio.h"33#include "nio_util.h"34#include "net_util.h"3536#include "sun_nio_ch_Net.h"37#include "sun_nio_ch_PollArrayWrapper.h"3839/**40* Definitions to allow for building with older SDK include files.41*/4243#ifndef MCAST_BLOCK_SOURCE4445#define MCAST_BLOCK_SOURCE 4346#define MCAST_UNBLOCK_SOURCE 4447#define MCAST_JOIN_SOURCE_GROUP 4548#define MCAST_LEAVE_SOURCE_GROUP 464950#endif /* MCAST_BLOCK_SOURCE */5152struct my_ip_mreq_source {53IN_ADDR imr_multiaddr;54IN_ADDR imr_sourceaddr;55IN_ADDR imr_interface;56};5758struct my_group_source_req {59ULONG gsr_interface;60SOCKADDR_STORAGE gsr_group;61SOCKADDR_STORAGE gsr_source;62};6364/**65* Copy IPv6 address as jbytearray to target66*/67#define COPY_INET6_ADDRESS(env, source, target) \68(*env)->GetByteArrayRegion(env, source, 0, 16, target)6970/**71* Enable or disable receipt of WSAECONNRESET errors.72*/73static void setConnectionReset(SOCKET s, BOOL enable) {74DWORD bytesReturned = 0;75WSAIoctl(s, SIO_UDP_CONNRESET, &enable, sizeof(enable),76NULL, 0, &bytesReturned, NULL, NULL);77}7879jint handleSocketError(JNIEnv *env, int errorValue)80{81NET_ThrowNew(env, errorValue, NULL);82return IOS_THROWN;83}8485static jclass isa_class; /* java.net.InetSocketAddress */86static jmethodID isa_ctorID; /* InetSocketAddress(InetAddress, int) */8788JNIEXPORT void JNICALL89Java_sun_nio_ch_Net_initIDs(JNIEnv *env, jclass clazz)90{91jclass cls = (*env)->FindClass(env, "java/net/InetSocketAddress");92CHECK_NULL(cls);93isa_class = (*env)->NewGlobalRef(env, cls);94if (isa_class == NULL) {95JNU_ThrowOutOfMemoryError(env, NULL);96return;97}98isa_ctorID = (*env)->GetMethodID(env, cls, "<init>", "(Ljava/net/InetAddress;I)V");99CHECK_NULL(isa_ctorID);100101initInetAddressIDs(env);102}103104JNIEXPORT jboolean JNICALL105Java_sun_nio_ch_Net_isIPv6Available0(JNIEnv* env, jclass cl)106{107/*108* Return true if IPv6 is configured109*/110return ipv6_available() ? JNI_TRUE : JNI_FALSE;111}112113JNIEXPORT jboolean JNICALL114Java_sun_nio_ch_Net_isReusePortAvailable0(JNIEnv* env, jclass c1)115{116// SO_REUSEPORT is not supported on Windows117return JNI_FALSE;118}119120JNIEXPORT jint JNICALL121Java_sun_nio_ch_Net_isExclusiveBindAvailable(JNIEnv *env, jclass clazz) {122return 1;123}124125JNIEXPORT jboolean JNICALL126Java_sun_nio_ch_Net_shouldSetBothIPv4AndIPv6Options0(JNIEnv* env, jclass cl)127{128/* Set both IPv4 and IPv6 socket options when setting multicast options */129return JNI_TRUE;130}131132JNIEXPORT jboolean JNICALL133Java_sun_nio_ch_Net_canIPv6SocketJoinIPv4Group0(JNIEnv* env, jclass cl)134{135/* IPv6 sockets can join IPv4 multicast groups */136return JNI_TRUE;137}138139JNIEXPORT jboolean JNICALL140Java_sun_nio_ch_Net_canJoin6WithIPv4Group0(JNIEnv* env, jclass cl)141{142/* IPV6_ADD_MEMBERSHIP cannot be used to join IPv4 multicast groups */143return JNI_FALSE;144}145146JNIEXPORT jboolean JNICALL147Java_sun_nio_ch_Net_canUseIPv6OptionsWithIPv4LocalAddress0(JNIEnv* env, jclass cl)148{149/* IPV6_XXX socket options cannot be used on IPv6 sockets bound to IPv4 address */150return JNI_FALSE;151}152153JNIEXPORT jint JNICALL154Java_sun_nio_ch_Net_socket0(JNIEnv *env, jclass cl, jboolean preferIPv6,155jboolean stream, jboolean reuse, jboolean fastLoopback)156{157SOCKET s;158int domain = (preferIPv6) ? AF_INET6 : AF_INET;159160s = socket(domain, (stream ? SOCK_STREAM : SOCK_DGRAM), 0);161if (s != INVALID_SOCKET) {162SetHandleInformation((HANDLE)s, HANDLE_FLAG_INHERIT, 0);163164/* IPV6_V6ONLY is true by default */165if (domain == AF_INET6) {166int opt = 0;167setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY,168(const char *)&opt, sizeof(opt));169}170171/* Disable WSAECONNRESET errors for initially unconnected UDP sockets */172if (!stream) {173setConnectionReset(s, FALSE);174}175176} else {177NET_ThrowNew(env, WSAGetLastError(), "socket");178}179180if (stream && fastLoopback) {181static int loopback_available = 1;182if (loopback_available) {183int rv = NET_EnableFastTcpLoopback((jint)s);184if (rv) {185if (rv == WSAEOPNOTSUPP) {186loopback_available = 0;187} else {188NET_ThrowNew(env, rv, "fastLoopback");189}190}191}192}193194return (jint)s;195}196197JNIEXPORT void JNICALL198Java_sun_nio_ch_Net_bind0(JNIEnv *env, jclass clazz, jobject fdo, jboolean preferIPv6,199jboolean isExclBind, jobject iao, jint port)200{201SOCKETADDRESS sa;202int rv;203int sa_len = 0;204205if (NET_InetAddressToSockaddr(env, iao, port, &sa, &sa_len, preferIPv6) != 0) {206return;207}208209rv = NET_WinBind(fdval(env, fdo), &sa, sa_len, isExclBind);210if (rv == SOCKET_ERROR)211NET_ThrowNew(env, WSAGetLastError(), "bind");212}213214JNIEXPORT void JNICALL215Java_sun_nio_ch_Net_listen(JNIEnv *env, jclass cl, jobject fdo, jint backlog)216{217if (listen(fdval(env,fdo), backlog) == SOCKET_ERROR) {218NET_ThrowNew(env, WSAGetLastError(), "listen");219}220}221222JNIEXPORT jint JNICALL223Java_sun_nio_ch_Net_connect0(JNIEnv *env, jclass clazz, jboolean preferIPv6, jobject fdo,224jobject iao, jint port)225{226SOCKETADDRESS sa;227int rv;228int so_rv;229int sa_len = 0;230SOCKET s = (SOCKET)fdval(env, fdo);231int type = 0, optlen = sizeof(type);232233if (NET_InetAddressToSockaddr(env, iao, port, &sa, &sa_len, preferIPv6) != 0) {234return IOS_THROWN;235}236237so_rv = getsockopt(s, SOL_SOCKET, SO_TYPE, (char*)&type, &optlen);238239/**240* Windows has a very long socket connect timeout of 2 seconds.241* If it's the loopback adapter we can shorten the wait interval.242*/243if (so_rv == 0 && type == SOCK_STREAM && IS_LOOPBACK_ADDRESS(&sa)) {244NET_EnableFastTcpLoopbackConnect((jint)s);245}246247rv = connect(s, &sa.sa, sa_len);248if (rv != 0) {249int err = WSAGetLastError();250if (err == WSAEINPROGRESS || err == WSAEWOULDBLOCK) {251return IOS_UNAVAILABLE;252}253NET_ThrowNew(env, err, "connect");254return IOS_THROWN;255} else {256/* Enable WSAECONNRESET errors when a UDP socket is connected */257if (so_rv == 0 && type == SOCK_DGRAM) {258setConnectionReset(s, TRUE);259}260}261return 1;262}263264JNIEXPORT jint JNICALL265Java_sun_nio_ch_Net_accept(JNIEnv *env, jclass clazz, jobject fdo, jobject newfdo,266jobjectArray isaa)267{268jint fd = fdval(env,fdo);269jint newfd;270SOCKETADDRESS sa;271int addrlen = sizeof(sa);272jobject remote_ia;273jint remote_port = 0;274jobject isa;275276memset((char *)&sa, 0, sizeof(sa));277newfd = (jint) accept(fd, &sa.sa, &addrlen);278if (newfd == INVALID_SOCKET) {279int theErr = (jint)WSAGetLastError();280if (theErr == WSAEWOULDBLOCK) {281return IOS_UNAVAILABLE;282}283JNU_ThrowIOExceptionWithLastError(env, "Accept failed");284return IOS_THROWN;285}286287SetHandleInformation((HANDLE)(UINT_PTR)newfd, HANDLE_FLAG_INHERIT, 0);288setfdval(env, newfdo, newfd);289290remote_ia = NET_SockaddrToInetAddress(env, &sa, (int *)&remote_port);291CHECK_NULL_RETURN(remote_ia, IOS_THROWN);292293isa = (*env)->NewObject(env, isa_class, isa_ctorID, remote_ia, remote_port);294CHECK_NULL_RETURN(isa, IOS_THROWN);295(*env)->SetObjectArrayElement(env, isaa, 0, isa);296297return 1;298}299300JNIEXPORT jint JNICALL301Java_sun_nio_ch_Net_localPort(JNIEnv *env, jclass clazz, jobject fdo)302{303SOCKETADDRESS sa;304int sa_len = sizeof(sa);305306if (getsockname(fdval(env, fdo), &sa.sa, &sa_len) == SOCKET_ERROR) {307int error = WSAGetLastError();308if (error == WSAEINVAL) {309return 0;310}311NET_ThrowNew(env, error, "getsockname");312return IOS_THROWN;313}314return NET_GetPortFromSockaddr(&sa);315}316317JNIEXPORT jobject JNICALL318Java_sun_nio_ch_Net_localInetAddress(JNIEnv *env, jclass clazz, jobject fdo)319{320SOCKETADDRESS sa;321int sa_len = sizeof(sa);322int port;323324if (getsockname(fdval(env, fdo), &sa.sa, &sa_len) == SOCKET_ERROR) {325NET_ThrowNew(env, WSAGetLastError(), "getsockname");326return NULL;327}328return NET_SockaddrToInetAddress(env, &sa, &port);329}330331JNIEXPORT jint JNICALL332Java_sun_nio_ch_Net_remotePort(JNIEnv *env, jclass clazz, jobject fdo)333{334SOCKETADDRESS sa;335int sa_len = sizeof(sa);336337if (getpeername(fdval(env, fdo), &sa.sa, &sa_len) == SOCKET_ERROR) {338int error = WSAGetLastError();339if (error == WSAEINVAL) {340return 0;341}342NET_ThrowNew(env, error, "getsockname");343return IOS_THROWN;344}345return NET_GetPortFromSockaddr(&sa);346}347348JNIEXPORT jobject JNICALL349Java_sun_nio_ch_Net_remoteInetAddress(JNIEnv *env, jclass clazz, jobject fdo)350{351SOCKETADDRESS sa;352int sa_len = sizeof(sa);353int port;354355if (getpeername(fdval(env, fdo), &sa.sa, &sa_len) == SOCKET_ERROR) {356NET_ThrowNew(env, WSAGetLastError(), "getsockname");357return NULL;358}359return NET_SockaddrToInetAddress(env, &sa, &port);360}361362JNIEXPORT jint JNICALL363Java_sun_nio_ch_Net_getIntOption0(JNIEnv *env, jclass clazz, jobject fdo,364jboolean mayNeedConversion, jint level, jint opt)365{366int result = 0;367struct linger linger;368char *arg;369int arglen, n;370371if (level == SOL_SOCKET && opt == SO_LINGER) {372arg = (char *)&linger;373arglen = sizeof(linger);374} else {375arg = (char *)&result;376arglen = sizeof(result);377}378379/**380* HACK: IP_TOS is deprecated on Windows and querying the option381* returns a protocol error. NET_GetSockOpt handles this and uses382* a fallback mechanism. Same applies to IPV6_TCLASS383*/384if ((level == IPPROTO_IP && opt == IP_TOS) || (level == IPPROTO_IPV6 && opt == IPV6_TCLASS)) {385mayNeedConversion = JNI_TRUE;386}387388if (mayNeedConversion) {389n = NET_GetSockOpt(fdval(env, fdo), level, opt, arg, &arglen);390} else {391n = getsockopt(fdval(env, fdo), level, opt, arg, &arglen);392}393if (n == SOCKET_ERROR) {394handleSocketError(env, WSAGetLastError());395return IOS_THROWN;396}397398if (level == SOL_SOCKET && opt == SO_LINGER)399return linger.l_onoff ? linger.l_linger : -1;400else401return result;402}403404JNIEXPORT void JNICALL405Java_sun_nio_ch_Net_setIntOption0(JNIEnv *env, jclass clazz, jobject fdo,406jboolean mayNeedConversion, jint level, jint opt, jint arg, jboolean ipv6)407{408struct linger linger;409char *parg;410int arglen, n;411412if (level == SOL_SOCKET && opt == SO_LINGER) {413parg = (char *)&linger;414arglen = sizeof(linger);415if (arg >= 0) {416linger.l_onoff = 1;417linger.l_linger = (unsigned short)arg;418} else {419linger.l_onoff = 0;420linger.l_linger = 0;421}422} else {423parg = (char *)&arg;424arglen = sizeof(arg);425}426427if (level == IPPROTO_IPV6 && opt == IPV6_TCLASS) {428/* No op */429return;430}431432if (mayNeedConversion) {433n = NET_SetSockOpt(fdval(env, fdo), level, opt, parg, arglen);434} else {435n = setsockopt(fdval(env, fdo), level, opt, parg, arglen);436}437if (n == SOCKET_ERROR)438handleSocketError(env, WSAGetLastError());439}440441JNIEXPORT jint JNICALL442Java_sun_nio_ch_Net_joinOrDrop4(JNIEnv *env, jobject this, jboolean join, jobject fdo,443jint group, jint interf, jint source)444{445struct ip_mreq mreq;446struct my_ip_mreq_source mreq_source;447int opt, n, optlen;448void* optval;449450if (source == 0) {451mreq.imr_multiaddr.s_addr = htonl(group);452mreq.imr_interface.s_addr = htonl(interf);453opt = (join) ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP;454optval = (void*)&mreq;455optlen = sizeof(mreq);456} else {457mreq_source.imr_multiaddr.s_addr = htonl(group);458mreq_source.imr_sourceaddr.s_addr = htonl(source);459mreq_source.imr_interface.s_addr = htonl(interf);460opt = (join) ? IP_ADD_SOURCE_MEMBERSHIP : IP_DROP_SOURCE_MEMBERSHIP;461optval = (void*)&mreq_source;462optlen = sizeof(mreq_source);463}464465n = setsockopt(fdval(env,fdo), IPPROTO_IP, opt, optval, optlen);466if (n == SOCKET_ERROR) {467if (join && (WSAGetLastError() == WSAENOPROTOOPT))468return IOS_UNAVAILABLE;469handleSocketError(env, WSAGetLastError());470}471return 0;472}473474JNIEXPORT jint JNICALL475Java_sun_nio_ch_Net_blockOrUnblock4(JNIEnv *env, jobject this, jboolean block, jobject fdo,476jint group, jint interf, jint source)477{478struct my_ip_mreq_source mreq_source;479int n;480int opt = (block) ? IP_BLOCK_SOURCE : IP_UNBLOCK_SOURCE;481482mreq_source.imr_multiaddr.s_addr = htonl(group);483mreq_source.imr_sourceaddr.s_addr = htonl(source);484mreq_source.imr_interface.s_addr = htonl(interf);485486n = setsockopt(fdval(env,fdo), IPPROTO_IP, opt,487(void*)&mreq_source, sizeof(mreq_source));488if (n == SOCKET_ERROR) {489if (block && (WSAGetLastError() == WSAENOPROTOOPT))490return IOS_UNAVAILABLE;491handleSocketError(env, WSAGetLastError());492}493return 0;494}495496/**497* Call setsockopt with a IPPROTO_IPV6 level socket option498* and a group_source_req structure as the option value. The499* given IPv6 group, interface index, and IPv6 source address500* are copied into the structure.501*/502static int setGroupSourceReqOption(JNIEnv* env,503jobject fdo,504int opt,505jbyteArray group,506jint index,507jbyteArray source)508{509struct my_group_source_req req;510struct sockaddr_in6* sin6;511512req.gsr_interface = (ULONG)index;513514sin6 = (struct sockaddr_in6*)&(req.gsr_group);515sin6->sin6_family = AF_INET6;516COPY_INET6_ADDRESS(env, group, (jbyte*)&(sin6->sin6_addr));517518sin6 = (struct sockaddr_in6*)&(req.gsr_source);519sin6->sin6_family = AF_INET6;520COPY_INET6_ADDRESS(env, source, (jbyte*)&(sin6->sin6_addr));521522return setsockopt(fdval(env,fdo), IPPROTO_IPV6, opt, (void*)&req, sizeof(req));523}524525JNIEXPORT jint JNICALL526Java_sun_nio_ch_Net_joinOrDrop6(JNIEnv *env, jobject this, jboolean join, jobject fdo,527jbyteArray group, jint index, jbyteArray source)528{529struct ipv6_mreq mreq6;530int n;531532if (source == NULL) {533int opt = (join) ? IPV6_ADD_MEMBERSHIP : IPV6_DROP_MEMBERSHIP;534COPY_INET6_ADDRESS(env, group, (jbyte*)&(mreq6.ipv6mr_multiaddr));535mreq6.ipv6mr_interface = (int)index;536n = setsockopt(fdval(env,fdo), IPPROTO_IPV6, opt,537(void*)&mreq6, sizeof(mreq6));538} else {539int opt = (join) ? MCAST_JOIN_SOURCE_GROUP : MCAST_LEAVE_SOURCE_GROUP;540n = setGroupSourceReqOption(env, fdo, opt, group, index, source);541}542543if (n == SOCKET_ERROR) {544handleSocketError(env, WSAGetLastError());545}546return 0;547}548549JNIEXPORT jint JNICALL550Java_sun_nio_ch_Net_blockOrUnblock6(JNIEnv *env, jobject this, jboolean block, jobject fdo,551jbyteArray group, jint index, jbyteArray source)552{553int opt = (block) ? MCAST_BLOCK_SOURCE : MCAST_UNBLOCK_SOURCE;554int n = setGroupSourceReqOption(env, fdo, opt, group, index, source);555if (n == SOCKET_ERROR) {556handleSocketError(env, WSAGetLastError());557}558return 0;559}560561JNIEXPORT void JNICALL562Java_sun_nio_ch_Net_setInterface4(JNIEnv* env, jobject this, jobject fdo, jint interf)563{564struct in_addr in;565int arglen = sizeof(struct in_addr);566int n;567568in.s_addr = htonl(interf);569570n = setsockopt(fdval(env, fdo), IPPROTO_IP, IP_MULTICAST_IF,571(void*)&(in.s_addr), arglen);572if (n == SOCKET_ERROR) {573handleSocketError(env, WSAGetLastError());574}575}576577JNIEXPORT jint JNICALL578Java_sun_nio_ch_Net_getInterface4(JNIEnv* env, jobject this, jobject fdo)579{580struct in_addr in;581int arglen = sizeof(struct in_addr);582int n;583584n = getsockopt(fdval(env, fdo), IPPROTO_IP, IP_MULTICAST_IF, (void*)&in, &arglen);585if (n == SOCKET_ERROR) {586handleSocketError(env, WSAGetLastError());587return IOS_THROWN;588}589return ntohl(in.s_addr);590}591592JNIEXPORT void JNICALL593Java_sun_nio_ch_Net_setInterface6(JNIEnv* env, jobject this, jobject fdo, jint index)594{595DWORD value = (jint)index;596int arglen = sizeof(value);597int n;598599n = setsockopt(fdval(env, fdo), IPPROTO_IPV6, IPV6_MULTICAST_IF,600(void*)&(index), arglen);601if (n == SOCKET_ERROR) {602handleSocketError(env, WSAGetLastError());603}604}605606JNIEXPORT jint JNICALL607Java_sun_nio_ch_Net_getInterface6(JNIEnv* env, jobject this, jobject fdo)608{609DWORD index;610int arglen = sizeof(index);611int n;612613n = getsockopt(fdval(env, fdo), IPPROTO_IPV6, IPV6_MULTICAST_IF, (void*)&index, &arglen);614if (n == SOCKET_ERROR) {615handleSocketError(env, WSAGetLastError());616return -1;617}618return (jint)index;619}620621JNIEXPORT void JNICALL622Java_sun_nio_ch_Net_shutdown(JNIEnv *env, jclass cl, jobject fdo, jint jhow) {623int how = (jhow == sun_nio_ch_Net_SHUT_RD) ? SD_RECEIVE :624(jhow == sun_nio_ch_Net_SHUT_WR) ? SD_SEND : SD_BOTH;625if (shutdown(fdval(env, fdo), how) == SOCKET_ERROR) {626NET_ThrowNew(env, WSAGetLastError(), "shutdown");627}628}629630JNIEXPORT jint JNICALL631Java_sun_nio_ch_Net_available(JNIEnv *env, jclass cl, jobject fdo)632{633int count = 0;634if (NET_SocketAvailable(fdval(env, fdo), &count) != 0) {635handleSocketError(env, WSAGetLastError());636return IOS_THROWN;637}638return (jint) count;639}640641JNIEXPORT jint JNICALL642Java_sun_nio_ch_Net_poll(JNIEnv* env, jclass this, jobject fdo, jint events, jlong timeout)643{644int rv;645int revents = 0;646struct timeval t;647fd_set rd, wr, ex;648jint fd = fdval(env, fdo);649650FD_ZERO(&rd);651FD_ZERO(&wr);652FD_ZERO(&ex);653if (events & POLLIN) {654FD_SET(fd, &rd);655}656if (events & POLLOUT) {657FD_SET(fd, &wr);658}659FD_SET(fd, &ex);660661if (timeout >= 0) {662t.tv_sec = (long)(timeout / 1000);663t.tv_usec = (timeout % 1000) * 1000;664}665666rv = select(fd+1, &rd, &wr, &ex, (timeout >= 0) ? &t : NULL);667668/* save last winsock error */669if (rv == SOCKET_ERROR) {670handleSocketError(env, WSAGetLastError());671return IOS_THROWN;672} else if (rv >= 0) {673rv = 0;674if (FD_ISSET(fd, &rd)) {675rv |= POLLIN;676}677if (FD_ISSET(fd, &wr)) {678rv |= POLLOUT;679}680if (FD_ISSET(fd, &ex)) {681rv |= POLLERR;682}683}684return rv;685}686687JNIEXPORT jboolean JNICALL688Java_sun_nio_ch_Net_pollConnect(JNIEnv* env, jclass this, jobject fdo, jlong timeout)689{690int optError = 0;691int result;692int n = sizeof(int);693jint fd = fdval(env, fdo);694fd_set wr, ex;695struct timeval t;696697FD_ZERO(&wr);698FD_ZERO(&ex);699FD_SET((u_int)fd, &wr);700FD_SET((u_int)fd, &ex);701702if (timeout >= 0) {703t.tv_sec = (long)(timeout / 1000);704t.tv_usec = (timeout % 1000) * 1000;705}706707result = select(fd+1, 0, &wr, &ex, (timeout >= 0) ? &t : NULL);708709if (result == SOCKET_ERROR) {710handleSocketError(env, WSAGetLastError());711return JNI_FALSE;712} else if (result == 0) {713return JNI_FALSE;714} else {715// connection established if writable and no error to check716if (FD_ISSET(fd, &wr) && !FD_ISSET(fd, &ex)) {717return JNI_TRUE;718}719result = getsockopt((SOCKET)fd,720SOL_SOCKET,721SO_ERROR,722(char *)&optError,723&n);724if (result == SOCKET_ERROR) {725int lastError = WSAGetLastError();726if (lastError != WSAEINPROGRESS) {727NET_ThrowNew(env, lastError, "getsockopt");728}729} else if (optError != NO_ERROR) {730handleSocketError(env, optError);731}732return JNI_FALSE;733}734}735736JNIEXPORT jshort JNICALL737Java_sun_nio_ch_Net_pollinValue(JNIEnv *env, jclass this)738{739return (jshort)POLLIN;740}741742JNIEXPORT jshort JNICALL743Java_sun_nio_ch_Net_polloutValue(JNIEnv *env, jclass this)744{745return (jshort)POLLOUT;746}747748JNIEXPORT jshort JNICALL749Java_sun_nio_ch_Net_pollerrValue(JNIEnv *env, jclass this)750{751return (jshort)POLLERR;752}753754JNIEXPORT jshort JNICALL755Java_sun_nio_ch_Net_pollhupValue(JNIEnv *env, jclass this)756{757return (jshort)POLLHUP;758}759760JNIEXPORT jshort JNICALL761Java_sun_nio_ch_Net_pollnvalValue(JNIEnv *env, jclass this)762{763return (jshort)POLLNVAL;764}765766JNIEXPORT jshort JNICALL767Java_sun_nio_ch_Net_pollconnValue(JNIEnv *env, jclass this)768{769return (jshort)POLLOUT;770}771772JNIEXPORT jint JNICALL773Java_sun_nio_ch_Net_sendOOB(JNIEnv* env, jclass this, jobject fdo, jbyte b)774{775int n = send(fdval(env, fdo), (const char*)&b, 1, MSG_OOB);776if (n == SOCKET_ERROR) {777if (WSAGetLastError() == WSAEWOULDBLOCK) {778return IOS_UNAVAILABLE;779} else {780JNU_ThrowIOExceptionWithLastError(env, "send failed");781return IOS_THROWN;782}783} else {784return n;785}786}787788JNIEXPORT jboolean JNICALL789Java_sun_nio_ch_Net_discardOOB(JNIEnv* env, jclass clazz, jobject fdo)790{791char buf[8];792jboolean discarded = JNI_FALSE;793for (;;) {794int n = recv(fdval(env, fdo), (char*)&buf, sizeof(buf), MSG_OOB);795if (n == SOCKET_ERROR) {796if (WSAGetLastError() != WSAEWOULDBLOCK) {797JNU_ThrowIOExceptionWithLastError(env, "recv failed");798}799return discarded;800}801if (n <= 0)802return discarded;803if (n < (int)sizeof(buf))804return JNI_TRUE;805discarded = JNI_TRUE;806}807}808809810