Path: blob/aarch64-shenandoah-jdk8u272-b10/jdk/src/windows/native/sun/nio/ch/Net.c
32288 views
/*1* Copyright (c) 2001, 2015, 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>27#include "jni.h"28#include "jni_util.h"29#include "jvm.h"30#include "jlong.h"3132#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 */5152typedef struct my_ip_mreq_source {53IN_ADDR imr_multiaddr;54IN_ADDR imr_sourceaddr;55IN_ADDR imr_interface;56};5758typedef struct 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}787980JNIEXPORT void JNICALL81Java_sun_nio_ch_Net_initIDs(JNIEnv *env, jclass clazz)82{83/* nothing to do */84}8586JNIEXPORT jboolean JNICALL87Java_sun_nio_ch_Net_isIPv6Available0(JNIEnv* env, jclass cl)88{89/*90* Return true if Windows Vista or newer, and IPv6 is configured91*/92OSVERSIONINFO ver;93ver.dwOSVersionInfoSize = sizeof(ver);94GetVersionEx(&ver);95if ((ver.dwPlatformId == VER_PLATFORM_WIN32_NT) &&96(ver.dwMajorVersion >= 6) && ipv6_available())97{98return JNI_TRUE;99}100return JNI_FALSE;101}102103JNIEXPORT jint JNICALL104Java_sun_nio_ch_Net_isExclusiveBindAvailable(JNIEnv *env, jclass clazz) {105OSVERSIONINFO ver;106int version;107ver.dwOSVersionInfoSize = sizeof(ver);108GetVersionEx(&ver);109version = ver.dwMajorVersion * 10 + ver.dwMinorVersion;110//if os <= xp exclusive binding is off by default111return version >= 60 ? 1 : 0;112}113114115JNIEXPORT jboolean JNICALL116Java_sun_nio_ch_Net_canIPv6SocketJoinIPv4Group0(JNIEnv* env, jclass cl)117{118return JNI_FALSE;119}120121JNIEXPORT jboolean JNICALL122Java_sun_nio_ch_Net_canJoin6WithIPv4Group0(JNIEnv* env, jclass cl)123{124return JNI_FALSE;125}126127JNIEXPORT jint JNICALL128Java_sun_nio_ch_Net_socket0(JNIEnv *env, jclass cl, jboolean preferIPv6,129jboolean stream, jboolean reuse, jboolean fastLoopback)130{131SOCKET s;132int domain = (preferIPv6) ? AF_INET6 : AF_INET;133134s = socket(domain, (stream ? SOCK_STREAM : SOCK_DGRAM), 0);135if (s != INVALID_SOCKET) {136SetHandleInformation((HANDLE)s, HANDLE_FLAG_INHERIT, 0);137138/* IPV6_V6ONLY is true by default */139if (domain == AF_INET6) {140int opt = 0;141setsockopt(s, IPPROTO_IPV6, IPV6_V6ONLY,142(const char *)&opt, sizeof(opt));143}144145/* Disable WSAECONNRESET errors for initially unconnected UDP sockets */146if (!stream) {147setConnectionReset(s, FALSE);148}149150} else {151NET_ThrowNew(env, WSAGetLastError(), "socket");152}153154if (stream && fastLoopback) {155static int loopback_available = 1;156if (loopback_available) {157int rv = NET_EnableFastTcpLoopback((jint)s);158if (rv) {159if (rv == WSAEOPNOTSUPP || rv == WSAEINVAL) {160loopback_available = 0;161} else {162NET_ThrowNew(env, rv, "fastLoopback");163}164}165}166}167168return (jint)s;169}170171JNIEXPORT void JNICALL172Java_sun_nio_ch_Net_bind0(JNIEnv *env, jclass clazz, jobject fdo, jboolean preferIPv6,173jboolean isExclBind, jobject iao, jint port)174{175SOCKETADDRESS sa;176int rv;177int sa_len;178179if (NET_InetAddressToSockaddr(env, iao, port, (struct sockaddr *)&sa, &sa_len, preferIPv6) != 0) {180return;181}182183rv = NET_WinBind(fdval(env, fdo), (struct sockaddr *)&sa, sa_len, isExclBind);184if (rv == SOCKET_ERROR)185NET_ThrowNew(env, WSAGetLastError(), "bind");186}187188JNIEXPORT void JNICALL189Java_sun_nio_ch_Net_listen(JNIEnv *env, jclass cl, jobject fdo, jint backlog)190{191if (listen(fdval(env,fdo), backlog) == SOCKET_ERROR) {192NET_ThrowNew(env, WSAGetLastError(), "listen");193}194}195196197JNIEXPORT jint JNICALL198Java_sun_nio_ch_Net_connect0(JNIEnv *env, jclass clazz, jboolean preferIPv6, jobject fdo,199jobject iao, jint port)200{201SOCKETADDRESS sa;202int rv;203int sa_len;204SOCKET s = (SOCKET)fdval(env, fdo);205206if (NET_InetAddressToSockaddr(env, iao, port, (struct sockaddr *)&sa, &sa_len, preferIPv6) != 0) {207return IOS_THROWN;208}209210rv = connect(s, (struct sockaddr *)&sa, sa_len);211if (rv != 0) {212int err = WSAGetLastError();213if (err == WSAEINPROGRESS || err == WSAEWOULDBLOCK) {214return IOS_UNAVAILABLE;215}216NET_ThrowNew(env, err, "connect");217return IOS_THROWN;218} else {219/* Enable WSAECONNRESET errors when a UDP socket is connected */220int type = 0, optlen = sizeof(type);221rv = getsockopt(s, SOL_SOCKET, SO_TYPE, (char*)&type, &optlen);222if (rv == 0 && type == SOCK_DGRAM) {223setConnectionReset(s, TRUE);224}225}226return 1;227}228229JNIEXPORT jint JNICALL230Java_sun_nio_ch_Net_localPort(JNIEnv *env, jclass clazz, jobject fdo)231{232SOCKETADDRESS sa;233int sa_len = sizeof(sa);234235if (getsockname(fdval(env, fdo), (struct sockaddr *)&sa, &sa_len) < 0) {236int error = WSAGetLastError();237if (error == WSAEINVAL) {238return 0;239}240NET_ThrowNew(env, error, "getsockname");241return IOS_THROWN;242}243return NET_GetPortFromSockaddr((struct sockaddr *)&sa);244}245246JNIEXPORT jobject JNICALL247Java_sun_nio_ch_Net_localInetAddress(JNIEnv *env, jclass clazz, jobject fdo)248{249SOCKETADDRESS sa;250int sa_len = sizeof(sa);251int port;252253if (getsockname(fdval(env, fdo), (struct sockaddr *)&sa, &sa_len) < 0) {254NET_ThrowNew(env, WSAGetLastError(), "getsockname");255return NULL;256}257return NET_SockaddrToInetAddress(env, (struct sockaddr *)&sa, &port);258}259260JNIEXPORT jint JNICALL261Java_sun_nio_ch_Net_remotePort(JNIEnv *env, jclass clazz, jobject fdo)262{263SOCKETADDRESS sa;264int sa_len = sizeof(sa);265266if (getpeername(fdval(env, fdo), (struct sockaddr *)&sa, &sa_len) < 0) {267int error = WSAGetLastError();268if (error == WSAEINVAL) {269return 0;270}271NET_ThrowNew(env, error, "getsockname");272return IOS_THROWN;273}274return NET_GetPortFromSockaddr((struct sockaddr *)&sa);275}276277JNIEXPORT jobject JNICALL278Java_sun_nio_ch_Net_remoteInetAddress(JNIEnv *env, jclass clazz, jobject fdo)279{280SOCKETADDRESS sa;281int sa_len = sizeof(sa);282int port;283284if (getpeername(fdval(env, fdo), (struct sockaddr *)&sa, &sa_len) < 0) {285NET_ThrowNew(env, WSAGetLastError(), "getsockname");286return NULL;287}288return NET_SockaddrToInetAddress(env, (struct sockaddr *)&sa, &port);289}290291JNIEXPORT jint JNICALL292Java_sun_nio_ch_Net_getIntOption0(JNIEnv *env, jclass clazz, jobject fdo,293jboolean mayNeedConversion, jint level, jint opt)294{295int result = 0;296struct linger linger;297char *arg;298int arglen, n;299300if (level == SOL_SOCKET && opt == SO_LINGER) {301arg = (char *)&linger;302arglen = sizeof(linger);303} else {304arg = (char *)&result;305arglen = sizeof(result);306}307308/**309* HACK: IP_TOS is deprecated on Windows and querying the option310* returns a protocol error. NET_GetSockOpt handles this and uses311* a fallback mechanism. Same applies to IPV6_TCLASS312*/313if ((level == IPPROTO_IP && opt == IP_TOS) || (level == IPPROTO_IPV6 && opt == IPV6_TCLASS)) {314mayNeedConversion = JNI_TRUE;315}316317if (mayNeedConversion) {318n = NET_GetSockOpt(fdval(env, fdo), level, opt, arg, &arglen);319} else {320n = getsockopt(fdval(env, fdo), level, opt, arg, &arglen);321}322if (n < 0) {323handleSocketError(env, WSAGetLastError());324return IOS_THROWN;325}326327if (level == SOL_SOCKET && opt == SO_LINGER)328return linger.l_onoff ? linger.l_linger : -1;329else330return result;331}332333JNIEXPORT void JNICALL334Java_sun_nio_ch_Net_setIntOption0(JNIEnv *env, jclass clazz, jobject fdo,335jboolean mayNeedConversion, jint level, jint opt, jint arg, jboolean ipv6)336{337struct linger linger;338char *parg;339int arglen, n;340341if (level == SOL_SOCKET && opt == SO_LINGER) {342parg = (char *)&linger;343arglen = sizeof(linger);344if (arg >= 0) {345linger.l_onoff = 1;346linger.l_linger = (unsigned short)arg;347} else {348linger.l_onoff = 0;349linger.l_linger = 0;350}351} else {352parg = (char *)&arg;353arglen = sizeof(arg);354}355356if (level == IPPROTO_IPV6 && opt == IPV6_TCLASS) {357/* No op */358return;359}360361if (mayNeedConversion) {362n = NET_SetSockOpt(fdval(env, fdo), level, opt, parg, arglen);363} else {364n = setsockopt(fdval(env, fdo), level, opt, parg, arglen);365}366if (n < 0)367handleSocketError(env, WSAGetLastError());368}369370JNIEXPORT jint JNICALL371Java_sun_nio_ch_Net_joinOrDrop4(JNIEnv *env, jobject this, jboolean join, jobject fdo,372jint group, jint interf, jint source)373{374struct ip_mreq mreq;375struct my_ip_mreq_source mreq_source;376int opt, n, optlen;377void* optval;378379if (source == 0) {380mreq.imr_multiaddr.s_addr = htonl(group);381mreq.imr_interface.s_addr = htonl(interf);382opt = (join) ? IP_ADD_MEMBERSHIP : IP_DROP_MEMBERSHIP;383optval = (void*)&mreq;384optlen = sizeof(mreq);385} else {386mreq_source.imr_multiaddr.s_addr = htonl(group);387mreq_source.imr_sourceaddr.s_addr = htonl(source);388mreq_source.imr_interface.s_addr = htonl(interf);389opt = (join) ? IP_ADD_SOURCE_MEMBERSHIP : IP_DROP_SOURCE_MEMBERSHIP;390optval = (void*)&mreq_source;391optlen = sizeof(mreq_source);392}393394n = setsockopt(fdval(env,fdo), IPPROTO_IP, opt, optval, optlen);395if (n < 0) {396if (join && (WSAGetLastError() == WSAENOPROTOOPT))397return IOS_UNAVAILABLE;398handleSocketError(env, WSAGetLastError());399}400return 0;401}402403JNIEXPORT jint JNICALL404Java_sun_nio_ch_Net_blockOrUnblock4(JNIEnv *env, jobject this, jboolean block, jobject fdo,405jint group, jint interf, jint source)406{407struct my_ip_mreq_source mreq_source;408int n;409int opt = (block) ? IP_BLOCK_SOURCE : IP_UNBLOCK_SOURCE;410411mreq_source.imr_multiaddr.s_addr = htonl(group);412mreq_source.imr_sourceaddr.s_addr = htonl(source);413mreq_source.imr_interface.s_addr = htonl(interf);414415n = setsockopt(fdval(env,fdo), IPPROTO_IP, opt,416(void*)&mreq_source, sizeof(mreq_source));417if (n < 0) {418if (block && (WSAGetLastError() == WSAENOPROTOOPT))419return IOS_UNAVAILABLE;420handleSocketError(env, WSAGetLastError());421}422return 0;423}424425/**426* Call setsockopt with a IPPROTO_IPV6 level socket option427* and a group_source_req structure as the option value. The428* given IPv6 group, interface index, and IPv6 source address429* are copied into the structure.430*/431static int setGroupSourceReqOption(JNIEnv* env,432jobject fdo,433int opt,434jbyteArray group,435jint index,436jbyteArray source)437{438struct my_group_source_req req;439struct sockaddr_in6* sin6;440441req.gsr_interface = (ULONG)index;442443sin6 = (struct sockaddr_in6*)&(req.gsr_group);444sin6->sin6_family = AF_INET6;445COPY_INET6_ADDRESS(env, group, (jbyte*)&(sin6->sin6_addr));446447sin6 = (struct sockaddr_in6*)&(req.gsr_source);448sin6->sin6_family = AF_INET6;449COPY_INET6_ADDRESS(env, source, (jbyte*)&(sin6->sin6_addr));450451return setsockopt(fdval(env,fdo), IPPROTO_IPV6, opt, (void*)&req, sizeof(req));452}453454JNIEXPORT jint JNICALL455Java_sun_nio_ch_Net_joinOrDrop6(JNIEnv *env, jobject this, jboolean join, jobject fdo,456jbyteArray group, jint index, jbyteArray source)457{458struct ipv6_mreq mreq6;459int n;460461if (source == NULL) {462int opt = (join) ? IPV6_ADD_MEMBERSHIP : IPV6_DROP_MEMBERSHIP;463COPY_INET6_ADDRESS(env, group, (jbyte*)&(mreq6.ipv6mr_multiaddr));464mreq6.ipv6mr_interface = (int)index;465n = setsockopt(fdval(env,fdo), IPPROTO_IPV6, opt,466(void*)&mreq6, sizeof(mreq6));467} else {468int opt = (join) ? MCAST_JOIN_SOURCE_GROUP : MCAST_LEAVE_SOURCE_GROUP;469n = setGroupSourceReqOption(env, fdo, opt, group, index, source);470}471472if (n < 0) {473handleSocketError(env, errno);474}475return 0;476}477478JNIEXPORT jint JNICALL479Java_sun_nio_ch_Net_blockOrUnblock6(JNIEnv *env, jobject this, jboolean block, jobject fdo,480jbyteArray group, jint index, jbyteArray source)481{482int opt = (block) ? MCAST_BLOCK_SOURCE : MCAST_UNBLOCK_SOURCE;483int n = setGroupSourceReqOption(env, fdo, opt, group, index, source);484if (n < 0) {485handleSocketError(env, errno);486}487return 0;488}489490JNIEXPORT void JNICALL491Java_sun_nio_ch_Net_setInterface4(JNIEnv* env, jobject this, jobject fdo, jint interf)492{493struct in_addr in;494int arglen = sizeof(struct in_addr);495int n;496497in.s_addr = htonl(interf);498499n = setsockopt(fdval(env, fdo), IPPROTO_IP, IP_MULTICAST_IF,500(void*)&(in.s_addr), arglen);501if (n < 0) {502handleSocketError(env, WSAGetLastError());503}504}505506JNIEXPORT jint JNICALL507Java_sun_nio_ch_Net_getInterface4(JNIEnv* env, jobject this, jobject fdo)508{509struct in_addr in;510int arglen = sizeof(struct in_addr);511int n;512513n = getsockopt(fdval(env, fdo), IPPROTO_IP, IP_MULTICAST_IF, (void*)&in, &arglen);514if (n < 0) {515handleSocketError(env, WSAGetLastError());516return IOS_THROWN;517}518return ntohl(in.s_addr);519}520521JNIEXPORT void JNICALL522Java_sun_nio_ch_Net_setInterface6(JNIEnv* env, jobject this, jobject fdo, jint index)523{524int value = (jint)index;525int arglen = sizeof(value);526int n;527528n = setsockopt(fdval(env, fdo), IPPROTO_IPV6, IPV6_MULTICAST_IF,529(void*)&(index), arglen);530if (n < 0) {531handleSocketError(env, errno);532}533}534535JNIEXPORT jint JNICALL536Java_sun_nio_ch_Net_getInterface6(JNIEnv* env, jobject this, jobject fdo)537{538int index;539int arglen = sizeof(index);540int n;541542n = getsockopt(fdval(env, fdo), IPPROTO_IPV6, IPV6_MULTICAST_IF, (void*)&index, &arglen);543if (n < 0) {544handleSocketError(env, errno);545return -1;546}547return (jint)index;548}549550JNIEXPORT void JNICALL551Java_sun_nio_ch_Net_shutdown(JNIEnv *env, jclass cl, jobject fdo, jint jhow) {552int how = (jhow == sun_nio_ch_Net_SHUT_RD) ? SD_RECEIVE :553(jhow == sun_nio_ch_Net_SHUT_WR) ? SD_SEND : SD_BOTH;554if (shutdown(fdval(env, fdo), how) == SOCKET_ERROR) {555NET_ThrowNew(env, WSAGetLastError(), "shutdown");556}557}558559JNIEXPORT jint JNICALL560Java_sun_nio_ch_Net_poll(JNIEnv* env, jclass this, jobject fdo, jint events, jlong timeout)561{562int rv;563int revents = 0;564struct timeval t;565int lastError = 0;566fd_set rd, wr, ex;567jint fd = fdval(env, fdo);568569t.tv_sec = timeout / 1000;570t.tv_usec = (timeout % 1000) * 1000;571572FD_ZERO(&rd);573FD_ZERO(&wr);574FD_ZERO(&ex);575if (events & POLLIN) {576FD_SET(fd, &rd);577}578if (events & POLLOUT ||579events & POLLCONN) {580FD_SET(fd, &wr);581}582FD_SET(fd, &ex);583584rv = select(fd+1, &rd, &wr, &ex, &t);585586/* save last winsock error */587if (rv == SOCKET_ERROR) {588handleSocketError(env, lastError);589return IOS_THROWN;590} else if (rv >= 0) {591rv = 0;592if (FD_ISSET(fd, &rd)) {593rv |= POLLIN;594}595if (FD_ISSET(fd, &wr)) {596rv |= POLLOUT;597}598if (FD_ISSET(fd, &ex)) {599rv |= POLLERR;600}601}602return rv;603}604605JNIEXPORT jshort JNICALL606Java_sun_nio_ch_Net_pollinValue(JNIEnv *env, jclass this)607{608return (jshort)POLLIN;609}610611JNIEXPORT jshort JNICALL612Java_sun_nio_ch_Net_polloutValue(JNIEnv *env, jclass this)613{614return (jshort)POLLOUT;615}616617JNIEXPORT jshort JNICALL618Java_sun_nio_ch_Net_pollerrValue(JNIEnv *env, jclass this)619{620return (jshort)POLLERR;621}622623JNIEXPORT jshort JNICALL624Java_sun_nio_ch_Net_pollhupValue(JNIEnv *env, jclass this)625{626return (jshort)POLLHUP;627}628629JNIEXPORT jshort JNICALL630Java_sun_nio_ch_Net_pollnvalValue(JNIEnv *env, jclass this)631{632return (jshort)POLLNVAL;633}634635JNIEXPORT jshort JNICALL636Java_sun_nio_ch_Net_pollconnValue(JNIEnv *env, jclass this)637{638return (jshort)POLLCONN;639}640641642