com_zerotierone_sdk_Node.cpp 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314
  1. /*
  2. * ZeroTier One - Network Virtualization Everywhere
  3. * Copyright (C) 2011-2015 ZeroTier, Inc.
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. *
  18. * --
  19. *
  20. * ZeroTier may be used and distributed under the terms of the GPLv3, which
  21. * are available at: http://www.gnu.org/licenses/gpl-3.0.html
  22. *
  23. * If you would like to embed ZeroTier into a commercial application or
  24. * redistribute it in a modified binary form, please contact ZeroTier Networks
  25. * LLC. Start here: http://www.zerotier.com/
  26. */
  27. #include "com_zerotierone_sdk_Node.h"
  28. #include "ZT_jnicache.h"
  29. #include "ZT_jniutils.h"
  30. #include <ZeroTierOne.h>
  31. #include "Mutex.hpp"
  32. #include <map>
  33. #include <string>
  34. #include <cassert>
  35. #include <cstring>
  36. #define LOG_TAG "Node"
  37. namespace {
  38. struct JniRef
  39. {
  40. JniRef()
  41. : jvm(NULL)
  42. , node(NULL)
  43. , dataStoreGetListener(NULL)
  44. , dataStorePutListener(NULL)
  45. , packetSender(NULL)
  46. , eventListener(NULL)
  47. , frameListener(NULL)
  48. , configListener(NULL)
  49. , pathChecker(NULL)
  50. , callbacks(NULL)
  51. {
  52. callbacks = (ZT_Node_Callbacks*)malloc(sizeof(ZT_Node_Callbacks));
  53. memset(callbacks, 0, sizeof(ZT_Node_Callbacks));
  54. }
  55. ~JniRef()
  56. {
  57. JNIEnv *env = NULL;
  58. jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  59. env->DeleteGlobalRef(dataStoreGetListener);
  60. env->DeleteGlobalRef(dataStorePutListener);
  61. env->DeleteGlobalRef(packetSender);
  62. env->DeleteGlobalRef(eventListener);
  63. env->DeleteGlobalRef(frameListener);
  64. env->DeleteGlobalRef(configListener);
  65. env->DeleteGlobalRef(pathChecker);
  66. free(callbacks);
  67. callbacks = NULL;
  68. }
  69. int64_t id;
  70. JavaVM *jvm;
  71. ZT_Node *node;
  72. jobject dataStoreGetListener;
  73. jobject dataStorePutListener;
  74. jobject packetSender;
  75. jobject eventListener;
  76. jobject frameListener;
  77. jobject configListener;
  78. jobject pathChecker;
  79. ZT_Node_Callbacks *callbacks;
  80. };
  81. int VirtualNetworkConfigFunctionCallback(
  82. ZT_Node *node,
  83. void *userData,
  84. void *threadData,
  85. uint64_t nwid,
  86. void **,
  87. enum ZT_VirtualNetworkConfigOperation operation,
  88. const ZT_VirtualNetworkConfig *config)
  89. {
  90. LOGV("VirtualNetworkConfigFunctionCallback");
  91. JniRef *ref = (JniRef*)userData;
  92. JNIEnv *env = NULL;
  93. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  94. if (ref->configListener == NULL) {
  95. LOGE("configListener is NULL");
  96. return -100;
  97. }
  98. jobject operationObject = createVirtualNetworkConfigOperation(env, operation);
  99. if(operationObject == NULL)
  100. {
  101. LOGE("Error creating VirtualNetworkConfigOperation object");
  102. return -101;
  103. }
  104. jobject networkConfigObject = newNetworkConfig(env, *config);
  105. if(networkConfigObject == NULL)
  106. {
  107. LOGE("Error creating VirtualNetworkConfig object");
  108. return -102;
  109. }
  110. return env->CallIntMethod(
  111. ref->configListener,
  112. VirtualNetworkConfigListener_onNetworkConfigurationUpdated_method,
  113. (jlong)nwid, operationObject, networkConfigObject);
  114. }
  115. void VirtualNetworkFrameFunctionCallback(ZT_Node *node,
  116. void *userData,
  117. void *threadData,
  118. uint64_t nwid,
  119. void**,
  120. uint64_t sourceMac,
  121. uint64_t destMac,
  122. unsigned int etherType,
  123. unsigned int vlanid,
  124. const void *frameData,
  125. unsigned int frameLength)
  126. {
  127. LOGV("VirtualNetworkFrameFunctionCallback");
  128. #ifndef NDEBUG
  129. unsigned char* local = (unsigned char*)frameData;
  130. LOGV("Type Bytes: 0x%02x%02x", local[12], local[13]);
  131. #endif
  132. JniRef *ref = (JniRef*)userData;
  133. assert(ref->node == node);
  134. JNIEnv *env = NULL;
  135. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  136. if (ref->frameListener == NULL) {
  137. LOGE("frameListener is NULL");
  138. return;
  139. }
  140. jbyteArray dataArray = env->NewByteArray(frameLength);
  141. if(env->ExceptionCheck() || dataArray == NULL)
  142. {
  143. LOGE("Couldn't create frame data array");
  144. return;
  145. }
  146. void *data = env->GetPrimitiveArrayCritical(dataArray, NULL);
  147. memcpy(data, frameData, frameLength);
  148. env->ReleasePrimitiveArrayCritical(dataArray, data, 0);
  149. if(env->ExceptionCheck())
  150. {
  151. LOGE("Error setting frame data to array");
  152. return;
  153. }
  154. env->CallVoidMethod(ref->frameListener, VirtualNetworkFrameListener_onVirtualNetworkFrame_method, (jlong)nwid, (jlong)sourceMac, (jlong)destMac, (jlong)etherType, (jlong)vlanid, dataArray);
  155. }
  156. void EventCallback(ZT_Node *node,
  157. void *userData,
  158. void *threadData,
  159. enum ZT_Event event,
  160. const void *data) {
  161. LOGV("EventCallback");
  162. JniRef *ref = (JniRef *) userData;
  163. if (ref->node != node && event != ZT_EVENT_UP) {
  164. LOGE("Nodes not equal. ref->node %p, node %p. Event: %d", ref->node, node, event);
  165. return;
  166. }
  167. JNIEnv *env = NULL;
  168. ref->jvm->GetEnv((void **) &env, JNI_VERSION_1_6);
  169. if (ref->eventListener == NULL) {
  170. LOGE("eventListener is NULL");
  171. return;
  172. }
  173. jobject eventObject = createEvent(env, event);
  174. if (env->ExceptionCheck() || eventObject == NULL) {
  175. return;
  176. }
  177. switch (event) {
  178. case ZT_EVENT_UP: {
  179. LOGD("Event Up");
  180. env->CallVoidMethod(ref->eventListener, EventListener_onEvent_method, eventObject);
  181. break;
  182. }
  183. case ZT_EVENT_OFFLINE: {
  184. LOGD("Event Offline");
  185. env->CallVoidMethod(ref->eventListener, EventListener_onEvent_method, eventObject);
  186. break;
  187. }
  188. case ZT_EVENT_ONLINE: {
  189. LOGD("Event Online");
  190. env->CallVoidMethod(ref->eventListener, EventListener_onEvent_method, eventObject);
  191. break;
  192. }
  193. case ZT_EVENT_DOWN: {
  194. LOGD("Event Down");
  195. env->CallVoidMethod(ref->eventListener, EventListener_onEvent_method, eventObject);
  196. break;
  197. }
  198. case ZT_EVENT_FATAL_ERROR_IDENTITY_COLLISION: {
  199. LOGV("Identity Collision");
  200. // call onEvent()
  201. env->CallVoidMethod(ref->eventListener, EventListener_onEvent_method, eventObject);
  202. }
  203. break;
  204. case ZT_EVENT_TRACE: {
  205. LOGV("Trace Event");
  206. // call onTrace()
  207. if (data != NULL) {
  208. const char *message = (const char *) data;
  209. jstring messageStr = env->NewStringUTF(message);
  210. env->CallVoidMethod(ref->eventListener, EventListener_onTrace_method, messageStr);
  211. }
  212. }
  213. break;
  214. case ZT_EVENT_USER_MESSAGE:
  215. case ZT_EVENT_REMOTE_TRACE:
  216. default:
  217. break;
  218. }
  219. }
  220. void StatePutFunction(
  221. ZT_Node *node,
  222. void *userData,
  223. void *threadData,
  224. enum ZT_StateObjectType type,
  225. const uint64_t id[2],
  226. const void *buffer,
  227. int bufferLength) {
  228. char p[4096] = {0};
  229. bool secure = false;
  230. switch (type) {
  231. case ZT_STATE_OBJECT_IDENTITY_PUBLIC:
  232. snprintf(p, sizeof(p), "identity.public");
  233. break;
  234. case ZT_STATE_OBJECT_IDENTITY_SECRET:
  235. snprintf(p, sizeof(p), "identity.secret");
  236. secure = true;
  237. break;
  238. case ZT_STATE_OBJECT_PLANET:
  239. snprintf(p, sizeof(p), "planet");
  240. break;
  241. case ZT_STATE_OBJECT_MOON:
  242. snprintf(p, sizeof(p), "moons.d/%.16llx.moon", (unsigned long long)id[0]);
  243. break;
  244. case ZT_STATE_OBJECT_NETWORK_CONFIG:
  245. snprintf(p, sizeof(p), "networks.d/%.16llx.conf", (unsigned long long)id[0]);
  246. break;
  247. case ZT_STATE_OBJECT_PEER:
  248. snprintf(p, sizeof(p), "peers.d/%.10llx", (unsigned long long)id[0]);
  249. break;
  250. default:
  251. return;
  252. }
  253. if (strlen(p) < 1) {
  254. return;
  255. }
  256. JniRef *ref = (JniRef*)userData;
  257. JNIEnv *env = NULL;
  258. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  259. if (ref->dataStorePutListener == NULL) {
  260. LOGE("dataStorePutListener is NULL");
  261. return;
  262. }
  263. jstring nameStr = env->NewStringUTF(p);
  264. if (bufferLength >= 0) {
  265. LOGD("JNI: Write file: %s", p);
  266. // set operation
  267. jbyteArray bufferObj = env->NewByteArray(bufferLength);
  268. if(env->ExceptionCheck() || bufferObj == NULL)
  269. {
  270. LOGE("Error creating byte array buffer!");
  271. return;
  272. }
  273. env->SetByteArrayRegion(bufferObj, 0, bufferLength, (jbyte*)buffer);
  274. env->CallIntMethod(ref->dataStorePutListener,
  275. DataStorePutListener_onDataStorePut_method,
  276. nameStr, bufferObj, secure);
  277. } else {
  278. LOGD("JNI: Delete file: %s", p);
  279. env->CallIntMethod(ref->dataStorePutListener, DataStorePutListener_onDelete_method, nameStr);
  280. }
  281. }
  282. int StateGetFunction(
  283. ZT_Node *node,
  284. void *userData,
  285. void *threadData,
  286. ZT_StateObjectType type,
  287. const uint64_t id[2],
  288. void *buffer,
  289. unsigned int bufferLength) {
  290. char p[4096] = {0};
  291. switch (type) {
  292. case ZT_STATE_OBJECT_IDENTITY_PUBLIC:
  293. snprintf(p, sizeof(p), "identity.public");
  294. break;
  295. case ZT_STATE_OBJECT_IDENTITY_SECRET:
  296. snprintf(p, sizeof(p), "identity.secret");
  297. break;
  298. case ZT_STATE_OBJECT_PLANET:
  299. snprintf(p, sizeof(p), "planet");
  300. break;
  301. case ZT_STATE_OBJECT_MOON:
  302. snprintf(p, sizeof(p), "moons.d/%.16llx.moon", (unsigned long long)id[0]);
  303. break;
  304. case ZT_STATE_OBJECT_NETWORK_CONFIG:
  305. snprintf(p, sizeof(p), "networks.d/%.16llx.conf", (unsigned long long)id[0]);
  306. break;
  307. case ZT_STATE_OBJECT_PEER:
  308. snprintf(p, sizeof(p), "peers.d/%.10llx", (unsigned long long)id[0]);
  309. break;
  310. default:
  311. return -100;
  312. }
  313. if (strlen(p) < 1) {
  314. return -101;
  315. }
  316. JniRef *ref = (JniRef*)userData;
  317. JNIEnv *env = NULL;
  318. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  319. if (ref->dataStoreGetListener == NULL) {
  320. LOGE("dataStoreGetListener is NULL");
  321. return -102;
  322. }
  323. jstring nameStr = env->NewStringUTF(p);
  324. if(nameStr == NULL)
  325. {
  326. LOGE("Error creating name string object");
  327. return -103; // out of memory
  328. }
  329. jbyteArray bufferObj = env->NewByteArray(bufferLength);
  330. if(bufferObj == NULL)
  331. {
  332. LOGE("Error creating byte[] buffer of size: %u", bufferLength);
  333. return -104;
  334. }
  335. LOGV("Calling onDataStoreGet(%s, %p)", p, buffer);
  336. int retval = (int)env->CallLongMethod(
  337. ref->dataStoreGetListener,
  338. DataStoreGetListener_onDataStoreGet_method,
  339. nameStr,
  340. bufferObj);
  341. LOGV("onDataStoreGet returned %d", retval);
  342. if(retval > 0)
  343. {
  344. void *data = env->GetPrimitiveArrayCritical(bufferObj, NULL);
  345. memcpy(buffer, data, retval);
  346. env->ReleasePrimitiveArrayCritical(bufferObj, data, 0);
  347. }
  348. return retval;
  349. }
  350. int WirePacketSendFunction(ZT_Node *node,
  351. void *userData,
  352. void *threadData,
  353. int64_t localSocket,
  354. const struct sockaddr_storage *remoteAddress,
  355. const void *buffer,
  356. unsigned int bufferSize,
  357. unsigned int ttl)
  358. {
  359. LOGV("WirePacketSendFunction(%lld, %p, %p, %d)", (long long)localSocket, remoteAddress, buffer, bufferSize);
  360. JniRef *ref = (JniRef*)userData;
  361. assert(ref->node == node);
  362. JNIEnv *env = NULL;
  363. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  364. if (ref->packetSender == NULL) {
  365. LOGE("packetSender is NULL");
  366. return -100;
  367. }
  368. jobject remoteAddressObj = newInetSocketAddress(env, *remoteAddress);
  369. jbyteArray bufferObj = env->NewByteArray(bufferSize);
  370. env->SetByteArrayRegion(bufferObj, 0, bufferSize, (jbyte*)buffer);
  371. int retval = env->CallIntMethod(ref->packetSender, PacketSender_onSendPacketRequested_method, localSocket, remoteAddressObj, bufferObj);
  372. LOGV("JNI Packet Sender returned: %d", retval);
  373. return retval;
  374. }
  375. int PathCheckFunction(ZT_Node *node,
  376. void *userPtr,
  377. void *threadPtr,
  378. uint64_t address,
  379. int64_t localSocket,
  380. const struct sockaddr_storage *remoteAddress)
  381. {
  382. JniRef *ref = (JniRef*)userPtr;
  383. assert(ref->node == node);
  384. if(ref->pathChecker == NULL) {
  385. return true;
  386. }
  387. JNIEnv *env = NULL;
  388. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  389. //
  390. // was:
  391. // struct sockaddr_storage nullAddress = {0};
  392. //
  393. // but was getting this warning:
  394. // warning: suggest braces around initialization of subobject
  395. //
  396. // when building ZeroTierOne
  397. //
  398. struct sockaddr_storage nullAddress;
  399. //
  400. // It is possible to assume knowledge about internals of sockaddr_storage and construct
  401. // correct 0-initializer, but it is simpler to just treat sockaddr_storage as opaque and
  402. // use memset here to fill with 0
  403. //
  404. // This is also done in InetAddress.hpp for InetAddress
  405. //
  406. memset(&nullAddress, 0, sizeof(sockaddr_storage));
  407. jobject remoteAddressObj = NULL;
  408. if(memcmp(remoteAddress, &nullAddress, sizeof(sockaddr_storage)) != 0)
  409. {
  410. remoteAddressObj = newInetSocketAddress(env, *remoteAddress);
  411. }
  412. return env->CallBooleanMethod(ref->pathChecker, PathChecker_onPathCheck_method, address, localSocket, remoteAddressObj);
  413. }
  414. int PathLookupFunction(ZT_Node *node,
  415. void *userPtr,
  416. void *threadPtr,
  417. uint64_t address,
  418. int ss_family,
  419. struct sockaddr_storage *result)
  420. {
  421. JniRef *ref = (JniRef*)userPtr;
  422. assert(ref->node == node);
  423. if(ref->pathChecker == NULL) {
  424. return false;
  425. }
  426. JNIEnv *env = NULL;
  427. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  428. jobject sockAddressObject = env->CallObjectMethod(ref->pathChecker, PathChecker_onPathLookup_method, address, ss_family);
  429. if(sockAddressObject == NULL)
  430. {
  431. LOGE("Unable to call onPathLookup implementation");
  432. return false;
  433. }
  434. jint port = env->CallIntMethod(sockAddressObject, InetSocketAddress_getPort_method);
  435. jobject addressObject = env->CallObjectMethod(sockAddressObject, InetSocketAddress_getAddress_method);
  436. jbyteArray addressBytes = (jbyteArray)env->CallObjectMethod(addressObject, InetAddress_getAddress_method);
  437. if(addressBytes == NULL)
  438. {
  439. LOGE("Unable to call InetAddress.getBytes()");
  440. return false;
  441. }
  442. int addressSize = env->GetArrayLength(addressBytes);
  443. if(addressSize == 4)
  444. {
  445. // IPV4
  446. sockaddr_in *addr = (sockaddr_in*)result;
  447. addr->sin_family = AF_INET;
  448. addr->sin_port = htons(port);
  449. void *data = env->GetPrimitiveArrayCritical(addressBytes, NULL);
  450. memcpy(&addr->sin_addr, data, 4);
  451. env->ReleasePrimitiveArrayCritical(addressBytes, data, 0);
  452. }
  453. else if (addressSize == 16)
  454. {
  455. // IPV6
  456. sockaddr_in6 *addr = (sockaddr_in6*)result;
  457. addr->sin6_family = AF_INET6;
  458. addr->sin6_port = htons(port);
  459. void *data = env->GetPrimitiveArrayCritical(addressBytes, NULL);
  460. memcpy(&addr->sin6_addr, data, 16);
  461. env->ReleasePrimitiveArrayCritical(addressBytes, data, 0);
  462. }
  463. else
  464. {
  465. return false;
  466. }
  467. return true;
  468. }
  469. typedef std::map<int64_t, JniRef*> NodeMap;
  470. static NodeMap nodeMap;
  471. ZeroTier::Mutex nodeMapMutex;
  472. ZT_Node* findNode(int64_t nodeId)
  473. {
  474. ZeroTier::Mutex::Lock lock(nodeMapMutex);
  475. NodeMap::iterator found = nodeMap.find(nodeId);
  476. if(found != nodeMap.end())
  477. {
  478. JniRef *ref = found->second;
  479. return ref->node;
  480. }
  481. return NULL;
  482. }
  483. }
  484. #ifdef __cplusplus
  485. extern "C" {
  486. #endif
  487. JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved)
  488. {
  489. setupJNICache(vm);
  490. return JNI_VERSION_1_6;
  491. }
  492. JNIEXPORT void JNICALL JNI_OnUnload(JavaVM *vm, void *reserved)
  493. {
  494. teardownJNICache(vm);
  495. }
  496. /*
  497. * Class: com_zerotier_sdk_Node
  498. * Method: node_init
  499. * Signature: (J)Lcom/zerotier/sdk/ResultCode;
  500. */
  501. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_node_1init(
  502. JNIEnv *env, jobject obj, jlong now)
  503. {
  504. LOGV("Creating ZT_Node struct");
  505. jobject resultObject = ResultCode_RESULT_OK_enum;
  506. ZT_Node *node;
  507. JniRef *ref = new JniRef;
  508. ref->id = (int64_t)now;
  509. env->GetJavaVM(&ref->jvm);
  510. jobject dataStoreGetListener = env->GetObjectField(obj, Node_getListener_field);
  511. if(dataStoreGetListener == NULL)
  512. {
  513. return NULL;
  514. }
  515. ref->dataStoreGetListener = env->NewGlobalRef(dataStoreGetListener);
  516. jobject dataStorePutListener = env->GetObjectField(obj, Node_putListener_field);
  517. if(dataStorePutListener == NULL)
  518. {
  519. return NULL;
  520. }
  521. ref->dataStorePutListener = env->NewGlobalRef(dataStorePutListener);
  522. jobject packetSender = env->GetObjectField(obj, Node_sender_field);
  523. if(packetSender == NULL)
  524. {
  525. return NULL;
  526. }
  527. ref->packetSender = env->NewGlobalRef(packetSender);
  528. jobject frameListener = env->GetObjectField(obj, Node_frameListener_field);
  529. if(frameListener == NULL)
  530. {
  531. return NULL;
  532. }
  533. ref->frameListener = env->NewGlobalRef(frameListener);
  534. jobject configListener = env->GetObjectField(obj, Node_configListener_field);
  535. if(configListener == NULL)
  536. {
  537. return NULL;
  538. }
  539. ref->configListener = env->NewGlobalRef(configListener);
  540. jobject eventListener = env->GetObjectField(obj, Node_eventListener_field);
  541. if(eventListener == NULL)
  542. {
  543. return NULL;
  544. }
  545. ref->eventListener = env->NewGlobalRef(eventListener);
  546. jobject pathChecker = env->GetObjectField(obj, Node_pathChecker_field);
  547. if(pathChecker != NULL)
  548. {
  549. ref->pathChecker = env->NewGlobalRef(pathChecker);
  550. }
  551. ref->callbacks->stateGetFunction = &StateGetFunction;
  552. ref->callbacks->statePutFunction = &StatePutFunction;
  553. ref->callbacks->wirePacketSendFunction = &WirePacketSendFunction;
  554. ref->callbacks->virtualNetworkFrameFunction = &VirtualNetworkFrameFunctionCallback;
  555. ref->callbacks->virtualNetworkConfigFunction = &VirtualNetworkConfigFunctionCallback;
  556. ref->callbacks->eventCallback = &EventCallback;
  557. ref->callbacks->pathCheckFunction = &PathCheckFunction;
  558. ref->callbacks->pathLookupFunction = &PathLookupFunction;
  559. ZT_ResultCode rc = ZT_Node_new(
  560. &node,
  561. ref,
  562. NULL,
  563. ref->callbacks,
  564. (int64_t)now);
  565. if(rc != ZT_RESULT_OK)
  566. {
  567. LOGE("Error creating Node: %d", rc);
  568. resultObject = createResultObject(env, rc);
  569. if (env->ExceptionCheck() || resultObject == NULL) {
  570. return NULL;
  571. }
  572. if(node)
  573. {
  574. ZT_Node_delete(node);
  575. node = NULL;
  576. }
  577. delete ref;
  578. ref = NULL;
  579. return resultObject;
  580. }
  581. ZeroTier::Mutex::Lock lock(nodeMapMutex);
  582. ref->node = node;
  583. nodeMap.insert(std::make_pair(ref->id, ref));
  584. return resultObject;
  585. }
  586. /*
  587. * Class: com_zerotier_sdk_Node
  588. * Method: node_delete
  589. * Signature: (J)V
  590. */
  591. JNIEXPORT void JNICALL Java_com_zerotier_sdk_Node_node_1delete(
  592. JNIEnv *env, jobject obj, jlong id)
  593. {
  594. LOGV("Destroying ZT_Node struct");
  595. int64_t nodeId = (int64_t)id;
  596. NodeMap::iterator found;
  597. {
  598. ZeroTier::Mutex::Lock lock(nodeMapMutex);
  599. found = nodeMap.find(nodeId);
  600. }
  601. if(found != nodeMap.end())
  602. {
  603. JniRef *ref = found->second;
  604. nodeMap.erase(found);
  605. ZT_Node_delete(ref->node);
  606. delete ref;
  607. ref = NULL;
  608. }
  609. else
  610. {
  611. LOGE("Attempted to delete a node that doesn't exist!");
  612. }
  613. }
  614. /*
  615. * Class: com_zerotier_sdk_Node
  616. * Method: processVirtualNetworkFrame
  617. * Signature: (JJJJJII[B[J)Lcom/zerotier/sdk/ResultCode;
  618. */
  619. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_processVirtualNetworkFrame(
  620. JNIEnv *env, jobject obj,
  621. jlong id,
  622. jlong in_now,
  623. jlong in_nwid,
  624. jlong in_sourceMac,
  625. jlong in_destMac,
  626. jint in_etherType,
  627. jint in_vlanId,
  628. jbyteArray in_frameData,
  629. jlongArray out_nextBackgroundTaskDeadline)
  630. {
  631. int64_t nodeId = (int64_t) id;
  632. ZT_Node *node = findNode(nodeId);
  633. if(node == NULL)
  634. {
  635. // cannot find valid node. We should never get here.
  636. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  637. }
  638. unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
  639. if(nbtd_len < 1)
  640. {
  641. // array for next background task length has 0 elements!
  642. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  643. }
  644. int64_t now = (int64_t)in_now;
  645. uint64_t nwid = (uint64_t)in_nwid;
  646. uint64_t sourceMac = (uint64_t)in_sourceMac;
  647. uint64_t destMac = (uint64_t)in_destMac;
  648. unsigned int etherType = (unsigned int)in_etherType;
  649. unsigned int vlanId = (unsigned int)in_vlanId;
  650. unsigned int frameLength = env->GetArrayLength(in_frameData);
  651. void *frameData = env->GetPrimitiveArrayCritical(in_frameData, NULL);
  652. void *localData = malloc(frameLength);
  653. memcpy(localData, frameData, frameLength);
  654. env->ReleasePrimitiveArrayCritical(in_frameData, frameData, 0);
  655. int64_t nextBackgroundTaskDeadline = 0;
  656. ZT_ResultCode rc = ZT_Node_processVirtualNetworkFrame(
  657. node,
  658. NULL,
  659. now,
  660. nwid,
  661. sourceMac,
  662. destMac,
  663. etherType,
  664. vlanId,
  665. (const void*)localData,
  666. frameLength,
  667. &nextBackgroundTaskDeadline);
  668. free(localData);
  669. jlong *outDeadline = (jlong*)env->GetPrimitiveArrayCritical(out_nextBackgroundTaskDeadline, NULL);
  670. outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
  671. env->ReleasePrimitiveArrayCritical(out_nextBackgroundTaskDeadline, outDeadline, 0);
  672. return createResultObject(env, rc);
  673. }
  674. /*
  675. * Class: com_zerotier_sdk_Node
  676. * Method: processWirePacket
  677. * Signature: (JJJLjava/net/InetSocketAddress;[B[J)Lcom/zerotier/sdk/ResultCode;
  678. */
  679. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_processWirePacket(
  680. JNIEnv *env, jobject obj,
  681. jlong id,
  682. jlong in_now,
  683. jlong in_localSocket,
  684. jobject in_remoteAddress,
  685. jbyteArray in_packetData,
  686. jlongArray out_nextBackgroundTaskDeadline)
  687. {
  688. int64_t nodeId = (int64_t) id;
  689. ZT_Node *node = findNode(nodeId);
  690. if(node == NULL)
  691. {
  692. // cannot find valid node. We should never get here.
  693. LOGE("Couldn't find a valid node!");
  694. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  695. }
  696. unsigned int nbtd_len = (unsigned int)env->GetArrayLength(out_nextBackgroundTaskDeadline);
  697. if(nbtd_len < 1)
  698. {
  699. LOGE("nbtd_len < 1");
  700. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  701. }
  702. int64_t now = (int64_t)in_now;
  703. jobject remoteAddrObject = env->CallObjectMethod(in_remoteAddress, InetSocketAddress_getAddress_method);
  704. if(remoteAddrObject == NULL)
  705. {
  706. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  707. }
  708. // call InetSocketAddress.getPort()
  709. int remotePort = env->CallIntMethod(in_remoteAddress, InetSocketAddress_getPort_method);
  710. if(env->ExceptionCheck())
  711. {
  712. LOGE("Exception calling InetSocketAddress.getPort()");
  713. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  714. }
  715. // Call InetAddress.getAddress()
  716. jbyteArray remoteAddressArray = (jbyteArray)env->CallObjectMethod(remoteAddrObject, InetAddress_getAddress_method);
  717. if(remoteAddressArray == NULL)
  718. {
  719. LOGE("Unable to call getAddress()");
  720. // unable to call getAddress()
  721. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  722. }
  723. unsigned int addrSize = env->GetArrayLength(remoteAddressArray);
  724. // get the address bytes
  725. jbyte *addr = (jbyte*)env->GetPrimitiveArrayCritical(remoteAddressArray, NULL);
  726. sockaddr_storage remoteAddress = {};
  727. if(addrSize == 16)
  728. {
  729. // IPV6 address
  730. sockaddr_in6 ipv6 = {};
  731. ipv6.sin6_family = AF_INET6;
  732. ipv6.sin6_port = htons(remotePort);
  733. memcpy(ipv6.sin6_addr.s6_addr, addr, 16);
  734. memcpy(&remoteAddress, &ipv6, sizeof(sockaddr_in6));
  735. }
  736. else if(addrSize == 4)
  737. {
  738. // IPV4 address
  739. sockaddr_in ipv4 = {};
  740. ipv4.sin_family = AF_INET;
  741. ipv4.sin_port = htons(remotePort);
  742. memcpy(&ipv4.sin_addr, addr, 4);
  743. memcpy(&remoteAddress, &ipv4, sizeof(sockaddr_in));
  744. }
  745. else
  746. {
  747. LOGE("Unknown IP version");
  748. // unknown address type
  749. env->ReleasePrimitiveArrayCritical(remoteAddressArray, addr, 0);
  750. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  751. }
  752. env->ReleasePrimitiveArrayCritical(remoteAddressArray, addr, 0);
  753. unsigned int packetLength = (unsigned int)env->GetArrayLength(in_packetData);
  754. if(packetLength == 0)
  755. {
  756. LOGE("Empty packet?!?");
  757. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  758. }
  759. void *packetData = env->GetPrimitiveArrayCritical(in_packetData, NULL);
  760. void *localData = malloc(packetLength);
  761. memcpy(localData, packetData, packetLength);
  762. env->ReleasePrimitiveArrayCritical(in_packetData, packetData, 0);
  763. int64_t nextBackgroundTaskDeadline = 0;
  764. ZT_ResultCode rc = ZT_Node_processWirePacket(
  765. node,
  766. NULL,
  767. now,
  768. in_localSocket,
  769. &remoteAddress,
  770. localData,
  771. packetLength,
  772. &nextBackgroundTaskDeadline);
  773. if(rc != ZT_RESULT_OK)
  774. {
  775. LOGE("ZT_Node_processWirePacket returned: %d", rc);
  776. }
  777. free(localData);
  778. jlong *outDeadline = (jlong*)env->GetPrimitiveArrayCritical(out_nextBackgroundTaskDeadline, NULL);
  779. outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
  780. env->ReleasePrimitiveArrayCritical(out_nextBackgroundTaskDeadline, outDeadline, 0);
  781. return createResultObject(env, rc);
  782. }
  783. /*
  784. * Class: com_zerotier_sdk_Node
  785. * Method: processBackgroundTasks
  786. * Signature: (JJ[J)Lcom/zerotier/sdk/ResultCode;
  787. */
  788. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_processBackgroundTasks(
  789. JNIEnv *env, jobject obj,
  790. jlong id,
  791. jlong in_now,
  792. jlongArray out_nextBackgroundTaskDeadline)
  793. {
  794. int64_t nodeId = (int64_t) id;
  795. ZT_Node *node = findNode(nodeId);
  796. if(node == NULL)
  797. {
  798. // cannot find valid node. We should never get here.
  799. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  800. }
  801. unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
  802. if(nbtd_len < 1)
  803. {
  804. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  805. }
  806. int64_t now = (int64_t)in_now;
  807. int64_t nextBackgroundTaskDeadline = 0;
  808. ZT_ResultCode rc = ZT_Node_processBackgroundTasks(node, NULL, now, &nextBackgroundTaskDeadline);
  809. jlong *outDeadline = (jlong*)env->GetPrimitiveArrayCritical(out_nextBackgroundTaskDeadline, NULL);
  810. outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
  811. env->ReleasePrimitiveArrayCritical(out_nextBackgroundTaskDeadline, outDeadline, 0);
  812. return createResultObject(env, rc);
  813. }
  814. /*
  815. * Class: com_zerotier_sdk_Node
  816. * Method: join
  817. * Signature: (JJ)Lcom/zerotier/sdk/ResultCode;
  818. */
  819. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_join(
  820. JNIEnv *env, jobject obj, jlong id, jlong in_nwid)
  821. {
  822. int64_t nodeId = (int64_t) id;
  823. ZT_Node *node = findNode(nodeId);
  824. if(node == NULL)
  825. {
  826. // cannot find valid node. We should never get here.
  827. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  828. }
  829. uint64_t nwid = (uint64_t)in_nwid;
  830. ZT_ResultCode rc = ZT_Node_join(node, nwid, NULL, NULL);
  831. return createResultObject(env, rc);
  832. }
  833. /*
  834. * Class: com_zerotier_sdk_Node
  835. * Method: leave
  836. * Signature: (JJ)Lcom/zerotier/sdk/ResultCode;
  837. */
  838. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_leave(
  839. JNIEnv *env, jobject obj, jlong id, jlong in_nwid)
  840. {
  841. int64_t nodeId = (int64_t) id;
  842. ZT_Node *node = findNode(nodeId);
  843. if(node == NULL)
  844. {
  845. // cannot find valid node. We should never get here.
  846. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  847. }
  848. uint64_t nwid = (uint64_t)in_nwid;
  849. ZT_ResultCode rc = ZT_Node_leave(node, nwid, NULL, NULL);
  850. return createResultObject(env, rc);
  851. }
  852. /*
  853. * Class: com_zerotier_sdk_Node
  854. * Method: multicastSubscribe
  855. * Signature: (JJJJ)Lcom/zerotier/sdk/ResultCode;
  856. */
  857. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_multicastSubscribe(
  858. JNIEnv *env, jobject obj,
  859. jlong id,
  860. jlong in_nwid,
  861. jlong in_multicastGroup,
  862. jlong in_multicastAdi)
  863. {
  864. int64_t nodeId = (int64_t) id;
  865. ZT_Node *node = findNode(nodeId);
  866. if(node == NULL)
  867. {
  868. // cannot find valid node. We should never get here.
  869. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  870. }
  871. uint64_t nwid = (uint64_t)in_nwid;
  872. uint64_t multicastGroup = (uint64_t)in_multicastGroup;
  873. unsigned long multicastAdi = (unsigned long)in_multicastAdi;
  874. ZT_ResultCode rc = ZT_Node_multicastSubscribe(
  875. node, NULL, nwid, multicastGroup, multicastAdi);
  876. return createResultObject(env, rc);
  877. }
  878. /*
  879. * Class: com_zerotier_sdk_Node
  880. * Method: multicastUnsubscribe
  881. * Signature: (JJJJ)Lcom/zerotier/sdk/ResultCode;
  882. */
  883. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_multicastUnsubscribe(
  884. JNIEnv *env, jobject obj,
  885. jlong id,
  886. jlong in_nwid,
  887. jlong in_multicastGroup,
  888. jlong in_multicastAdi)
  889. {
  890. int64_t nodeId = (int64_t) id;
  891. ZT_Node *node = findNode(nodeId);
  892. if(node == NULL)
  893. {
  894. // cannot find valid node. We should never get here.
  895. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  896. }
  897. uint64_t nwid = (uint64_t)in_nwid;
  898. uint64_t multicastGroup = (uint64_t)in_multicastGroup;
  899. unsigned long multicastAdi = (unsigned long)in_multicastAdi;
  900. ZT_ResultCode rc = ZT_Node_multicastUnsubscribe(
  901. node, nwid, multicastGroup, multicastAdi);
  902. return createResultObject(env, rc);
  903. }
  904. /*
  905. * Class: com_zerotier_sdk_Node
  906. * Method: orbit
  907. * Signature: (JJJ)Lcom/zerotier/sdk/ResultCode;
  908. */
  909. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_orbit(
  910. JNIEnv *env, jobject obj,
  911. jlong id,
  912. jlong in_moonWorldId,
  913. jlong in_moonSeed)
  914. {
  915. int64_t nodeId = (int64_t)id;
  916. ZT_Node *node = findNode(nodeId);
  917. if(node == NULL)
  918. {
  919. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  920. }
  921. uint64_t moonWorldId = (uint64_t)in_moonWorldId;
  922. uint64_t moonSeed = (uint64_t)in_moonSeed;
  923. ZT_ResultCode rc = ZT_Node_orbit(node, NULL, moonWorldId, moonSeed);
  924. return createResultObject(env, rc);
  925. }
  926. /*
  927. * Class: com_zerotier_sdk_Node
  928. * Method: deorbit
  929. * Signature: (JJ)L/com/zerotier/sdk/ResultCode;
  930. */
  931. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_deorbit(
  932. JNIEnv *env, jobject obj,
  933. jlong id,
  934. jlong in_moonWorldId)
  935. {
  936. int64_t nodeId = (int64_t)id;
  937. ZT_Node *node = findNode(nodeId);
  938. if(node == NULL)
  939. {
  940. return ResultCode_RESULT_FATAL_ERROR_INTERNAL_enum;
  941. }
  942. uint64_t moonWorldId = (uint64_t)in_moonWorldId;
  943. ZT_ResultCode rc = ZT_Node_deorbit(node, NULL, moonWorldId);
  944. return createResultObject(env, rc);
  945. }
  946. /*
  947. * Class: com_zerotier_sdk_Node
  948. * Method: address
  949. * Signature: (J)J
  950. */
  951. JNIEXPORT jlong JNICALL Java_com_zerotier_sdk_Node_address(
  952. JNIEnv *env , jobject obj, jlong id)
  953. {
  954. int64_t nodeId = (int64_t) id;
  955. ZT_Node *node = findNode(nodeId);
  956. if(node == NULL)
  957. {
  958. // cannot find valid node. We should never get here.
  959. return 0;
  960. }
  961. uint64_t address = ZT_Node_address(node);
  962. return (jlong)address;
  963. }
  964. /*
  965. * Class: com_zerotier_sdk_Node
  966. * Method: status
  967. * Signature: (J)Lcom/zerotier/sdk/NodeStatus;
  968. */
  969. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_status
  970. (JNIEnv *env, jobject obj, jlong id)
  971. {
  972. int64_t nodeId = (int64_t) id;
  973. ZT_Node *node = findNode(nodeId);
  974. if(node == NULL)
  975. {
  976. // cannot find valid node. We should never get here.
  977. return 0;
  978. }
  979. // create a com.zerotier.sdk.NodeStatus object
  980. jobject nodeStatusObj = env->NewObject(NodeStatus_class, NodeStatus_ctor);
  981. if(nodeStatusObj == NULL)
  982. {
  983. return NULL;
  984. }
  985. ZT_NodeStatus nodeStatus;
  986. ZT_Node_status(node, &nodeStatus);
  987. env->SetLongField(nodeStatusObj, NodeStatus_address_field, nodeStatus.address);
  988. jstring pubIdentStr = env->NewStringUTF(nodeStatus.publicIdentity);
  989. if(pubIdentStr == NULL)
  990. {
  991. return NULL; // out of memory
  992. }
  993. env->SetObjectField(nodeStatusObj, NodeStatus_publicIdentity_field, pubIdentStr);
  994. jstring secIdentStr = env->NewStringUTF(nodeStatus.secretIdentity);
  995. if(secIdentStr == NULL)
  996. {
  997. return NULL; // out of memory
  998. }
  999. env->SetObjectField(nodeStatusObj, NodeStatus_secretIdentity_field, secIdentStr);
  1000. env->SetBooleanField(nodeStatusObj, NodeStatus_online_field, nodeStatus.online);
  1001. return nodeStatusObj;
  1002. }
  1003. /*
  1004. * Class: com_zerotier_sdk_Node
  1005. * Method: networkConfig
  1006. * Signature: (JJ)Lcom/zerotier/sdk/VirtualNetworkConfig;
  1007. */
  1008. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_networkConfig(
  1009. JNIEnv *env, jobject obj, jlong id, jlong nwid)
  1010. {
  1011. int64_t nodeId = (int64_t) id;
  1012. ZT_Node *node = findNode(nodeId);
  1013. if(node == NULL)
  1014. {
  1015. // cannot find valid node. We should never get here.
  1016. return 0;
  1017. }
  1018. ZT_VirtualNetworkConfig *vnetConfig = ZT_Node_networkConfig(node, nwid);
  1019. jobject vnetConfigObject = newNetworkConfig(env, *vnetConfig);
  1020. ZT_Node_freeQueryResult(node, vnetConfig);
  1021. return vnetConfigObject;
  1022. }
  1023. /*
  1024. * Class: com_zerotier_sdk_Node
  1025. * Method: version
  1026. * Signature: ()Lcom/zerotier/sdk/Version;
  1027. */
  1028. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_version(
  1029. JNIEnv *env, jobject obj)
  1030. {
  1031. int major = 0;
  1032. int minor = 0;
  1033. int revision = 0;
  1034. ZT_version(&major, &minor, &revision);
  1035. return newVersion(env, major, minor, revision);
  1036. }
  1037. /*
  1038. * Class: com_zerotier_sdk_Node
  1039. * Method: peers
  1040. * Signature: (J)[Lcom/zerotier/sdk/Peer;
  1041. */
  1042. JNIEXPORT jobjectArray JNICALL Java_com_zerotier_sdk_Node_peers(
  1043. JNIEnv *env, jobject obj, jlong id)
  1044. {
  1045. int64_t nodeId = (int64_t) id;
  1046. ZT_Node *node = findNode(nodeId);
  1047. if(node == NULL)
  1048. {
  1049. // cannot find valid node. We should never get here.
  1050. return 0;
  1051. }
  1052. ZT_PeerList *peerList = ZT_Node_peers(node);
  1053. if(peerList == NULL)
  1054. {
  1055. LOGE("ZT_Node_peers returned NULL");
  1056. return NULL;
  1057. }
  1058. jobjectArray peerArrayObj = env->NewObjectArray(
  1059. peerList->peerCount, Peer_class, NULL);
  1060. if(env->ExceptionCheck() || peerArrayObj == NULL)
  1061. {
  1062. LOGE("Error creating Peer[] array");
  1063. ZT_Node_freeQueryResult(node, peerList);
  1064. return NULL;
  1065. }
  1066. for(unsigned int i = 0; i < peerList->peerCount; ++i)
  1067. {
  1068. jobject peerObj = newPeer(env, peerList->peers[i]);
  1069. env->SetObjectArrayElement(peerArrayObj, i, peerObj);
  1070. if(env->ExceptionCheck())
  1071. {
  1072. LOGE("Error assigning Peer object to array");
  1073. break;
  1074. }
  1075. env->DeleteLocalRef(peerObj);
  1076. }
  1077. ZT_Node_freeQueryResult(node, peerList);
  1078. peerList = NULL;
  1079. return peerArrayObj;
  1080. }
  1081. /*
  1082. * Class: com_zerotier_sdk_Node
  1083. * Method: networks
  1084. * Signature: (J)[Lcom/zerotier/sdk/VirtualNetworkConfig;
  1085. */
  1086. JNIEXPORT jobjectArray JNICALL Java_com_zerotier_sdk_Node_networks(
  1087. JNIEnv *env, jobject obj, jlong id)
  1088. {
  1089. int64_t nodeId = (int64_t) id;
  1090. ZT_Node *node = findNode(nodeId);
  1091. if(node == NULL)
  1092. {
  1093. // cannot find valid node. We should never get here.
  1094. return 0;
  1095. }
  1096. ZT_VirtualNetworkList *networkList = ZT_Node_networks(node);
  1097. if(networkList == NULL)
  1098. {
  1099. return NULL;
  1100. }
  1101. jobjectArray networkListObject = env->NewObjectArray(
  1102. networkList->networkCount, VirtualNetworkConfig_class, NULL);
  1103. if(env->ExceptionCheck() || networkListObject == NULL)
  1104. {
  1105. LOGE("Error creating VirtualNetworkConfig[] array");
  1106. ZT_Node_freeQueryResult(node, networkList);
  1107. return NULL;
  1108. }
  1109. for(unsigned int i = 0; i < networkList->networkCount; ++i)
  1110. {
  1111. jobject networkObject = newNetworkConfig(env, networkList->networks[i]);
  1112. env->SetObjectArrayElement(networkListObject, i, networkObject);
  1113. if(env->ExceptionCheck())
  1114. {
  1115. LOGE("Error assigning VirtualNetworkConfig object to array");
  1116. break;
  1117. }
  1118. env->DeleteLocalRef(networkObject);
  1119. }
  1120. ZT_Node_freeQueryResult(node, networkList);
  1121. return networkListObject;
  1122. }
  1123. #ifdef __cplusplus
  1124. } // extern "C"
  1125. #endif