| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093 | 
							- /*
 
-  * ZeroTier One - Network Virtualization Everywhere
 
-  * Copyright (C) 2011-2015  ZeroTier, Inc.
 
-  *
 
-  * 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/>.
 
-  *
 
-  * --
 
-  *
 
-  * ZeroTier may be used and distributed under the terms of the GPLv3, which
 
-  * are available at: http://www.gnu.org/licenses/gpl-3.0.html
 
-  *
 
-  * If you would like to embed ZeroTier into a commercial application or
 
-  * redistribute it in a modified binary form, please contact ZeroTier Networks
 
-  * LLC. Start here: http://www.zerotier.com/
 
-  */
 
- #include "com_zerotierone_sdk_Node.h"
 
- #include "ZT1_jniutils.h"
 
- #include <ZeroTierOne.h>
 
- #include <map>
 
- #include <string>
 
- #include <assert.h>
 
- #include <string.h>
 
- #ifdef __cplusplus
 
- extern "C" {
 
- #endif
 
- namespace {
 
-     struct JniRef
 
-     {
 
-         JniRef()
 
-             : env(NULL)
 
-             , node(NULL)
 
-             , dataStoreGetListener(NULL)
 
-             , dataStorePutListener(NULL)
 
-             , packetSender(NULL)
 
-             , frameListener(NULL)
 
-             , configListener(NULL)
 
-         {}
 
-         uint64_t id;
 
-         JNIEnv *env;
 
-         ZT1_Node *node;
 
-         jobject dataStoreGetListener;
 
-         jobject dataStorePutListener;
 
-         jobject packetSender;
 
-         jobject frameListener;
 
-         jobject configListener;
 
-     };
 
-     int VirtualNetworkConfigFunctionCallback(
 
-         ZT1_Node *node,
 
-         void *userData,
 
-         uint64_t nwid,
 
-         enum ZT1_VirtualNetworkConfigOperation operation,
 
-         const ZT1_VirtualNetworkConfig *config)
 
-     {
 
-         JniRef *ref = (JniRef*)userData;
 
-         assert(ref->node == node);
 
-         JNIEnv *env = ref->env;
 
-         static jclass configListenerClass = NULL;
 
-         static jmethodID callbackMethod = NULL;
 
-         if(configListenerClass == NULL)
 
-         {
 
-             configListenerClass = env->GetObjectClass(ref->configListener);
 
-             if(configListenerClass == NULL)
 
-             {
 
-                 return -1;
 
-             }
 
-         }
 
-         if(callbackMethod == NULL)
 
-         {
 
-             callbackMethod = env->GetMethodID(configListenerClass,
 
-                 "onNetworkConfigurationUpdated",
 
-                 "(JLcom/zerotierone/sdk/VirtualNetworkConfigOperation;Lcom/zerotierone/sdk/VirtualNetworkConfig;)I");
 
-             if(callbackMethod == NULL)
 
-             {
 
-                 return -2;
 
-             }
 
-         }
 
-         jobject operationObject = createVirtualNetworkConfigOperation(env, operation);
 
-         if(operationObject == NULL)
 
-         {
 
-             return -3;
 
-         }
 
-         jobject networkConfigObject = newNetworkConfig(env, *config);
 
-         if(networkConfigObject == NULL)
 
-         {
 
-             return -4;
 
-         }
 
-         return env->CallIntMethod(ref->configListener, callbackMethod, (jlong)nwid, operationObject, networkConfigObject);
 
-     }
 
-     void VirtualNetworkFrameFunctionCallback(ZT1_Node *node,void *userData,
 
-         uint64_t nwid,
 
-         uint64_t sourceMac,
 
-         uint64_t destMac,
 
-         unsigned int etherType,
 
-         unsigned int vlanid,
 
-         const void *frameData,
 
-         unsigned int frameLength)
 
-     {
 
-         JniRef *ref = (JniRef*)userData;
 
-         assert(ref->node == node);
 
-         JNIEnv *env = ref->env;
 
-         static jclass frameListenerClass = NULL;
 
-         static jmethodID callbackMethod = NULL;
 
-         if(frameListenerClass == NULL)
 
-         {
 
-             frameListenerClass = env->GetObjectClass(ref->frameListener);
 
-             if(frameListenerClass == NULL)
 
-             {
 
-                 return;
 
-             }
 
-         }
 
-         if(callbackMethod == NULL)
 
-         {
 
-             callbackMethod = env->GetMethodID(frameListenerClass,
 
-                 "onVirtualNetworkFrame", "(JJJJJ[B)V");
 
-             if(callbackMethod == NULL)
 
-             {
 
-                 return;
 
-             }
 
-         }
 
-         jbyteArray dataArray = env->NewByteArray(frameLength);
 
-         env->SetByteArrayRegion(dataArray, 0, frameLength, (jbyte*)frameData);
 
-         env->CallVoidMethod(ref->frameListener, callbackMethod, nwid, sourceMac, destMac, etherType, vlanid, dataArray);
 
-     }
 
-     void EventCallback(ZT1_Node *node,void *userData,enum ZT1_Event event, const void *data)
 
-     {
 
-         JniRef *ref = (JniRef*)userData;
 
-         assert(ref->node == node);
 
-         JNIEnv *env = ref->env;
 
-     }
 
-     long DataStoreGetFunction(ZT1_Node *node,void *userData,
 
-         const char *objectName,
 
-         void *buffer,
 
-         unsigned long bufferSize,
 
-         unsigned long bufferIndex,
 
-         unsigned long *out_objectSize)
 
-     {
 
-         JniRef *ref = (JniRef*)userData;
 
-         assert(ref->node == node);
 
-         JNIEnv *env = ref->env;
 
-         static jclass dataStoreGetClass = NULL;
 
-         static jmethodID callbackMethod = NULL;
 
-         if(dataStoreGetClass == NULL)
 
-         {
 
-             dataStoreGetClass = env->GetObjectClass(ref->dataStoreGetListener);
 
-             if(dataStoreGetClass == NULL)
 
-             {
 
-                 return -2;
 
-             }
 
-         }
 
-         if(callbackMethod == NULL)
 
-         {
 
-             callbackMethod = env->GetMethodID(dataStoreGetClass,
 
-                 "onDataStoreGet",
 
-                 "(Ljava/lang/String;[BJ[J)J");
 
-             if(callbackMethod == NULL)
 
-             {
 
-                 return -2;
 
-             }
 
-         }
 
-         jstring nameStr = env->NewStringUTF(objectName);
 
-         jbyteArray bufferObj = env->NewByteArray(bufferSize);
 
-         jlongArray objectSizeObj = env->NewLongArray(1);
 
-         long retval = env->CallLongMethod(
 
-             dataStoreGetClass, callbackMethod, nameStr, bufferObj, bufferIndex, objectSizeObj);
 
-         env->GetByteArrayRegion(bufferObj, 0, bufferSize, (jbyte*)buffer);
 
-         env->GetLongArrayRegion(objectSizeObj, 0, 1, (jlong*)&out_objectSize);
 
-         env->ReleaseByteArrayElements(bufferObj, (jbyte*)buffer, 0);
 
-         env->ReleaseLongArrayElements(objectSizeObj, (jlong*)&out_objectSize, 0);
 
-         return retval;
 
-     }
 
-     int DataStorePutFunction(ZT1_Node *node,void *userData,
 
-         const char *objectName,
 
-         const void *buffer,
 
-         unsigned long bufferSize,
 
-         int secure)
 
-     {
 
-         JniRef *ref = (JniRef*)userData;
 
-         assert(ref->node == node);
 
-         JNIEnv *env = ref->env;
 
-         static jclass dataStorePutClass = NULL;
 
-         static jmethodID callbackMethod = NULL;
 
-         static jmethodID deleteMethod = NULL;
 
-         if(dataStorePutClass == NULL)
 
-         {
 
-             dataStorePutClass = env->GetObjectClass(ref->dataStorePutListener);
 
-             if(dataStorePutClass == NULL)
 
-             {
 
-                 return -1;
 
-             }
 
-         }
 
-         if(callbackMethod == NULL)
 
-         {
 
-             callbackMethod = env->GetMethodID(dataStorePutClass,
 
-                 "onDataStorePut",
 
-                 "(Ljava/lang/String;[BZ)I");
 
-             if(callbackMethod == NULL)
 
-             {
 
-                 return -2;
 
-             }
 
-         }
 
-         if(deleteMethod == NULL)
 
-         {
 
-             deleteMethod = env->GetMethodID(dataStorePutClass,
 
-                 "onDelete", "(Ljava/lang/String;)I");
 
-             if(deleteMethod == NULL)
 
-             {
 
-                 return -3;
 
-             }
 
-         }
 
-         jstring nameStr = env->NewStringUTF(objectName);
 
-         if(buffer == NULL)
 
-         {
 
-             // delete operation
 
-             return env->CallIntMethod(dataStorePutClass, deleteMethod, nameStr);
 
-         }
 
-         else
 
-         {
 
-             // set operation
 
-             jbyteArray bufferObj = env->NewByteArray(bufferSize);
 
-             env->SetByteArrayRegion(bufferObj, 0, bufferSize, (jbyte*)buffer);
 
-             bool secure = secure != 0;
 
-             return env->CallIntMethod(dataStorePutClass, callbackMethod,
 
-                 nameStr, bufferObj, secure);
 
-         }
 
-     }
 
-     int WirePacketSendFunction(ZT1_Node *node,void *userData,const struct sockaddr_storage *,unsigned int,const void *,unsigned int)
 
-     {
 
-         JniRef *ref = (JniRef*)userData;
 
-         assert(ref->node == node);
 
-         JNIEnv *env = ref->env;
 
-         return 0;
 
-     }
 
-     typedef std::map<uint64_t, JniRef*> NodeMap;
 
-     static NodeMap nodeMap;
 
-     ZT1_Node* findNode(uint64_t nodeId)
 
-     {
 
-         NodeMap::iterator found = nodeMap.find(nodeId);
 
-         if(found != nodeMap.end())
 
-         {
 
-             JniRef *ref = found->second;
 
-             return ref->node;
 
-         }
 
-         return NULL;
 
-     }
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    node_init
 
-  * Signature: (J)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_node_1init(
 
-     JNIEnv *env, jobject obj, jlong now)
 
- {
 
-     jobject resultObject = createResultObject(env, ZT1_RESULT_OK);
 
-     ZT1_Node *node;
 
-     JniRef *ref = new JniRef;
 
-     ZT1_ResultCode rc = ZT1_Node_new(
 
-         &node,
 
-         ref,
 
-         (uint64_t)now,
 
-         &DataStoreGetFunction,
 
-         &DataStorePutFunction,
 
-         &WirePacketSendFunction,
 
-         &VirtualNetworkFrameFunctionCallback,
 
-         &VirtualNetworkConfigFunctionCallback,
 
-         &EventCallback);
 
-     if(rc != ZT1_RESULT_OK)
 
-     {
 
-         resultObject = createResultObject(env, rc);
 
-         if(node)
 
-         {
 
-             ZT1_Node_delete(node);
 
-             node = NULL;
 
-         }
 
-         delete ref;
 
-         ref = NULL;
 
-         return resultObject;
 
-     }
 
-     
 
-     ref->id = (uint64_t)now;
 
-     ref->env = env;
 
-     ref->node = node;
 
-     jclass cls = env->GetObjectClass(obj);
 
-     jfieldID fid = env->GetFieldID(
 
-         cls, "getListener", "Lcom/zerotierone/sdk/DataStoreGetListener;");
 
-     if(fid == NULL)
 
-     {
 
-         return NULL; // exception already thrown
 
-     }
 
-     ref->dataStoreGetListener = env->GetObjectField(obj, fid);
 
-     if(ref->dataStoreGetListener == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     fid = env->GetFieldID(
 
-         cls, "putListener", "Lcom/zerotierone/sdk/DataStorePutLisetner;");
 
-     if(fid == NULL)
 
-     {
 
-         return NULL; // exception already thrown
 
-     }
 
-     ref->dataStorePutListener = env->GetObjectField(obj, fid);
 
-     if(ref->dataStorePutListener == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     fid = env->GetFieldID(
 
-         cls, "sender", "Lcom/zerotierone/sdk/PacketSender;");
 
-     if(fid == NULL)
 
-     {
 
-         return NULL; // exception already thrown
 
-     }
 
-     ref->packetSender = env->GetObjectField(obj, fid);
 
-     if(ref->packetSender == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     fid = env->GetFieldID(
 
-         cls, "frameListener", "Lcom/zerotierone/sdk/VirtualNetworkFrameListener;");
 
-     if(fid == NULL)
 
-     {
 
-         return NULL; // exception already thrown
 
-     }
 
-     ref->frameListener = env->GetObjectField(obj, fid);
 
-     if(ref->frameListener = NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     fid = env->GetFieldID(
 
-         cls, "configListener", "Lcom/zerotierone/sdk/VirtualNetworkConfigListener;");
 
-     if(fid == NULL)
 
-     {
 
-         return NULL; // exception already thrown
 
-     }
 
-     ref->configListener = env->GetObjectField(obj, fid);
 
-     if(ref->configListener == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     nodeMap.insert(std::make_pair(ref->id, ref));
 
-     return resultObject;
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    node_delete
 
-  * Signature: (J)V
 
-  */
 
- JNIEXPORT void JNICALL Java_com_zerotierone_sdk_Node_node_1delete(
 
-     JNIEnv *env, jobject obj, jlong id)
 
- {
 
-     uint64_t nodeId = (uint64_t)id;
 
-     NodeMap::iterator found = nodeMap.find(nodeId);
 
-     if(found != nodeMap.end())
 
-     {
 
-         JniRef *ref = found->second;
 
-         nodeMap.erase(found);
 
-         ZT1_Node_delete(ref->node);
 
-         delete ref;
 
-         ref = NULL;
 
-     }
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    processVirtualNetworkFrame
 
-  * Signature: (JJJJJII[B[J)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_processVirtualNetworkFrame(
 
-     JNIEnv *env, jobject obj, 
 
-     jlong id, 
 
-     jlong in_now, 
 
-     jlong in_nwid,
 
-     jlong in_sourceMac,
 
-     jlong in_destMac,
 
-     jint in_etherType,
 
-     jint in_vlanId,
 
-     jbyteArray in_frameData,
 
-     jlongArray out_nextBackgroundTaskDeadline)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
 
-     if(nbtd_len < 1)
 
-     {
 
-         // array for next background task length has 0 elements!
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t now = (uint64_t)in_now;
 
-     uint64_t nwid = (uint64_t)in_nwid;
 
-     uint64_t sourceMac = (uint64_t)in_sourceMac;
 
-     uint64_t destMac = (uint64_t)in_destMac;
 
-     unsigned int etherType = (unsigned int)in_etherType;
 
-     unsigned int vlanId = (unsigned int)in_vlanId;
 
-     unsigned int frameLength = env->GetArrayLength(in_frameData);
 
-     jbyte *frameData =env->GetByteArrayElements(in_frameData, NULL);
 
-     uint64_t nextBackgroundTaskDeadline = 0;
 
-     ZT1_ResultCode rc = ZT1_Node_processVirtualNetworkFrame(
 
-         node,
 
-         now,
 
-         nwid,
 
-         sourceMac,
 
-         destMac,
 
-         etherType,
 
-         vlanId,
 
-         (const void*)frameData,
 
-         frameLength,
 
-         &nextBackgroundTaskDeadline);
 
-     jlong *outDeadline = env->GetLongArrayElements(out_nextBackgroundTaskDeadline, NULL);
 
-     outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
 
-     env->ReleaseLongArrayElements(out_nextBackgroundTaskDeadline, outDeadline, 0);
 
-     env->ReleaseByteArrayElements(in_frameData, frameData, 0);
 
-     return createResultObject(env, rc);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    processWirePacket
 
-  * Signature: (JJLjava/net/InetAddress;I[B[J)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_processWirePacket(
 
-     JNIEnv *env, jobject obj, 
 
-     jlong id,
 
-     jlong in_now, 
 
-     jobject in_remoteAddress,
 
-     jint in_linkDesparation,
 
-     jbyteArray in_packetData,
 
-     jlongArray out_nextBackgroundTaskDeadline)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
 
-     if(nbtd_len < 1)
 
-     {
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t now = (uint64_t)in_now;
 
-     unsigned int linkDesparation = (unsigned int)in_linkDesparation;
 
-     // get the java.net.InetAddress class and getAddress() method
 
-     jclass inetAddressClass = env->FindClass("java/net/InetAddress");
 
-     if(inetAddressClass == NULL)
 
-     {
 
-         // can't find java.net.InetAddress
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     jmethodID getAddressMethod = env->GetMethodID(
 
-         inetAddressClass, "getAddress", "()[B");
 
-     if(getAddressMethod == NULL)
 
-     {
 
-         // cant find InetAddress.getAddres()
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     // Call InetAddress.getAddress()
 
-     jbyteArray addressArray = (jbyteArray)env->CallObjectMethod(in_remoteAddress, getAddressMethod);
 
-     if(addressArray == NULL)
 
-     {
 
-         // unable to call getAddress()
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     unsigned int addrSize = env->GetArrayLength(addressArray);
 
-     // get the address bytes
 
-     jbyte *addr = env->GetByteArrayElements(addressArray, NULL);
 
-     sockaddr_storage remoteAddress = {};
 
-     if(addrSize == 16)
 
-     {
 
-         // IPV6 address
 
-         sockaddr_in6 ipv6 = {};
 
-         ipv6.sin6_family = AF_INET6;
 
-         memcpy(ipv6.sin6_addr.s6_addr, addr, 16);
 
-         memcpy(&remoteAddress, &ipv6, sizeof(sockaddr_in6));
 
-     }
 
-     else if(addrSize = 4)
 
-     {
 
-         // IPV4 address
 
-         sockaddr_in ipv4 = {};
 
-         ipv4.sin_family = AF_INET;
 
-         memcpy(&ipv4.sin_addr, addr, 4);
 
-         memcpy(&remoteAddress, &ipv4, sizeof(sockaddr_in));
 
-     }
 
-     else
 
-     {
 
-         // unknown address type
 
-         env->ReleaseByteArrayElements(addressArray, addr, 0);
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     unsigned int packetLength = env->GetArrayLength(in_packetData);
 
-     jbyte *packetData = env->GetByteArrayElements(in_packetData, NULL);
 
-     uint64_t nextBackgroundTaskDeadline = 0;
 
-     ZT1_ResultCode rc = ZT1_Node_processWirePacket(
 
-         node,
 
-         now,
 
-         &remoteAddress,
 
-         linkDesparation,
 
-         packetData,
 
-         packetLength,
 
-         &nextBackgroundTaskDeadline);
 
-     jlong *outDeadline = env->GetLongArrayElements(out_nextBackgroundTaskDeadline, NULL);
 
-     outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
 
-     env->ReleaseLongArrayElements(out_nextBackgroundTaskDeadline, outDeadline, 0);
 
-     env->ReleaseByteArrayElements(addressArray, addr, 0);
 
-     env->ReleaseByteArrayElements(in_packetData, packetData, 0);
 
-     return createResultObject(env, ZT1_RESULT_OK);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    processBackgroundTasks
 
-  * Signature: (JJ[J)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_processBackgroundTasks(
 
-     JNIEnv *env, jobject obj, 
 
-     jlong id,
 
-     jlong in_now,
 
-     jlongArray out_nextBackgroundTaskDeadline)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
 
-     if(nbtd_len < 1)
 
-     {
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t now = (uint64_t)in_now;
 
-     uint64_t nextBackgroundTaskDeadline = 0;
 
-     ZT1_ResultCode rc = ZT1_Node_processBackgroundTasks(node, now, &nextBackgroundTaskDeadline);
 
-     jlong *outDeadline = env->GetLongArrayElements(out_nextBackgroundTaskDeadline, NULL);
 
-     outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
 
-     env->ReleaseLongArrayElements(out_nextBackgroundTaskDeadline, outDeadline, 0);
 
-     return createResultObject(env, rc);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    join
 
-  * Signature: (JJ)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_join(
 
-     JNIEnv *env, jobject obj, jlong id, jlong in_nwid)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t nwid = (uint64_t)in_nwid;
 
-     ZT1_ResultCode rc = ZT1_Node_join(node, nwid);
 
-     return createResultObject(env, rc);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    leave
 
-  * Signature: (JJ)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_leave(
 
-     JNIEnv *env, jobject obj, jlong id, jlong in_nwid)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t nwid = (uint64_t)in_nwid;
 
-     ZT1_ResultCode rc = ZT1_Node_leave(node, nwid);
 
-     return createResultObject(env, rc);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    multicastSubscribe
 
-  * Signature: (JJJJ)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_multicastSubscribe(
 
-     JNIEnv *env, jobject obj, 
 
-     jlong id, 
 
-     jlong in_nwid,
 
-     jlong in_multicastGroup,
 
-     jlong in_multicastAdi)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t nwid = (uint64_t)in_nwid;
 
-     uint64_t multicastGroup = (uint64_t)in_multicastGroup;
 
-     uint64_t multicastAdi = (uint64_t)in_multicastAdi;
 
-     ZT1_ResultCode rc = ZT1_Node_multicastSubscribe(
 
-         node, nwid, multicastGroup, multicastAdi);
 
-     return createResultObject(env, rc);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    multicastUnsubscribe
 
-  * Signature: (JJJJ)Lcom/zerotierone/sdk/ResultCode;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_multicastUnsubscribe(
 
-     JNIEnv *env, jobject obj, 
 
-     jlong id, 
 
-     jlong in_nwid,
 
-     jlong in_multicastGroup,
 
-     jlong in_multicastAdi)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
 
-     }
 
-     uint64_t nwid = (uint64_t)in_nwid;
 
-     uint64_t multicastGroup = (uint64_t)in_multicastGroup;
 
-     uint64_t multicastAdi = (uint64_t)in_multicastAdi;
 
-     ZT1_ResultCode rc = ZT1_Node_multicastUnsubscribe(
 
-         node, nwid, multicastGroup, multicastAdi);
 
-     return createResultObject(env, rc);
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    address
 
-  * Signature: (J)J
 
-  */
 
- JNIEXPORT jlong JNICALL Java_com_zerotierone_sdk_Node_address(
 
-     JNIEnv *env , jobject obj, jlong id)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return 0;
 
-     }
 
-     uint64_t address = ZT1_Node_address(node);
 
-     return (jlong)address;
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    status
 
-  * Signature: (J)Lcom/zerotierone/sdk/NodeStatus;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_status
 
-    (JNIEnv *env, jobject obj, jlong id)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return 0;
 
-     }
 
-     // static so we only have to look these up once
 
-     static jclass nodeStatusClass = NULL;
 
-     static jmethodID nodeStatusConstructor = NULL;
 
-     // create a com.zerotierone.sdk.NodeStatus object
 
-     if(nodeStatusClass == NULL)
 
-     {
 
-         nodeStatusClass = env->FindClass("com/zerotierone/sdk/NodeStatus");
 
-         if(nodeStatusClass == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     
 
-     if(nodeStatusConstructor == NULL)
 
-     {
 
-         nodeStatusConstructor = env->GetMethodID(
 
-             nodeStatusClass, "<init>", "()V");
 
-         if(nodeStatusConstructor == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     jobject nodeStatusObj = env->NewObject(nodeStatusClass, nodeStatusConstructor);
 
-     if(nodeStatusObj == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     ZT1_NodeStatus nodeStatus;
 
-     ZT1_Node_status(node, &nodeStatus);
 
-     static jfieldID addressField = NULL;
 
-     static jfieldID publicIdentityField = NULL;
 
-     static jfieldID secretIdentityField = NULL;
 
-     static jfieldID onlineField = NULL;
 
-     if(addressField == NULL)
 
-     {
 
-         addressField = env->GetFieldID(nodeStatusClass, "address", "J");
 
-         if(addressField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     if(publicIdentityField == NULL)
 
-     {
 
-         publicIdentityField = env->GetFieldID(nodeStatusClass, "publicIdentity", "Ljava/lang/String;");
 
-         if(publicIdentityField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     if(secretIdentityField == NULL)
 
-     {
 
-         secretIdentityField = env->GetFieldID(nodeStatusClass, "secretIdentity", "Ljava/lang/String;");
 
-         if(secretIdentityField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     if(onlineField == NULL)
 
-     {
 
-         onlineField = env->GetFieldID(nodeStatusClass, "online", "Z");
 
-         if(onlineField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     env->SetLongField(nodeStatusObj, addressField, nodeStatus.address);
 
-     jstring pubIdentStr = env->NewStringUTF(nodeStatus.publicIdentity);
 
-     if(pubIdentStr == NULL)
 
-     {
 
-         return NULL; // out of memory
 
-     }
 
-     env->SetObjectField(nodeStatusObj, publicIdentityField, pubIdentStr);
 
-     jstring secIdentStr = env->NewStringUTF(nodeStatus.secretIdentity);
 
-     if(secIdentStr == NULL)
 
-     {
 
-         return NULL; // out of memory
 
-     }
 
-     env->SetObjectField(nodeStatusObj, secretIdentityField, secIdentStr);
 
-     env->SetBooleanField(nodeStatusObj, onlineField, nodeStatus.online);
 
-     return nodeStatusObj;
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    networkConfig
 
-  * Signature: (J)Lcom/zerotierone/sdk/VirtualNetworkConfig;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_networkConfig(
 
-     JNIEnv *env, jobject obj, jlong id, jlong nwid)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return 0;
 
-     }
 
-     ZT1_VirtualNetworkConfig *vnetConfig = ZT1_Node_networkConfig(node, nwid);
 
-     
 
-     jobject vnetConfigObject = newNetworkConfig(env, *vnetConfig);
 
-     ZT1_Node_freeQueryResult(node, vnetConfig);
 
-     return vnetConfigObject;
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    version
 
-  * Signature: (J)Lcom/zerotierone/sdk/Version;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_version(
 
-     JNIEnv *env, jobject obj)
 
- {
 
-     // create a com.zerotierone.sdk.Version object
 
-     jclass versionClass = env->FindClass("com/zerotierone/sdk/Version");
 
-     if(versionClass == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     jmethodID versionConstructor = env->GetMethodID(
 
-         versionClass, "<init>", "()V");
 
-     if(versionConstructor == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     jobject versionObj = env->NewObject(versionClass, versionConstructor);
 
-     if(versionObj == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     int major = 0;
 
-     int minor = 0;
 
-     int revision = 0;
 
-     unsigned long featureFlags = 0;
 
-     ZT1_version(&major, &minor, &revision, &featureFlags);
 
-     // copy data to Version object
 
-     static jfieldID majorField = NULL;
 
-     static jfieldID minorField = NULL;
 
-     static jfieldID revisionField = NULL;
 
-     static jfieldID featureFlagsField = NULL;
 
-     if(majorField == NULL)
 
-     {
 
-         majorField = env->GetFieldID(versionClass, "major", "I");
 
-         if(majorField = NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     if(minorField == NULL)
 
-     {
 
-         minorField = env->GetFieldID(versionClass, "minor", "I");
 
-         if(minorField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     if(revisionField == NULL)
 
-     {
 
-         revisionField = env->GetFieldID(versionClass, "revision", "I");
 
-         if(revisionField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     if(featureFlagsField == NULL)
 
-     {
 
-         featureFlagsField = env->GetFieldID(versionClass, "featureFlags", "J");
 
-         if(featureFlagsField == NULL)
 
-         {
 
-             return NULL;
 
-         }
 
-     }
 
-     env->SetIntField(versionObj, majorField, (jint)major);
 
-     env->SetIntField(versionObj, minorField, (jint)minor);
 
-     env->SetIntField(versionObj, revisionField, (jint)revision);
 
-     env->SetLongField(versionObj, featureFlagsField, (jlong)featureFlags);
 
-     return versionObj;
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    peers
 
-  * Signature: (J)Ljava/util/ArrayList;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_peers(
 
-     JNIEnv *env, jobject obj, jlong id)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return 0;
 
-     }
 
-     ZT1_PeerList *peerList = ZT1_Node_peers(node);
 
-     if(peerList == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     jobject peerListObject = newArrayList(env);
 
-     if(peerListObject == NULL)
 
-     {
 
-         ZT1_Node_freeQueryResult(node, peerList);
 
-         return NULL;
 
-     }
 
-     for(unsigned int i = 0; i < peerList->peerCount; ++i)
 
-     {
 
-         jobject peerObj = newPeer(env, peerList->peers[i]);
 
-         appendItemToArrayList(env, peerListObject, peerObj);
 
-     }
 
-     ZT1_Node_freeQueryResult(node, peerList);
 
-     peerList = NULL;
 
-     return peerListObject;
 
- }
 
- /*
 
-  * Class:     com_zerotierone_sdk_Node
 
-  * Method:    networks
 
-  * Signature: (J)Ljava/util/ArrayList;
 
-  */
 
- JNIEXPORT jobject JNICALL Java_com_zerotierone_sdk_Node_networks(
 
-     JNIEnv *env, jobject obj, jlong id)
 
- {
 
-     uint64_t nodeId = (uint64_t) id;
 
-     ZT1_Node *node = findNode(nodeId);
 
-     if(node == NULL)
 
-     {
 
-         // cannot find valid node.  We should  never get here.
 
-         return 0;
 
-     }
 
-     ZT1_VirtualNetworkList *networkList = ZT1_Node_networks(node);
 
-     if(networkList == NULL)
 
-     {
 
-         return NULL;
 
-     }
 
-     jobject networkListObject = newArrayList(env);
 
-     if(networkListObject == NULL)
 
-     {
 
-         ZT1_Node_freeQueryResult(node, networkList);
 
-         return NULL;
 
-     }
 
-     for(unsigned int i = 0; i < networkList->networkCount; ++i)
 
-     {
 
-         jobject networkObject = newNetworkConfig(env, networkList->networks[i]);
 
-         appendItemToArrayList(env, networkListObject, networkObject);
 
-     }
 
-     ZT1_Node_freeQueryResult(node, networkList);
 
-     return networkListObject;
 
- }
 
- #ifdef __cplusplus
 
- } // extern "C"
 
- #endif
 
 
  |