Path: blob/master/src/java.base/windows/native/libnio/ch/UnixDomainSockets.c
41134 views
/*1* Copyright (c) 2020, 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 "java_net_InetAddress.h"37#include "sun_nio_ch_Net.h"38#include "sun_nio_ch_PollArrayWrapper.h"3940jbyteArray sockaddrToUnixAddressBytes(JNIEnv *env, struct sockaddr_un *sa, socklen_t len)41{42if (sa->sun_family == AF_UNIX) {43int namelen = (int)strlen(sa->sun_path);44jbyteArray name = (*env)->NewByteArray(env, namelen);45if (name != NULL) {46(*env)->SetByteArrayRegion(env, name, 0, namelen, (jbyte*)sa->sun_path);47if ((*env)->ExceptionOccurred(env)) {48return NULL;49}50}51return name;52}53return NULL;54}5556jint unixSocketAddressToSockaddr(JNIEnv *env, jbyteArray addr, struct sockaddr_un *sa, int *len)57{58memset(sa, 0, sizeof(struct sockaddr_un));59sa->sun_family = AF_UNIX;60if (addr == 0L) {61/* Do explicit bind on Windows */62*len = (int)(offsetof(struct sockaddr_un, sun_path));63return 0;64}65int ret;66jboolean isCopy;67char *pname = (*env)->GetByteArrayElements(env, addr, &isCopy);68if (pname == NULL) {69JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Unix domain path not present");70return -1;71}7273size_t name_len = (size_t)(*env)->GetArrayLength(env, addr);74if (name_len > MAX_UNIX_DOMAIN_PATH_LEN) {75JNU_ThrowByName(env, JNU_JAVANETPKG "SocketException", "Unix domain path too long");76ret = -1;77} else {78strncpy(sa->sun_path, pname, name_len);79*len = (int)(offsetof(struct sockaddr_un, sun_path) + name_len);80ret = 0;81}82(*env)->ReleaseByteArrayElements(env, addr, pname, JNI_ABORT);83return ret;84}8586JNIEXPORT jboolean JNICALL87Java_sun_nio_ch_UnixDomainSockets_socketSupported(JNIEnv *env, jclass cl)88{89SOCKET s = socket(PF_UNIX, SOCK_STREAM, 0);90if (s == INVALID_SOCKET) {91return JNI_FALSE;92}93closesocket(s);94return JNI_TRUE;95}9697JNIEXPORT jint JNICALL98Java_sun_nio_ch_UnixDomainSockets_socket0(JNIEnv *env, jclass cl)99{100SOCKET s = socket(PF_UNIX, SOCK_STREAM, 0);101if (s == INVALID_SOCKET) {102return handleSocketError(env, WSAGetLastError());103}104SetHandleInformation((HANDLE)s, HANDLE_FLAG_INHERIT, 0);105return (int)s;106}107108/**109* Windows does not support auto bind. So, the windows version of unixSocketAddressToSockaddr110* looks out for a null 'uaddr' and handles it specially111*/112JNIEXPORT void JNICALL113Java_sun_nio_ch_UnixDomainSockets_bind0(JNIEnv *env, jclass clazz, jobject fdo, jbyteArray addr)114{115struct sockaddr_un sa;116int sa_len = 0;117int rv = 0;118119if (unixSocketAddressToSockaddr(env, addr, &sa, &sa_len) != 0)120return;121122rv = bind(fdval(env, fdo), (struct sockaddr *)&sa, sa_len);123if (rv == SOCKET_ERROR) {124int err = WSAGetLastError();125NET_ThrowNew(env, err, "bind");126}127}128129JNIEXPORT jint JNICALL130Java_sun_nio_ch_UnixDomainSockets_connect0(JNIEnv *env, jclass clazz, jobject fdo, jbyteArray addr)131{132struct sockaddr_un sa;133int sa_len = 0;134int rv;135136if (unixSocketAddressToSockaddr(env, addr, &sa, &sa_len) != 0) {137return IOS_THROWN;138}139140rv = connect(fdval(env, fdo), (const struct sockaddr *)&sa, sa_len);141if (rv != 0) {142int err = WSAGetLastError();143if (err == WSAEINPROGRESS || err == WSAEWOULDBLOCK) {144return IOS_UNAVAILABLE;145}146NET_ThrowNew(env, err, "connect");147return IOS_THROWN;148}149return 1;150}151152JNIEXPORT jint JNICALL153Java_sun_nio_ch_UnixDomainSockets_accept0(JNIEnv *env, jclass clazz, jobject fdo, jobject newfdo,154jobjectArray array)155{156jint fd = fdval(env, fdo);157jint newfd;158struct sockaddr_un sa;159socklen_t sa_len = sizeof(sa);160jbyteArray address;161162memset((char *)&sa, 0, sizeof(sa));163newfd = (jint) accept(fd, (struct sockaddr *)&sa, &sa_len);164if (newfd == INVALID_SOCKET) {165int theErr = (jint)WSAGetLastError();166if (theErr == WSAEWOULDBLOCK) {167return IOS_UNAVAILABLE;168}169JNU_ThrowIOExceptionWithLastError(env, "Accept failed");170return IOS_THROWN;171}172173SetHandleInformation((HANDLE)(UINT_PTR)newfd, HANDLE_FLAG_INHERIT, 0);174setfdval(env, newfdo, newfd);175176address = sockaddrToUnixAddressBytes(env, &sa, sa_len);177CHECK_NULL_RETURN(address, IOS_THROWN);178(*env)->SetObjectArrayElement(env, array, 0, address);179180return 1;181}182183JNIEXPORT jbyteArray JNICALL184Java_sun_nio_ch_UnixDomainSockets_localAddress0(JNIEnv *env, jclass clazz, jobject fdo)185{186struct sockaddr_un sa;187int sa_len = sizeof(sa);188189if (getsockname(fdval(env, fdo), (struct sockaddr *)&sa, &sa_len) == SOCKET_ERROR) {190JNU_ThrowIOExceptionWithLastError(env, "getsockname");191return NULL;192}193return sockaddrToUnixAddressBytes(env, &sa, sa_len);194}195196197198