2017-01-19 17:31:34 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2016 The Netty Project
|
|
|
|
*
|
|
|
|
* The Netty Project licenses this file to you under the Apache License,
|
|
|
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
|
|
|
* with the License. You may obtain a copy of the License at:
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Since glibc 2.8, the _GNU_SOURCE feature test macro must be defined
|
|
|
|
* (before including any header files) in order to obtain the
|
|
|
|
* definition of the ucred structure. See <a href=https://linux.die.net/man/7/unix>
|
|
|
|
*/
|
|
|
|
#define _GNU_SOURCE
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <netinet/in.h>
|
2017-12-08 01:00:52 +01:00
|
|
|
#include <sys/sendfile.h>
|
2017-01-19 17:31:34 +01:00
|
|
|
#include <linux/tcp.h> // TCP_NOTSENT_LOWAT is a linux specific define
|
2017-06-21 20:06:08 +02:00
|
|
|
|
|
|
|
#include "netty_epoll_linuxsocket.h"
|
|
|
|
#include "netty_unix_errors.h"
|
2017-01-19 17:31:34 +01:00
|
|
|
#include "netty_unix_filedescriptor.h"
|
2017-06-21 20:06:08 +02:00
|
|
|
#include "netty_unix_jni.h"
|
2017-01-19 17:31:34 +01:00
|
|
|
#include "netty_unix_socket.h"
|
|
|
|
#include "netty_unix_util.h"
|
|
|
|
|
|
|
|
// TCP_FASTOPEN is defined in linux 3.7. We define this here so older kernels can compile.
|
|
|
|
#ifndef TCP_FASTOPEN
|
|
|
|
#define TCP_FASTOPEN 23
|
|
|
|
#endif
|
|
|
|
|
2017-10-26 17:17:02 +02:00
|
|
|
// TCP_FASTOPEN_CONNECT is defined in linux 4.11. We define this here so older kernels can compile.
|
|
|
|
#ifndef TCP_FASTOPEN_CONNECT
|
|
|
|
#define TCP_FASTOPEN_CONNECT 30
|
|
|
|
#endif
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
// TCP_NOTSENT_LOWAT is defined in linux 3.12. We define this here so older kernels can compile.
|
|
|
|
#ifndef TCP_NOTSENT_LOWAT
|
|
|
|
#define TCP_NOTSENT_LOWAT 25
|
|
|
|
#endif
|
|
|
|
|
2018-09-07 20:50:51 +02:00
|
|
|
// SO_BUSY_POLL is defined in linux 3.11. We define this here so older kernels can compile.
|
|
|
|
#ifndef SO_BUSY_POLL
|
|
|
|
#define SO_BUSY_POLL 46
|
|
|
|
#endif
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
static jclass peerCredentialsClass = NULL;
|
|
|
|
static jmethodID peerCredentialsMethodId = NULL;
|
|
|
|
|
2017-12-08 01:00:52 +01:00
|
|
|
static jfieldID fileChannelFieldId = NULL;
|
|
|
|
static jfieldID transferredFieldId = NULL;
|
|
|
|
static jfieldID fdFieldId = NULL;
|
|
|
|
static jfieldID fileDescriptorFieldId = NULL;
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
// JNI Registered Methods Begin
|
|
|
|
static void netty_epoll_linuxsocket_setTcpCork(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_CORK, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpQuickAck(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_QUICKACK, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpDeferAccept(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_DEFER_ACCEPT, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpNotSentLowAt(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_NOTSENT_LOWAT, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpFastOpen(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_FASTOPEN, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
2017-10-26 17:17:02 +02:00
|
|
|
static void netty_epoll_linuxsocket_setTcpFastOpenConnect(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_FASTOPEN_CONNECT, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
static void netty_epoll_linuxsocket_setTcpKeepIdle(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_KEEPIDLE, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpKeepIntvl(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_KEEPINTVL, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpKeepCnt(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_KEEPCNT, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setTcpUserTimeout(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_TCP, TCP_USER_TIMEOUT, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void netty_epoll_linuxsocket_setIpFreeBind(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_IP, IP_FREEBIND, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
2017-06-09 14:38:26 +02:00
|
|
|
static void netty_epoll_linuxsocket_setIpTransparent(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, SOL_IP, IP_TRANSPARENT, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
2018-04-26 08:00:36 +02:00
|
|
|
static void netty_epoll_linuxsocket_setIpRecvOrigDestAddr(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, IPPROTO_IP, IP_RECVORIGDSTADDR, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
2018-09-07 20:50:51 +02:00
|
|
|
static void netty_epoll_linuxsocket_setSoBusyPoll(JNIEnv* env, jclass clazz, jint fd, jint optval) {
|
|
|
|
netty_unix_socket_setOption(env, fd, SOL_SOCKET, SO_BUSY_POLL, &optval, sizeof(optval));
|
|
|
|
}
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
static void netty_epoll_linuxsocket_setTcpMd5Sig(JNIEnv* env, jclass clazz, jint fd, jbyteArray address, jint scopeId, jbyteArray key) {
|
|
|
|
struct sockaddr_storage addr;
|
|
|
|
socklen_t addrSize;
|
|
|
|
if (netty_unix_socket_initSockaddr(env, address, scopeId, 0, &addr, &addrSize) == -1) {
|
2019-04-12 15:15:27 +02:00
|
|
|
netty_unix_errors_throwIOException(env, "Could not init sockaddr");
|
2017-01-19 17:31:34 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct tcp_md5sig md5sig;
|
|
|
|
memset(&md5sig, 0, sizeof(md5sig));
|
|
|
|
md5sig.tcpm_addr.ss_family = addr.ss_family;
|
|
|
|
|
|
|
|
struct sockaddr_in* ipaddr;
|
|
|
|
struct sockaddr_in6* ip6addr;
|
|
|
|
|
|
|
|
switch (addr.ss_family) {
|
|
|
|
case AF_INET:
|
|
|
|
ipaddr = (struct sockaddr_in*) &addr;
|
|
|
|
memcpy(&((struct sockaddr_in *) &md5sig.tcpm_addr)->sin_addr, &ipaddr->sin_addr, sizeof(ipaddr->sin_addr));
|
|
|
|
break;
|
|
|
|
case AF_INET6:
|
|
|
|
ip6addr = (struct sockaddr_in6*) &addr;
|
|
|
|
memcpy(&((struct sockaddr_in6 *) &md5sig.tcpm_addr)->sin6_addr, &ip6addr->sin6_addr, sizeof(ip6addr->sin6_addr));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key != NULL) {
|
|
|
|
md5sig.tcpm_keylen = (*env)->GetArrayLength(env, key);
|
|
|
|
(*env)->GetByteArrayRegion(env, key, 0, md5sig.tcpm_keylen, (void *) &md5sig.tcpm_key);
|
|
|
|
if ((*env)->ExceptionCheck(env) == JNI_TRUE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (setsockopt(fd, IPPROTO_TCP, TCP_MD5SIG, &md5sig, sizeof(md5sig)) < 0) {
|
2019-04-12 15:15:27 +02:00
|
|
|
netty_unix_errors_throwIOExceptionErrorNo(env, "setsockopt() failed: ", errno);
|
2017-01-19 17:31:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_getTcpKeepIdle(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_KEEPIDLE, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_getTcpKeepIntvl(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_KEEPINTVL, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_getTcpKeepCnt(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_KEEPCNT, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_getTcpUserTimeout(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_USER_TIMEOUT, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_isIpFreeBind(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_IP, IP_FREEBIND, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2017-06-09 14:38:26 +02:00
|
|
|
static jint netty_epoll_linuxsocket_isIpTransparent(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, SOL_IP, IP_TRANSPARENT, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2018-04-26 08:00:36 +02:00
|
|
|
static jint netty_epoll_linuxsocket_isIpRecvOrigDestAddr(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_IP, IP_RECVORIGDSTADDR, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2017-07-22 16:31:42 +02:00
|
|
|
static void netty_epoll_linuxsocket_getTcpInfo(JNIEnv* env, jclass clazz, jint fd, jlongArray array) {
|
2017-01-19 17:31:34 +01:00
|
|
|
struct tcp_info tcp_info;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_INFO, &tcp_info, sizeof(tcp_info)) == -1) {
|
|
|
|
return;
|
|
|
|
}
|
2017-07-22 16:31:42 +02:00
|
|
|
jlong cArray[32];
|
|
|
|
// Expand to 64 bits, then cast away unsigned-ness.
|
|
|
|
cArray[0] = (jlong) (uint64_t) tcp_info.tcpi_state;
|
|
|
|
cArray[1] = (jlong) (uint64_t) tcp_info.tcpi_ca_state;
|
|
|
|
cArray[2] = (jlong) (uint64_t) tcp_info.tcpi_retransmits;
|
|
|
|
cArray[3] = (jlong) (uint64_t) tcp_info.tcpi_probes;
|
|
|
|
cArray[4] = (jlong) (uint64_t) tcp_info.tcpi_backoff;
|
|
|
|
cArray[5] = (jlong) (uint64_t) tcp_info.tcpi_options;
|
|
|
|
cArray[6] = (jlong) (uint64_t) tcp_info.tcpi_snd_wscale;
|
|
|
|
cArray[7] = (jlong) (uint64_t) tcp_info.tcpi_rcv_wscale;
|
|
|
|
cArray[8] = (jlong) (uint64_t) tcp_info.tcpi_rto;
|
|
|
|
cArray[9] = (jlong) (uint64_t) tcp_info.tcpi_ato;
|
|
|
|
cArray[10] = (jlong) (uint64_t) tcp_info.tcpi_snd_mss;
|
|
|
|
cArray[11] = (jlong) (uint64_t) tcp_info.tcpi_rcv_mss;
|
|
|
|
cArray[12] = (jlong) (uint64_t) tcp_info.tcpi_unacked;
|
|
|
|
cArray[13] = (jlong) (uint64_t) tcp_info.tcpi_sacked;
|
|
|
|
cArray[14] = (jlong) (uint64_t) tcp_info.tcpi_lost;
|
|
|
|
cArray[15] = (jlong) (uint64_t) tcp_info.tcpi_retrans;
|
|
|
|
cArray[16] = (jlong) (uint64_t) tcp_info.tcpi_fackets;
|
|
|
|
cArray[17] = (jlong) (uint64_t) tcp_info.tcpi_last_data_sent;
|
|
|
|
cArray[18] = (jlong) (uint64_t) tcp_info.tcpi_last_ack_sent;
|
|
|
|
cArray[19] = (jlong) (uint64_t) tcp_info.tcpi_last_data_recv;
|
|
|
|
cArray[20] = (jlong) (uint64_t) tcp_info.tcpi_last_ack_recv;
|
|
|
|
cArray[21] = (jlong) (uint64_t) tcp_info.tcpi_pmtu;
|
|
|
|
cArray[22] = (jlong) (uint64_t) tcp_info.tcpi_rcv_ssthresh;
|
|
|
|
cArray[23] = (jlong) (uint64_t) tcp_info.tcpi_rtt;
|
|
|
|
cArray[24] = (jlong) (uint64_t) tcp_info.tcpi_rttvar;
|
|
|
|
cArray[25] = (jlong) (uint64_t) tcp_info.tcpi_snd_ssthresh;
|
|
|
|
cArray[26] = (jlong) (uint64_t) tcp_info.tcpi_snd_cwnd;
|
|
|
|
cArray[27] = (jlong) (uint64_t) tcp_info.tcpi_advmss;
|
|
|
|
cArray[28] = (jlong) (uint64_t) tcp_info.tcpi_reordering;
|
|
|
|
cArray[29] = (jlong) (uint64_t) tcp_info.tcpi_rcv_rtt;
|
|
|
|
cArray[30] = (jlong) (uint64_t) tcp_info.tcpi_rcv_space;
|
|
|
|
cArray[31] = (jlong) (uint64_t) tcp_info.tcpi_total_retrans;
|
|
|
|
|
|
|
|
(*env)->SetLongArrayRegion(env, array, 0, 32, cArray);
|
2017-01-19 17:31:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_isTcpCork(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_CORK, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2018-09-07 20:50:51 +02:00
|
|
|
static jint netty_epoll_linuxsocket_getSoBusyPoll(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, SOL_SOCKET, SO_BUSY_POLL, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
static jint netty_epoll_linuxsocket_getTcpDeferAccept(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_DEFER_ACCEPT, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jint netty_epoll_linuxsocket_isTcpQuickAck(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_QUICKACK, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2017-10-26 17:17:02 +02:00
|
|
|
static jint netty_epoll_linuxsocket_isTcpFastOpenConnect(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
2017-11-21 12:12:15 +01:00
|
|
|
// We call netty_unix_socket_getOption0 directly so we can handle ENOPROTOOPT by ourself.
|
|
|
|
if (netty_unix_socket_getOption0(fd, IPPROTO_TCP, TCP_FASTOPEN_CONNECT, &optval, sizeof(optval)) == -1) {
|
2017-11-14 11:16:21 +01:00
|
|
|
if (errno == ENOPROTOOPT) {
|
|
|
|
// Not supported by the system, so just return 0.
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
netty_unix_socket_getOptionHandleError(env, errno);
|
2017-10-26 17:17:02 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
2017-01-19 17:31:34 +01:00
|
|
|
static jint netty_epoll_linuxsocket_getTcpNotSentLowAt(JNIEnv* env, jclass clazz, jint fd) {
|
|
|
|
int optval;
|
|
|
|
if (netty_unix_socket_getOption(env, fd, IPPROTO_TCP, TCP_NOTSENT_LOWAT, &optval, sizeof(optval)) == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return optval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static jobject netty_epoll_linuxsocket_getPeerCredentials(JNIEnv *env, jclass clazz, jint fd) {
|
|
|
|
struct ucred credentials;
|
|
|
|
if(netty_unix_socket_getOption(env,fd, SOL_SOCKET, SO_PEERCRED, &credentials, sizeof (credentials)) == -1) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
jintArray gids = (*env)->NewIntArray(env, 1);
|
|
|
|
(*env)->SetIntArrayRegion(env, gids, 0, 1, (jint*) &credentials.gid);
|
|
|
|
return (*env)->NewObject(env, peerCredentialsClass, peerCredentialsMethodId, credentials.pid, credentials.uid, gids);
|
|
|
|
}
|
2017-12-08 01:00:52 +01:00
|
|
|
|
|
|
|
static jlong netty_epoll_linuxsocket_sendFile(JNIEnv* env, jclass clazz, jint fd, jobject fileRegion, jlong base_off, jlong off, jlong len) {
|
|
|
|
jobject fileChannel = (*env)->GetObjectField(env, fileRegion, fileChannelFieldId);
|
|
|
|
if (fileChannel == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get DefaultFileRegion.file");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
jobject fileDescriptor = (*env)->GetObjectField(env, fileChannel, fileDescriptorFieldId);
|
|
|
|
if (fileDescriptor == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get FileChannelImpl.fd");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
jint srcFd = (*env)->GetIntField(env, fileDescriptor, fdFieldId);
|
|
|
|
if (srcFd == -1) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get FileDescriptor.fd");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ssize_t res;
|
|
|
|
off_t offset = base_off + off;
|
|
|
|
int err;
|
|
|
|
do {
|
|
|
|
res = sendfile(fd, srcFd, &offset, (size_t) len);
|
|
|
|
} while (res == -1 && ((err = errno) == EINTR));
|
|
|
|
if (res < 0) {
|
|
|
|
return -err;
|
|
|
|
}
|
|
|
|
if (res > 0) {
|
|
|
|
// update the transferred field in DefaultFileRegion
|
|
|
|
(*env)->SetLongField(env, fileRegion, transferredFieldId, off + res);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
2017-01-19 17:31:34 +01:00
|
|
|
// JNI Registered Methods End
|
|
|
|
|
|
|
|
// JNI Method Registration Table Begin
|
|
|
|
static const JNINativeMethod fixed_method_table[] = {
|
|
|
|
{ "setTcpCork", "(II)V", (void *) netty_epoll_linuxsocket_setTcpCork },
|
2018-09-07 20:50:51 +02:00
|
|
|
{ "setSoBusyPoll", "(II)V", (void *) netty_epoll_linuxsocket_setSoBusyPoll },
|
2017-01-19 17:31:34 +01:00
|
|
|
{ "setTcpQuickAck", "(II)V", (void *) netty_epoll_linuxsocket_setTcpQuickAck },
|
|
|
|
{ "setTcpDeferAccept", "(II)V", (void *) netty_epoll_linuxsocket_setTcpDeferAccept },
|
|
|
|
{ "setTcpNotSentLowAt", "(II)V", (void *) netty_epoll_linuxsocket_setTcpNotSentLowAt },
|
|
|
|
{ "isTcpCork", "(I)I", (void *) netty_epoll_linuxsocket_isTcpCork },
|
2018-09-07 20:50:51 +02:00
|
|
|
{ "getSoBusyPoll", "(I)I", (void *) netty_epoll_linuxsocket_getSoBusyPoll },
|
2017-01-19 17:31:34 +01:00
|
|
|
{ "getTcpDeferAccept", "(I)I", (void *) netty_epoll_linuxsocket_getTcpDeferAccept },
|
|
|
|
{ "getTcpNotSentLowAt", "(I)I", (void *) netty_epoll_linuxsocket_getTcpNotSentLowAt },
|
|
|
|
{ "isTcpQuickAck", "(I)I", (void *) netty_epoll_linuxsocket_isTcpQuickAck },
|
|
|
|
{ "setTcpFastOpen", "(II)V", (void *) netty_epoll_linuxsocket_setTcpFastOpen },
|
2017-10-26 17:17:02 +02:00
|
|
|
{ "setTcpFastOpenConnect", "(II)V", (void *) netty_epoll_linuxsocket_setTcpFastOpenConnect },
|
|
|
|
{ "isTcpFastOpenConnect", "(I)I", (void *) netty_epoll_linuxsocket_isTcpFastOpenConnect },
|
2017-01-19 17:31:34 +01:00
|
|
|
{ "setTcpKeepIdle", "(II)V", (void *) netty_epoll_linuxsocket_setTcpKeepIdle },
|
|
|
|
{ "setTcpKeepIntvl", "(II)V", (void *) netty_epoll_linuxsocket_setTcpKeepIntvl },
|
|
|
|
{ "setTcpKeepCnt", "(II)V", (void *) netty_epoll_linuxsocket_setTcpKeepCnt },
|
|
|
|
{ "setTcpUserTimeout", "(II)V", (void *) netty_epoll_linuxsocket_setTcpUserTimeout },
|
|
|
|
{ "setIpFreeBind", "(II)V", (void *) netty_epoll_linuxsocket_setIpFreeBind },
|
2017-06-09 14:38:26 +02:00
|
|
|
{ "setIpTransparent", "(II)V", (void *) netty_epoll_linuxsocket_setIpTransparent },
|
2018-04-26 08:00:36 +02:00
|
|
|
{ "setIpRecvOrigDestAddr", "(II)V", (void *) netty_epoll_linuxsocket_setIpRecvOrigDestAddr },
|
2017-01-19 17:31:34 +01:00
|
|
|
{ "getTcpKeepIdle", "(I)I", (void *) netty_epoll_linuxsocket_getTcpKeepIdle },
|
|
|
|
{ "getTcpKeepIntvl", "(I)I", (void *) netty_epoll_linuxsocket_getTcpKeepIntvl },
|
|
|
|
{ "getTcpKeepCnt", "(I)I", (void *) netty_epoll_linuxsocket_getTcpKeepCnt },
|
|
|
|
{ "getTcpUserTimeout", "(I)I", (void *) netty_epoll_linuxsocket_getTcpUserTimeout },
|
|
|
|
{ "isIpFreeBind", "(I)I", (void *) netty_epoll_linuxsocket_isIpFreeBind },
|
2017-06-09 14:38:26 +02:00
|
|
|
{ "isIpTransparent", "(I)I", (void *) netty_epoll_linuxsocket_isIpTransparent },
|
2018-04-26 08:00:36 +02:00
|
|
|
{ "isIpRecvOrigDestAddr", "(I)I", (void *) netty_epoll_linuxsocket_isIpRecvOrigDestAddr },
|
2017-07-22 16:31:42 +02:00
|
|
|
{ "getTcpInfo", "(I[J)V", (void *) netty_epoll_linuxsocket_getTcpInfo },
|
2019-04-12 21:32:22 +02:00
|
|
|
{ "setTcpMd5Sig", "(I[BI[B)V", (void *) netty_epoll_linuxsocket_setTcpMd5Sig }
|
2017-12-08 01:00:52 +01:00
|
|
|
// "sendFile" has a dynamic signature
|
2017-01-19 17:31:34 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static const jint fixed_method_table_size = sizeof(fixed_method_table) / sizeof(fixed_method_table[0]);
|
|
|
|
|
|
|
|
static jint dynamicMethodsTableSize() {
|
2017-12-08 01:00:52 +01:00
|
|
|
return fixed_method_table_size + 2; // 2 is for the dynamic method signatures.
|
2017-01-19 17:31:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static JNINativeMethod* createDynamicMethodsTable(const char* packagePrefix) {
|
|
|
|
JNINativeMethod* dynamicMethods = malloc(sizeof(JNINativeMethod) * dynamicMethodsTableSize());
|
|
|
|
memcpy(dynamicMethods, fixed_method_table, sizeof(fixed_method_table));
|
|
|
|
JNINativeMethod* dynamicMethod = &dynamicMethods[fixed_method_table_size];
|
|
|
|
char* dynamicTypeName = netty_unix_util_prepend(packagePrefix, "io/netty/channel/unix/PeerCredentials;");
|
|
|
|
dynamicMethod->name = "getPeerCredentials";
|
|
|
|
dynamicMethod->signature = netty_unix_util_prepend("(I)L", dynamicTypeName);
|
|
|
|
dynamicMethod->fnPtr = (void *) netty_epoll_linuxsocket_getPeerCredentials;
|
|
|
|
free(dynamicTypeName);
|
2017-12-08 01:00:52 +01:00
|
|
|
|
|
|
|
++dynamicMethod;
|
|
|
|
dynamicTypeName = netty_unix_util_prepend(packagePrefix, "io/netty/channel/DefaultFileRegion;JJJ)J");
|
|
|
|
dynamicMethod->name = "sendFile";
|
|
|
|
dynamicMethod->signature = netty_unix_util_prepend("(IL", dynamicTypeName);
|
|
|
|
dynamicMethod->fnPtr = (void *) netty_epoll_linuxsocket_sendFile;
|
|
|
|
free(dynamicTypeName);
|
2017-01-19 17:31:34 +01:00
|
|
|
return dynamicMethods;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void freeDynamicMethodsTable(JNINativeMethod* dynamicMethods) {
|
|
|
|
jint fullMethodTableSize = dynamicMethodsTableSize();
|
|
|
|
jint i = fixed_method_table_size;
|
|
|
|
for (; i < fullMethodTableSize; ++i) {
|
|
|
|
free(dynamicMethods[i].signature);
|
|
|
|
}
|
|
|
|
free(dynamicMethods);
|
|
|
|
}
|
|
|
|
// JNI Method Registration Table End
|
|
|
|
|
|
|
|
jint netty_epoll_linuxsocket_JNI_OnLoad(JNIEnv* env, const char* packagePrefix) {
|
|
|
|
JNINativeMethod* dynamicMethods = createDynamicMethodsTable(packagePrefix);
|
|
|
|
if (netty_unix_util_register_natives(env,
|
|
|
|
packagePrefix,
|
|
|
|
"io/netty/channel/epoll/LinuxSocket",
|
|
|
|
dynamicMethods,
|
|
|
|
dynamicMethodsTableSize()) != 0) {
|
|
|
|
freeDynamicMethodsTable(dynamicMethods);
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
freeDynamicMethodsTable(dynamicMethods);
|
|
|
|
dynamicMethods = NULL;
|
|
|
|
|
|
|
|
char* nettyClassName = netty_unix_util_prepend(packagePrefix, "io/netty/channel/unix/PeerCredentials");
|
|
|
|
jclass localPeerCredsClass = (*env)->FindClass(env, nettyClassName);
|
|
|
|
free(nettyClassName);
|
|
|
|
nettyClassName = NULL;
|
|
|
|
if (localPeerCredsClass == NULL) {
|
|
|
|
// pending exception...
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
peerCredentialsClass = (jclass) (*env)->NewGlobalRef(env, localPeerCredsClass);
|
|
|
|
if (peerCredentialsClass == NULL) {
|
|
|
|
// out-of-memory!
|
|
|
|
netty_unix_errors_throwOutOfMemoryError(env);
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
peerCredentialsMethodId = (*env)->GetMethodID(env, peerCredentialsClass, "<init>", "(II[I)V");
|
|
|
|
if (peerCredentialsMethodId == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get method ID: PeerCredentials.<init>(int, int, int[])");
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
|
2017-12-08 01:00:52 +01:00
|
|
|
nettyClassName = netty_unix_util_prepend(packagePrefix, "io/netty/channel/DefaultFileRegion");
|
|
|
|
jclass fileRegionCls = (*env)->FindClass(env, nettyClassName);
|
|
|
|
free(nettyClassName);
|
|
|
|
nettyClassName = NULL;
|
|
|
|
if (fileRegionCls == NULL) {
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
fileChannelFieldId = (*env)->GetFieldID(env, fileRegionCls, "file", "Ljava/nio/channels/FileChannel;");
|
|
|
|
if (fileChannelFieldId == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get field ID: DefaultFileRegion.file");
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
transferredFieldId = (*env)->GetFieldID(env, fileRegionCls, "transferred", "J");
|
|
|
|
if (transferredFieldId == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get field ID: DefaultFileRegion.transferred");
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
jclass fileChannelCls = (*env)->FindClass(env, "sun/nio/ch/FileChannelImpl");
|
|
|
|
if (fileChannelCls == NULL) {
|
|
|
|
// pending exception...
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
fileDescriptorFieldId = (*env)->GetFieldID(env, fileChannelCls, "fd", "Ljava/io/FileDescriptor;");
|
|
|
|
if (fileDescriptorFieldId == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get field ID: FileChannelImpl.fd");
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
jclass fileDescriptorCls = (*env)->FindClass(env, "java/io/FileDescriptor");
|
|
|
|
if (fileDescriptorCls == NULL) {
|
|
|
|
// pending exception...
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
fdFieldId = (*env)->GetFieldID(env, fileDescriptorCls, "fd", "I");
|
|
|
|
if (fdFieldId == NULL) {
|
|
|
|
netty_unix_errors_throwRuntimeException(env, "failed to get field ID: FileDescriptor.fd");
|
|
|
|
return JNI_ERR;
|
|
|
|
}
|
|
|
|
|
2017-06-21 20:06:08 +02:00
|
|
|
return NETTY_JNI_VERSION;
|
2017-01-19 17:31:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void netty_epoll_linuxsocket_JNI_OnUnLoad(JNIEnv* env) {
|
|
|
|
if (peerCredentialsClass != NULL) {
|
|
|
|
(*env)->DeleteGlobalRef(env, peerCredentialsClass);
|
|
|
|
peerCredentialsClass = NULL;
|
|
|
|
}
|
|
|
|
}
|