--- /dev/null 2018-06-04 09:38:58.268495706 -0700 +++ new/src/jdk.net/linux/native/libextnet/LinuxRdmaSocketDispatcherImpl.c 2018-06-04 11:02:26.185275316 -0700 @@ -0,0 +1,181 @@ +/* + * Copyright (c) 2000, 2018, Oracle and/or its affiliates. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ + +#include "jni.h" +#include "jni_util.h" +#include "jvm.h" +#include "jlong.h" +#include "nio_util.h" +#include "rdma_ch_LinuxRdmaSocketDispatcherImpl.h" +#include "java_lang_Long.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "nio.h" + +static int preCloseFD = -1; /* File descriptor to which we dup other fd's + before closing them for real */ +static jfieldID fd_fdID; + +jint +convertReturnVal(JNIEnv *env, jint n, jboolean reading) +{ + if (n > 0) + return n; + else if (n == 0) { + if (reading) { + return IOS_EOF; + } else { + return 0; + } + } + else if (errno == EAGAIN) + return IOS_UNAVAILABLE; + else if (errno == EINTR) + return IOS_INTERRUPTED; + else { + const char *msg = reading ? "Read failed" : "Write failed"; + JNU_ThrowIOExceptionWithLastError(env, msg); + return IOS_THROWN; + } +} + +jlong +convertLongReturnVal(JNIEnv *env, jlong n, jboolean reading) +{ + if (n > 0) + return n; + else if (n == 0) { + if (reading) { + return IOS_EOF; + } else { + return 0; + } + } + else if (errno == EAGAIN) + return IOS_UNAVAILABLE; + else if (errno == EINTR) + return IOS_INTERRUPTED; + else { + const char *msg = reading ? "Read failed" : "Write failed"; + JNU_ThrowIOExceptionWithLastError(env, msg); + return IOS_THROWN; + } +} + +JNIEXPORT void JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_init(JNIEnv *env, jclass cl) +{ + CHECK_NULL(cl = (*env)->FindClass(env, "java/io/FileDescriptor")); + CHECK_NULL(fd_fdID = (*env)->GetFieldID(env, cl, "fd", "I")); + + int sp[2]; + if (socketpair(PF_UNIX, SOCK_STREAM, 0, sp) < 0) { + JNU_ThrowIOExceptionWithLastError(env, "socketpair failed"); + return; + } + preCloseFD = sp[0]; + rclose(sp[1]); +} + +JNIEXPORT jint JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_read0(JNIEnv *env, jclass clazz, + jobject fdo, jlong address, jint len) +{ + jint fd = (*env)->GetIntField(env, fdo, fd_fdID); + void *buf = (void *)jlong_to_ptr(address); + return convertReturnVal(env, rread(fd, buf, len), JNI_TRUE); +} + +JNIEXPORT jlong JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_readv0(JNIEnv *env, jclass clazz, + jobject fdo, jlong address, jint len) +{ + jint fd = (*env)->GetIntField(env, fdo, fd_fdID); + struct iovec *iov = (struct iovec *)jlong_to_ptr(address); + return convertLongReturnVal(env, rreadv(fd, iov, len), JNI_TRUE); +} + +JNIEXPORT jint JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_write0(JNIEnv *env, jclass clazz, + jobject fdo, jlong address, jint len) +{ + jint fd = (*env)->GetIntField(env, fdo, fd_fdID); + void *buf = (void *)jlong_to_ptr(address); + return convertReturnVal(env, rwrite(fd, buf, len), JNI_FALSE); +} + +JNIEXPORT jlong JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_writev0(JNIEnv *env, jclass clazz, + jobject fdo, jlong address, jint len) +{ + jint fd = (*env)->GetIntField(env, fdo, fd_fdID); + struct iovec *iov = (struct iovec *)jlong_to_ptr(address); + return convertLongReturnVal(env, rwritev(fd, iov, len), JNI_FALSE); +} + +static jlong +handle(JNIEnv *env, jlong rv, char *msg) +{ + if (rv >= 0) + return rv; + if (errno == EINTR) + return IOS_INTERRUPTED; + JNU_ThrowIOExceptionWithLastError(env, msg); + return IOS_THROWN; +} + +static void closeFileDescriptor(JNIEnv *env, int fd) { + if (fd != -1) { + int result = rclose(fd); + if (result < 0) + JNU_ThrowIOExceptionWithLastError(env, "Close failed"); + } +} + +JNIEXPORT void JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_close0(JNIEnv *env, jclass clazz, jobject fdo) +{ + jint fd = (*env)->GetIntField(env, fdo, fd_fdID); + closeFileDescriptor(env, fd); +} + +JNIEXPORT void JNICALL +Java_rdma_ch_LinuxRdmaSocketDispatcherImpl_preClose0(JNIEnv *env, jclass clazz, jobject fdo) +{ +/* + jint fd = (*env)->GetIntField(env, fdo, fd_fdID); + if (preCloseFD >= 0) { + if (dup2(preCloseFD, fd) < 0) + JNU_ThrowIOExceptionWithLastError(env, "dup2 failed"); + } +*/ +}