2017-10-13 13:14:04 -07:00
|
|
|
/*
|
|
|
|
|
* ZeroTier SDK - Network Virtualization Everywhere
|
2018-01-08 17:05:48 -08:00
|
|
|
* Copyright (C) 2011-2018 ZeroTier, Inc. https://www.zerotier.com/
|
2017-10-13 13:14:04 -07:00
|
|
|
*
|
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program 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 for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*
|
|
|
|
|
* --
|
|
|
|
|
*
|
|
|
|
|
* You can be released from the requirements of the license by purchasing
|
|
|
|
|
* a commercial license. Buying such a license is mandatory as soon as you
|
|
|
|
|
* develop commercial closed-source software that incorporates or links
|
|
|
|
|
* directly against ZeroTier software without disclosing the source code
|
|
|
|
|
* of your own application.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @file
|
|
|
|
|
*
|
2018-07-12 16:53:40 -07:00
|
|
|
* Javs JNI wrapper for POSIX-like socket API
|
2017-11-06 13:50:20 -08:00
|
|
|
* JNI naming convention: Java_PACKAGENAME_CLASSNAME_METHODNAME
|
2017-10-13 13:14:04 -07:00
|
|
|
*/
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
#ifdef SDK_JNI
|
|
|
|
|
|
2017-10-13 13:14:04 -07:00
|
|
|
#include <sys/socket.h>
|
2017-11-21 15:53:31 -08:00
|
|
|
#include <netinet/in.h>
|
|
|
|
|
#include <arpa/inet.h>
|
2017-10-13 13:14:04 -07:00
|
|
|
|
|
|
|
|
#include "libzt.h"
|
2017-11-21 15:53:31 -08:00
|
|
|
#include "libztDefs.h"
|
|
|
|
|
|
|
|
|
|
#include <jni.h>
|
2017-10-13 13:14:04 -07:00
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
|
extern "C" {
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
namespace ZeroTier {
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// prototype
|
|
|
|
|
jobject ss2inet(JNIEnv *env, struct sockaddr_storage *src_ss);
|
|
|
|
|
int sockinet2ss(JNIEnv *env, jobject src_inet, struct sockaddr_storage *dest_ss);
|
2017-10-13 13:14:04 -07:00
|
|
|
|
|
|
|
|
/****************************************************************************/
|
2017-11-21 15:53:31 -08:00
|
|
|
/* ZeroTier service controls */
|
2017-10-13 13:14:04 -07:00
|
|
|
/****************************************************************************/
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT void JNICALL Java_zerotier_ZeroTier_start(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jstring path, jboolean blocking)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
if (path) {
|
|
|
|
|
zts_start(env->GetStringUTFChars(path, NULL), blocking);
|
|
|
|
|
}
|
2017-10-16 15:34:43 -07:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT void JNICALL Java_zerotier_ZeroTier_startjoin(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jstring path, jlong nwid)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
if (path && nwid) {
|
|
|
|
|
zts_startjoin(env->GetStringUTFChars(path, NULL), (uint64_t)nwid);
|
2017-10-16 16:45:46 -07:00
|
|
|
}
|
2017-10-16 15:34:43 -07:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT void JNICALL Java_zerotier_ZeroTier_stop(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
|
|
|
|
{
|
|
|
|
|
zts_stop();
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jboolean JNICALL Java_zerotier_ZeroTier_core_1running(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
2017-11-06 13:50:20 -08:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
return zts_core_running();
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jboolean JNICALL Java_zerotier_ZeroTier_stack_1running(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
|
|
|
|
{
|
|
|
|
|
return zts_stack_running();
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jboolean JNICALL Java_zerotier_ZeroTier_ready(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
|
|
|
|
{
|
|
|
|
|
return zts_ready();
|
2017-11-06 13:50:20 -08:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_join(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid)
|
|
|
|
|
{
|
|
|
|
|
return zts_join((uint64_t)nwid);
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_leave(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid)
|
|
|
|
|
{
|
|
|
|
|
return zts_leave((uint64_t)nwid);
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jstring JNICALL Java_zerotier_ZeroTier_get_1path(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
|
|
|
|
{
|
|
|
|
|
char pathBuf[ZT_HOME_PATH_MAX_LEN];
|
|
|
|
|
zts_get_path(pathBuf, ZT_HOME_PATH_MAX_LEN);
|
|
|
|
|
return (*env).NewStringUTF(pathBuf);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jlong JNICALL Java_zerotier_ZeroTier_get_1node_1id(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
|
|
|
|
{
|
|
|
|
|
return zts_get_node_id();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: ZT_SOCKET_API uint64_t ZTCALL zts_get_node_id_from_file(const char *filepath);
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_get_1num_1assigned_1addresses(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid)
|
|
|
|
|
{
|
|
|
|
|
return zts_get_num_assigned_addresses(nwid);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jobject JNICALL Java_zerotier_ZeroTier_get_1address_1at_1index(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid, jint index)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
2017-10-16 16:45:46 -07:00
|
|
|
struct sockaddr_storage ss;
|
2017-11-21 15:53:31 -08:00
|
|
|
int err;
|
|
|
|
|
if((err = zts_get_address_at_index(nwid, index, &ss)) < 0) {
|
|
|
|
|
return NULL;
|
2017-10-16 16:45:46 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
return ss2inet(env, &ss);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jboolean JNICALL Java_zerotier_ZeroTier_has_1address(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid)
|
|
|
|
|
{
|
|
|
|
|
return zts_has_address(nwid);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jobject JNICALL Java_zerotier_ZeroTier_get_1address(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid, jint address_family)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_storage ss;
|
|
|
|
|
int err;
|
|
|
|
|
if ((err = zts_get_address((uint64_t)nwid, &ss, address_family)) < 0) {
|
|
|
|
|
return NULL;
|
2017-10-16 16:45:46 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
return ss2inet(env, &ss);
|
2017-10-16 15:34:43 -07:00
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jobject JNICALL Java_zerotier_ZeroTier_get_6plane_addr(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid, jlong nodeId)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_storage ss;
|
|
|
|
|
zts_get_6plane_addr(&ss, nwid, nodeId);
|
|
|
|
|
return ss2inet(env, &ss);
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jobject JNICALL Java_zerotier_ZeroTier_get_rfc4193_addr(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jlong nwid, jlong nodeId)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_storage ss;
|
|
|
|
|
zts_get_rfc4193_addr(&ss, nwid, nodeId);
|
|
|
|
|
return ss2inet(env, &ss);
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jlong JNICALL Java_zerotier_ZeroTier_get_peer_count(
|
|
|
|
|
JNIEnv *env, jobject thisObj)
|
|
|
|
|
{
|
|
|
|
|
return zts_get_peer_count();
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// TODO: ZT_SOCKET_API int ZTCALL zts_get_peer_address(char *peer, const uint64_t nodeId);
|
|
|
|
|
|
|
|
|
|
/****************************************************************************/
|
|
|
|
|
/* ZeroTier Socket API */
|
|
|
|
|
/****************************************************************************/
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_socket(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint family, jint type, jint protocol)
|
|
|
|
|
{
|
|
|
|
|
return zts_socket(family, type, protocol);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_connect(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jobject addr)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_storage ss;
|
|
|
|
|
if(sockinet2ss(env, addr, &ss) < 0) {
|
|
|
|
|
return -1; // possibly invalid address format
|
|
|
|
|
// TODO: set errno
|
|
|
|
|
}
|
2018-07-12 16:53:40 -07:00
|
|
|
socklen_t addrlen = ss.ss_family == AF_INET ? sizeof(struct sockaddr_in) : sizeof(struct sockaddr_in6);
|
|
|
|
|
return zts_connect(fd, (struct sockaddr *)&ss, addrlen);
|
2017-11-21 15:53:31 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_bind(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jobject addr)
|
2017-11-06 13:50:20 -08:00
|
|
|
{
|
|
|
|
|
struct sockaddr_storage ss;
|
2017-11-21 15:53:31 -08:00
|
|
|
int err;
|
|
|
|
|
if(sockinet2ss(env, addr, &ss) < 0) {
|
|
|
|
|
return -1; // possibly invalid address format
|
|
|
|
|
// TODO: set errno
|
2017-11-06 13:50:20 -08:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
//DEBUG_TEST("RESULT => %s : %d", inet_ntoa(in4->sin_addr), ntohs(in4->sin_port));
|
2018-07-12 16:53:40 -07:00
|
|
|
socklen_t addrlen = ss.ss_family == AF_INET ? sizeof(struct sockaddr_in) : sizeof(struct sockaddr_in6);
|
2017-11-06 13:50:20 -08:00
|
|
|
err = zts_bind(fd, (struct sockaddr*)&ss, addrlen);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2017-10-16 15:34:43 -07:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_listen(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, int backlog)
|
|
|
|
|
{
|
|
|
|
|
return zts_listen(fd, backlog);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_accept(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jobject addr, jint port)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_storage ss;
|
|
|
|
|
int err;
|
|
|
|
|
socklen_t addrlen = sizeof(struct sockaddr_storage);
|
|
|
|
|
if ((err = zts_accept(fd, (struct sockaddr *)&ss, &addrlen)) < 0) {
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
addr = ss2inet(env, &ss);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-16 15:34:43 -07:00
|
|
|
#if defined(__linux__)
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_accept4(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jobject addr, jint port, jint flags)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
struct sockaddr_storage ss;
|
|
|
|
|
int err;
|
|
|
|
|
socklen_t addrlen = sizeof(struct sockaddr_storage);
|
2017-12-04 17:00:52 -08:00
|
|
|
if ((err = zts_accept4(fd, (struct sockaddr *)&ss, &addrlen, flags)) < 0) {
|
2017-11-21 15:53:31 -08:00
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
addr = ss2inet(env, &ss);
|
|
|
|
|
return err;
|
2017-10-16 15:34:43 -07:00
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_setsockopt(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jint level, jint optname, jint optval, jint optlen)
|
|
|
|
|
{
|
2017-12-04 17:00:52 -08:00
|
|
|
return zts_setsockopt(fd, level, optname, (void*)(uintptr_t)optval, optlen);
|
2017-11-21 15:53:31 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_getsockopt(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jint level, jint optname, jint optval, jint optlen)
|
|
|
|
|
{
|
2017-12-04 17:00:52 -08:00
|
|
|
return zts_getsockopt(fd, level, optname, (void*)(uintptr_t)optval, (socklen_t *)optlen);
|
2017-11-21 15:53:31 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_getsockname(JNIEnv *env, jobject thisObj,
|
|
|
|
|
jint fd, jobject ztaddr)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
|
|
|
|
struct sockaddr_in addr;
|
2017-11-21 15:53:31 -08:00
|
|
|
int err = zts_getsockname(fd, (struct sockaddr *)&addr, (socklen_t *)sizeof(struct sockaddr));
|
|
|
|
|
jfieldID fid;
|
|
|
|
|
jclass c = (*env).GetObjectClass(ztaddr);
|
|
|
|
|
if (c) {
|
|
|
|
|
fid = (*env).GetFieldID(c, "port", "I");
|
|
|
|
|
(*env).SetIntField(ztaddr, fid, addr.sin_port);
|
|
|
|
|
fid = (*env).GetFieldID(c,"_rawAddr", "J");
|
|
|
|
|
(*env).SetLongField(ztaddr, fid,addr.sin_addr.s_addr);
|
|
|
|
|
}
|
|
|
|
|
return err;
|
2017-10-16 15:34:43 -07:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_getpeername(JNIEnv *env, jobject thisObj,
|
|
|
|
|
jint fd, jobject ztaddr)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
struct sockaddr_in addr;
|
|
|
|
|
int err = zts_getpeername(fd, (struct sockaddr *)&addr, (socklen_t *)sizeof(struct sockaddr));
|
|
|
|
|
jfieldID fid;
|
|
|
|
|
jclass c = (*env).GetObjectClass( ztaddr);
|
|
|
|
|
if (c) {
|
|
|
|
|
fid = (*env).GetFieldID(c, "port", "I");
|
|
|
|
|
(*env).SetIntField(ztaddr, fid, addr.sin_port);
|
|
|
|
|
fid = (*env).GetFieldID(c,"_rawAddr", "J");
|
|
|
|
|
(*env).SetLongField(ztaddr, fid,addr.sin_addr.s_addr);
|
|
|
|
|
}
|
|
|
|
|
return err;
|
2017-10-16 15:34:43 -07:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// TODO: ZT_SOCKET_API struct hostent *zts_gethostbyname(const char *name);
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_close(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd)
|
2017-10-16 15:34:43 -07:00
|
|
|
{
|
|
|
|
|
return zts_close(fd);
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// TODO: ZT_SOCKET_API int ZTCALL zts_select(int nfds, fd_set *readfds, fd_set *writefds, fd_set *exceptfds, struct timeval *timeout);
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_fcntl(
|
|
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jint cmd, jint flags)
|
|
|
|
|
{
|
|
|
|
|
return zts_fcntl(fd, cmd, flags);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// TODO: ZT_SOCKET_API int ZTCALL zts_ioctl(int fd, unsigned long request, void *argp);
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_send(JNIEnv *env, jobject thisObj, jint fd, jarray buf, jint len, int flags)
|
|
|
|
|
{
|
|
|
|
|
jbyte *body = (*env).GetByteArrayElements((_jbyteArray *)buf, 0);
|
|
|
|
|
char * bufp = (char *)malloc(sizeof(char)*len);
|
|
|
|
|
memcpy(bufp, body, len);
|
|
|
|
|
(*env).ReleaseByteArrayElements((_jbyteArray *)buf, body, 0);
|
|
|
|
|
int written_bytes = zts_write(fd, body, len);
|
|
|
|
|
return written_bytes;
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-16 15:34:43 -07:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_sendto(
|
2017-10-13 13:14:04 -07:00
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jarray buf, jint len, jint flags, jobject ztaddr)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_in addr;
|
2017-11-06 13:50:20 -08:00
|
|
|
int sent_bytes = 0;
|
|
|
|
|
jclass c = (*env).GetObjectClass( ztaddr);
|
|
|
|
|
if (c) {
|
|
|
|
|
jfieldID f = (*env).GetFieldID(c, "port", "I");
|
|
|
|
|
addr.sin_port = htons((*env).GetIntField( ztaddr, f));
|
|
|
|
|
f = (*env).GetFieldID(c, "_rawAddr", "J");
|
|
|
|
|
addr.sin_addr.s_addr = (*env).GetLongField( ztaddr, f);
|
|
|
|
|
addr.sin_family = AF_INET;
|
|
|
|
|
//LOGV("zt_sendto(): fd = %d\naddr = %s\nport=%d", fd, inet_ntoa(addr.sin_addr), ntohs(addr.sin_port));
|
|
|
|
|
// TODO: Optimize this
|
|
|
|
|
jbyte *body = (*env).GetByteArrayElements((_jbyteArray *)buf, 0);
|
|
|
|
|
char * bufp = (char *)malloc(sizeof(char)*len);
|
|
|
|
|
memcpy(bufp, body, len);
|
|
|
|
|
(*env).ReleaseByteArrayElements((_jbyteArray *)buf, body, 0);
|
|
|
|
|
// "connect" and send buffer contents
|
|
|
|
|
sent_bytes = zts_sendto(fd, body, len, flags, (struct sockaddr *)&addr, sizeof(addr));
|
|
|
|
|
}
|
2017-10-13 13:14:04 -07:00
|
|
|
return sent_bytes;
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// TODO: ZT_SOCKET_API ssize_t ZTCALL zts_sendmsg(int fd, const struct msghdr *msg, int flags);
|
|
|
|
|
// TODO: ZT_SOCKET_API ssize_t ZTCALL zts_recv(int fd, void *buf, size_t len, int flags);
|
|
|
|
|
|
2017-10-16 15:34:43 -07:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_recvfrom(
|
2017-10-13 13:14:04 -07:00
|
|
|
JNIEnv *env, jobject thisObj, jint fd, jbyteArray buf, jint len, jint flags, jobject ztaddr)
|
|
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
/*
|
2017-10-13 13:14:04 -07:00
|
|
|
struct sockaddr_in addr;
|
|
|
|
|
jbyte *body = (*env).GetByteArrayElements( buf, 0);
|
|
|
|
|
unsigned char buffer[ZT_SDK_MTU];
|
|
|
|
|
int payload_offset = sizeof(int32_t) + sizeof(struct sockaddr_storage);
|
|
|
|
|
int rxbytes = zts_recvfrom(fd, &buffer, len, flags, (struct sockaddr *)&addr, (socklen_t *)sizeof(struct sockaddr_storage));
|
2017-11-06 13:50:20 -08:00
|
|
|
if (rxbytes > 0) {
|
2017-10-13 13:14:04 -07:00
|
|
|
memcpy(body, (jbyte*)buffer + payload_offset, rxbytes);
|
2017-11-06 13:50:20 -08:00
|
|
|
}
|
2017-10-13 13:14:04 -07:00
|
|
|
(*env).ReleaseByteArrayElements( buf, body, 0);
|
|
|
|
|
// Update fields of Java ZTAddress object
|
|
|
|
|
jfieldID fid;
|
2017-11-06 13:50:20 -08:00
|
|
|
jclass c = (*env).GetObjectClass( ztaddr);
|
|
|
|
|
if (c) {
|
|
|
|
|
fid = (*env).GetFieldID(c, "port", "I");
|
|
|
|
|
(*env).SetIntField(ztaddr, fid, addr.sin_port);
|
|
|
|
|
fid = (*env).GetFieldID(c,"_rawAddr", "J");
|
|
|
|
|
(*env).SetLongField(ztaddr, fid,addr.sin_addr.s_addr);
|
|
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
*/
|
|
|
|
|
return 1;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// TODO: ZT_SOCKET_API ssize_t ZTCALL zts_recvmsg(int fd, struct msghdr *msg,int flags);
|
|
|
|
|
|
|
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_read(JNIEnv *env, jobject thisObj,
|
|
|
|
|
jint fd, jarray buf, jint len)
|
2017-10-13 13:14:04 -07:00
|
|
|
{
|
|
|
|
|
jbyte *body = (*env).GetByteArrayElements((_jbyteArray *)buf, 0);
|
2018-07-12 16:53:40 -07:00
|
|
|
int read_bytes = zts_read(fd, body, len);
|
2017-10-13 13:14:04 -07:00
|
|
|
(*env).ReleaseByteArrayElements((_jbyteArray *)buf, body, 0);
|
2017-11-21 15:53:31 -08:00
|
|
|
return read_bytes;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
|
|
|
|
|
2017-11-06 13:50:20 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_write(JNIEnv *env, jobject thisObj,
|
2017-10-13 13:14:04 -07:00
|
|
|
jint fd, jarray buf, jint len)
|
|
|
|
|
{
|
|
|
|
|
jbyte *body = (*env).GetByteArrayElements((_jbyteArray *)buf, 0);
|
|
|
|
|
char * bufp = (char *)malloc(sizeof(char)*len);
|
|
|
|
|
memcpy(bufp, body, len);
|
|
|
|
|
(*env).ReleaseByteArrayElements((_jbyteArray *)buf, body, 0);
|
|
|
|
|
int written_bytes = zts_write(fd, body, len);
|
|
|
|
|
return written_bytes;
|
|
|
|
|
}
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
JNIEXPORT jint JNICALL Java_zerotier_ZeroTier_shutdown(
|
|
|
|
|
JNIEnv *env, jobject thisObj, int fd, int how)
|
2017-10-13 13:14:04 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
return zts_shutdown(fd, how);
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-10-16 16:45:46 -07:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// TODO: ZT_SOCKET_API int ZTCALL zts_add_dns_nameserver(struct sockaddr *addr);
|
|
|
|
|
// TODO: ZT_SOCKET_API int ZTCALL zts_del_dns_nameserver(struct sockaddr *addr);
|
|
|
|
|
}
|
2017-10-16 15:12:37 -07:00
|
|
|
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// convenience function
|
|
|
|
|
jobject ss2inet(JNIEnv *env, struct sockaddr_storage *src_ss)
|
|
|
|
|
{
|
|
|
|
|
jobject dest_inet;
|
|
|
|
|
if(src_ss->ss_family == AF_INET)
|
2017-10-16 15:12:37 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
DEBUG_ERROR("converting from INET");
|
|
|
|
|
struct sockaddr_in *in4 = (struct sockaddr_in*)src_ss;
|
|
|
|
|
int arrlen = 4;
|
|
|
|
|
jbyteArray bytes = (*env).NewByteArray(arrlen);
|
|
|
|
|
jbyte *java_address_bytes;
|
|
|
|
|
java_address_bytes = (*env).GetByteArrayElements(bytes, NULL);
|
|
|
|
|
memcpy(java_address_bytes, &(in4->sin_addr.s_addr), arrlen);
|
|
|
|
|
(*env).ReleaseByteArrayElements(bytes, java_address_bytes, 0);
|
|
|
|
|
jclass cls = (*env).FindClass("java/net/InetAddress");
|
|
|
|
|
jmethodID mid = (*env).GetStaticMethodID(cls, "getByAddress", "([B)Ljava/net/InetAddress;");
|
|
|
|
|
dest_inet = (*env).CallStaticObjectMethod(cls, mid, bytes);
|
|
|
|
|
(*env).DeleteLocalRef(bytes);
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
if(src_ss->ss_family == AF_INET6)
|
2017-10-16 15:12:37 -07:00
|
|
|
{
|
2017-11-21 15:53:31 -08:00
|
|
|
DEBUG_ERROR("converting from INET6");
|
|
|
|
|
struct sockaddr_in6 *in6 = (struct sockaddr_in6*)src_ss;
|
|
|
|
|
int arrlen = 16;
|
|
|
|
|
jbyteArray bytes = (*env).NewByteArray(arrlen);
|
|
|
|
|
(*env).SetByteArrayRegion(bytes, 0, 16, (const jbyte *)&(in6->sin6_addr));
|
|
|
|
|
jclass cls = (*env).FindClass("java/net/InetAddress");
|
|
|
|
|
jmethodID mid = (*env).GetStaticMethodID(cls, "getByAddress", "([B)Ljava/net/InetAddress;");
|
|
|
|
|
dest_inet = (*env).CallStaticObjectMethod(cls, mid, bytes);
|
|
|
|
|
(*env).DeleteLocalRef(bytes);
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
return dest_inet;
|
|
|
|
|
}
|
2017-10-13 13:14:04 -07:00
|
|
|
|
|
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
int sockinet2ss(JNIEnv *env, jobject src_inet, struct sockaddr_storage *dest_ss)
|
|
|
|
|
{
|
|
|
|
|
struct sockaddr_in *in4 = (struct sockaddr_in*)dest_ss;
|
|
|
|
|
struct sockaddr_in6 *in6 = (struct sockaddr_in6*)dest_ss;
|
|
|
|
|
int port = 0;
|
|
|
|
|
int socket_family = 0;
|
|
|
|
|
socklen_t addrlen;
|
2017-10-13 13:14:04 -07:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
// ---
|
2017-10-13 13:14:04 -07:00
|
|
|
|
2017-11-21 15:53:31 -08:00
|
|
|
jclass c = (*env).GetObjectClass(src_inet);
|
|
|
|
|
if (!c) {
|
|
|
|
|
return -1;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
// get port
|
|
|
|
|
jmethodID getPort = (*env).GetMethodID(c, "getPort", "()I");
|
|
|
|
|
if (!getPort) {
|
|
|
|
|
return -1;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
port = (*env).CallIntMethod(src_inet, getPort);
|
|
|
|
|
// get internal InetAddress
|
|
|
|
|
jobject inetaddr;
|
|
|
|
|
jmethodID getAddress = (*env).GetMethodID(c, "getAddress", "()Ljava/net/InetAddress;");
|
|
|
|
|
if (!getAddress) {
|
|
|
|
|
return -1;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
inetaddr = (*env).CallObjectMethod(src_inet, getAddress);
|
|
|
|
|
if (!inetaddr) {
|
|
|
|
|
return -1;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
jclass inetClass = (*env).GetObjectClass(inetaddr);
|
|
|
|
|
if (!inetClass) {
|
|
|
|
|
return -1;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
// string representation of IP address
|
|
|
|
|
jmethodID getHostAddress = (*env).GetMethodID(inetClass, "getHostAddress", "()Ljava/lang/String;");
|
|
|
|
|
jstring addrstr = (jstring)(*env).CallObjectMethod(inetaddr, getHostAddress);
|
|
|
|
|
const char *addr_str = (*env).GetStringUTFChars(addrstr, NULL);
|
|
|
|
|
for (int i=0; i<strlen(addr_str); i++) {
|
|
|
|
|
if (addr_str[i]=='.') {
|
|
|
|
|
DEBUG_INFO("ipv4, inet_addr");
|
|
|
|
|
socket_family = AF_INET;
|
|
|
|
|
in4->sin_family = AF_INET;
|
|
|
|
|
in4->sin_port = htons(port);
|
|
|
|
|
in4->sin_addr.s_addr = inet_addr(addr_str);
|
|
|
|
|
/*
|
|
|
|
|
if (!inet_pton(AF_INET, addr_str, &(in4->sin_addr))) {
|
|
|
|
|
DEBUG_ERROR("error converting address %s", addr_str);
|
|
|
|
|
}
|
|
|
|
|
*/
|
|
|
|
|
addrlen = sizeof(struct sockaddr_in);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (addr_str[i]==':') {
|
|
|
|
|
DEBUG_INFO("ipv6");
|
|
|
|
|
socket_family = AF_INET6;
|
|
|
|
|
if (!inet_pton(AF_INET6, addr_str, &(in6->sin6_addr))) {
|
|
|
|
|
DEBUG_ERROR("error converting address %s", addr_str);
|
|
|
|
|
}
|
|
|
|
|
addrlen = sizeof(struct sockaddr_in6);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
2017-11-21 15:53:31 -08:00
|
|
|
(*env).ReleaseStringUTFChars(addrstr, addr_str);
|
|
|
|
|
DEBUG_TEST("RESULT => %s : %d", inet_ntoa(in4->sin_addr), ntohs(in4->sin_port));
|
2017-12-04 17:00:52 -08:00
|
|
|
return 0;
|
2017-10-13 13:14:04 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
|
}
|
2017-11-06 13:50:20 -08:00
|
|
|
#endif
|
2017-11-21 15:53:31 -08:00
|
|
|
|
|
|
|
|
#endif // SDK_JNI
|