com_zerotierone_sdk_Node.cpp 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303
  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 "ZT1_jniutils.h"
  29. #include "ZT1_jnilookup.h"
  30. #include <ZeroTierOne.h>
  31. #include <map>
  32. #include <string>
  33. #include <assert.h>
  34. #include <string.h>
  35. // global static JNI Lookup Object
  36. JniLookup lookup;
  37. #ifdef __cplusplus
  38. extern "C" {
  39. #endif
  40. namespace {
  41. struct JniRef
  42. {
  43. JniRef()
  44. : jvm(NULL)
  45. , node(NULL)
  46. , dataStoreGetListener(NULL)
  47. , dataStorePutListener(NULL)
  48. , packetSender(NULL)
  49. , eventListener(NULL)
  50. , frameListener(NULL)
  51. , configListener(NULL)
  52. {}
  53. ~JniRef()
  54. {
  55. JNIEnv *env = NULL;
  56. jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  57. env->DeleteGlobalRef(dataStoreGetListener);
  58. env->DeleteGlobalRef(dataStorePutListener);
  59. env->DeleteGlobalRef(packetSender);
  60. env->DeleteGlobalRef(eventListener);
  61. env->DeleteGlobalRef(frameListener);
  62. env->DeleteGlobalRef(configListener);
  63. }
  64. uint64_t id;
  65. JavaVM *jvm;
  66. ZT1_Node *node;
  67. jobject dataStoreGetListener;
  68. jobject dataStorePutListener;
  69. jobject packetSender;
  70. jobject eventListener;
  71. jobject frameListener;
  72. jobject configListener;
  73. };
  74. int VirtualNetworkConfigFunctionCallback(
  75. ZT1_Node *node,
  76. void *userData,
  77. uint64_t nwid,
  78. enum ZT1_VirtualNetworkConfigOperation operation,
  79. const ZT1_VirtualNetworkConfig *config)
  80. {
  81. LOGV("VritualNetworkConfigFunctionCallback");
  82. JniRef *ref = (JniRef*)userData;
  83. JNIEnv *env = NULL;
  84. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  85. jclass configListenerClass = env->GetObjectClass(ref->configListener);
  86. if(configListenerClass == NULL)
  87. {
  88. LOGE("Couldn't find class for VirtualNetworkConfigListener instance");
  89. return -1;
  90. }
  91. jmethodID configListenerCallbackMethod = lookup.findMethod(configListenerClass,
  92. "onNetworkConfigurationUpdated",
  93. "(JLcom/zerotier/sdk/VirtualNetworkConfigOperation;Lcom/zerotier/sdk/VirtualNetworkConfig;)I");
  94. if(configListenerCallbackMethod == NULL)
  95. {
  96. LOGE("Couldn't find onVirtualNetworkFrame() method");
  97. return -2;
  98. }
  99. jobject operationObject = createVirtualNetworkConfigOperation(env, operation);
  100. if(operationObject == NULL)
  101. {
  102. LOGE("Error creating VirtualNetworkConfigOperation object");
  103. return -3;
  104. }
  105. jobject networkConfigObject = newNetworkConfig(env, *config);
  106. if(networkConfigObject == NULL)
  107. {
  108. LOGE("Error creating VirtualNetworkConfig object");
  109. return -4;
  110. }
  111. return env->CallIntMethod(
  112. ref->configListener,
  113. configListenerCallbackMethod,
  114. (jlong)nwid, operationObject, networkConfigObject);
  115. }
  116. void VirtualNetworkFrameFunctionCallback(ZT1_Node *node,void *userData,
  117. uint64_t nwid,
  118. uint64_t sourceMac,
  119. uint64_t destMac,
  120. unsigned int etherType,
  121. unsigned int vlanid,
  122. const void *frameData,
  123. unsigned int frameLength)
  124. {
  125. LOGV("VirtualNetworkFrameFunctionCallback");
  126. unsigned char* local = (unsigned char*)frameData;
  127. LOGV("Type Bytes: 0x%02x%02x", local[12], local[13]);
  128. JniRef *ref = (JniRef*)userData;
  129. assert(ref->node == node);
  130. JNIEnv *env = NULL;
  131. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  132. jclass frameListenerClass = env->GetObjectClass(ref->frameListener);
  133. if(env->ExceptionCheck() || frameListenerClass == NULL)
  134. {
  135. LOGE("Couldn't find class for VirtualNetworkFrameListener instance");
  136. return;
  137. }
  138. jmethodID frameListenerCallbackMethod = lookup.findMethod(
  139. frameListenerClass,
  140. "onVirtualNetworkFrame", "(JJJJJ[B)V");
  141. if(env->ExceptionCheck() || frameListenerCallbackMethod == NULL)
  142. {
  143. LOGE("Couldn't find onVirtualNetworkFrame() method");
  144. return;
  145. }
  146. jbyteArray dataArray = env->NewByteArray(frameLength);
  147. if(env->ExceptionCheck() || dataArray == NULL)
  148. {
  149. LOGE("Couldn't create frame data array");
  150. return;
  151. }
  152. void *data = env->GetPrimitiveArrayCritical(dataArray, NULL);
  153. memcpy(data, frameData, frameLength);
  154. env->ReleasePrimitiveArrayCritical(dataArray, data, 0);
  155. if(env->ExceptionCheck())
  156. {
  157. LOGE("Error setting frame data to array");
  158. return;
  159. }
  160. env->CallVoidMethod(ref->frameListener, frameListenerCallbackMethod, (jlong)nwid, (jlong)sourceMac, (jlong)destMac, (jlong)etherType, (jlong)vlanid, dataArray);
  161. }
  162. void EventCallback(ZT1_Node *node,void *userData,enum ZT1_Event event, const void *data)
  163. {
  164. LOGV("EventCallback");
  165. JniRef *ref = (JniRef*)userData;
  166. assert(ref->node == node);
  167. JNIEnv *env = NULL;
  168. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  169. jclass eventListenerClass = env->GetObjectClass(ref->eventListener);
  170. if(eventListenerClass == NULL)
  171. {
  172. LOGE("Couldn't class for EventListener instance");
  173. return;
  174. }
  175. jmethodID onEventMethod = lookup.findMethod(eventListenerClass,
  176. "onEvent", "(Lcom/zerotier/sdk/Event;)V");
  177. if(onEventMethod == NULL)
  178. {
  179. LOGE("Couldn't find onEvent method");
  180. return;
  181. }
  182. jmethodID onOutOfDateMethod = lookup.findMethod(eventListenerClass,
  183. "onOutOfDate", "(Lcom/zerotier/sdk/Version;)V");
  184. if(onOutOfDateMethod == NULL)
  185. {
  186. LOGE("Couldn't find onOutOfDate method");
  187. return;
  188. }
  189. jmethodID onNetworkErrorMethod = lookup.findMethod(eventListenerClass,
  190. "onNetworkError", "(Lcom/zerotier/sdk/Event;Ljava/net/InetSocketAddress;)V");
  191. if(onNetworkErrorMethod == NULL)
  192. {
  193. LOGE("Couldn't find onNetworkError method");
  194. return;
  195. }
  196. jmethodID onTraceMethod = lookup.findMethod(eventListenerClass,
  197. "onTrace", "(Ljava/lang/String;)V");
  198. if(onTraceMethod == NULL)
  199. {
  200. LOGE("Couldn't find onTrace method");
  201. return;
  202. }
  203. jobject eventObject = createEvent(env, event);
  204. if(eventObject == NULL)
  205. {
  206. return;
  207. }
  208. switch(event)
  209. {
  210. case ZT1_EVENT_UP:
  211. case ZT1_EVENT_OFFLINE:
  212. case ZT1_EVENT_ONLINE:
  213. case ZT1_EVENT_DOWN:
  214. case ZT1_EVENT_FATAL_ERROR_IDENTITY_COLLISION:
  215. {
  216. LOGV("Regular Event");
  217. // call onEvent()
  218. env->CallVoidMethod(ref->eventListener, onEventMethod, eventObject);
  219. }
  220. break;
  221. case ZT1_EVENT_SAW_MORE_RECENT_VERSION:
  222. {
  223. LOGV("Version Event");
  224. // call onOutOfDate()
  225. if(data != NULL)
  226. {
  227. int *version = (int*)data;
  228. jobject verisonObj = newVersion(env, version[0], version[1], version[2], 0);
  229. env->CallVoidMethod(ref->eventListener, onOutOfDateMethod, verisonObj);
  230. }
  231. }
  232. break;
  233. case ZT1_EVENT_AUTHENTICATION_FAILURE:
  234. case ZT1_EVENT_INVALID_PACKET:
  235. {
  236. LOGV("Network Error Event");
  237. // call onNetworkError()
  238. if(data != NULL)
  239. {
  240. sockaddr_storage *addr = (sockaddr_storage*)data;
  241. jobject addressObj = newInetSocketAddress(env, *addr);
  242. env->CallVoidMethod(ref->eventListener, onNetworkErrorMethod, addressObj);
  243. }
  244. }
  245. break;
  246. case ZT1_EVENT_TRACE:
  247. {
  248. LOGV("Trace Event");
  249. // call onTrace()
  250. if(data != NULL)
  251. {
  252. const char* message = (const char*)data;
  253. jstring messageStr = env->NewStringUTF(message);
  254. env->CallVoidMethod(ref->eventListener, onTraceMethod, messageStr);
  255. }
  256. }
  257. break;
  258. }
  259. }
  260. long DataStoreGetFunction(ZT1_Node *node,void *userData,
  261. const char *objectName,
  262. void *buffer,
  263. unsigned long bufferSize,
  264. unsigned long bufferIndex,
  265. unsigned long *out_objectSize)
  266. {
  267. JniRef *ref = (JniRef*)userData;
  268. JNIEnv *env = NULL;
  269. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  270. jclass dataStoreGetClass = env->GetObjectClass(ref->dataStoreGetListener);
  271. if(dataStoreGetClass == NULL)
  272. {
  273. LOGE("Couldn't find class for DataStoreGetListener instance");
  274. return -2;
  275. }
  276. jmethodID dataStoreGetCallbackMethod = lookup.findMethod(
  277. dataStoreGetClass,
  278. "onDataStoreGet",
  279. "(Ljava/lang/String;[BJ[J)J");
  280. if(dataStoreGetCallbackMethod == NULL)
  281. {
  282. LOGE("Couldn't find onDataStoreGet method");
  283. return -2;
  284. }
  285. jstring nameStr = env->NewStringUTF(objectName);
  286. if(nameStr == NULL)
  287. {
  288. LOGE("Error creating name string object");
  289. return -2; // out of memory
  290. }
  291. jbyteArray bufferObj = env->NewByteArray(bufferSize);
  292. if(bufferObj == NULL)
  293. {
  294. LOGE("Error creating byte[] buffer of size: %lu", bufferSize);
  295. return -2;
  296. }
  297. jlongArray objectSizeObj = env->NewLongArray(1);
  298. if(objectSizeObj == NULL)
  299. {
  300. LOGE("Error creating long[1] array for actual object size");
  301. return -2; // couldn't create long[1] array
  302. }
  303. LOGV("Calling onDataStoreGet(%s, %p, %lu, %p)",
  304. objectName, buffer, bufferIndex, objectSizeObj);
  305. long retval = (long)env->CallLongMethod(
  306. ref->dataStoreGetListener, dataStoreGetCallbackMethod,
  307. nameStr, bufferObj, (jlong)bufferIndex, objectSizeObj);
  308. if(retval > 0)
  309. {
  310. void *data = env->GetPrimitiveArrayCritical(bufferObj, NULL);
  311. memcpy(buffer, data, retval);
  312. env->ReleasePrimitiveArrayCritical(bufferObj, data, 0);
  313. jlong *objSize = (jlong*)env->GetPrimitiveArrayCritical(objectSizeObj, NULL);
  314. *out_objectSize = (unsigned long)objSize[0];
  315. env->ReleasePrimitiveArrayCritical(objectSizeObj, objSize, 0);
  316. }
  317. LOGI("Out Object Size: %lu", *out_objectSize);
  318. return retval;
  319. }
  320. int DataStorePutFunction(ZT1_Node *node,void *userData,
  321. const char *objectName,
  322. const void *buffer,
  323. unsigned long bufferSize,
  324. int secure)
  325. {
  326. JniRef *ref = (JniRef*)userData;
  327. JNIEnv *env = NULL;
  328. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  329. jclass dataStorePutClass = env->GetObjectClass(ref->dataStorePutListener);
  330. if(dataStorePutClass == NULL)
  331. {
  332. LOGE("Couldn't find class for DataStorePutListener instance");
  333. return -1;
  334. }
  335. jmethodID dataStorePutCallbackMethod = lookup.findMethod(
  336. dataStorePutClass,
  337. "onDataStorePut",
  338. "(Ljava/lang/String;[BZ)I");
  339. if(dataStorePutCallbackMethod == NULL)
  340. {
  341. LOGE("Couldn't find onDataStorePut method");
  342. return -2;
  343. }
  344. jmethodID deleteMethod = lookup.findMethod(dataStorePutClass,
  345. "onDelete", "(Ljava/lang/String;)I");
  346. if(deleteMethod == NULL)
  347. {
  348. LOGE("Couldn't find onDelete method");
  349. return -3;
  350. }
  351. jstring nameStr = env->NewStringUTF(objectName);
  352. if(buffer == NULL)
  353. {
  354. // delete operation
  355. return env->CallIntMethod(
  356. ref->dataStorePutListener, deleteMethod, nameStr);
  357. }
  358. else
  359. {
  360. // set operation
  361. jbyteArray bufferObj = env->NewByteArray(bufferSize);
  362. if(env->ExceptionCheck() || bufferObj == NULL)
  363. {
  364. LOGE("Error creating byte array buffer!");
  365. return -4;
  366. }
  367. env->SetByteArrayRegion(bufferObj, 0, bufferSize, (jbyte*)buffer);
  368. bool bsecure = secure != 0;
  369. return env->CallIntMethod(ref->dataStorePutListener,
  370. dataStorePutCallbackMethod,
  371. nameStr, bufferObj, bsecure);
  372. }
  373. }
  374. int WirePacketSendFunction(ZT1_Node *node,void *userData,\
  375. const struct sockaddr_storage *address,
  376. const void *buffer,
  377. unsigned int bufferSize)
  378. {
  379. LOGV("WirePacketSendFunction(%p, %p, %d)", address, buffer, bufferSize);
  380. JniRef *ref = (JniRef*)userData;
  381. assert(ref->node == node);
  382. JNIEnv *env = NULL;
  383. ref->jvm->GetEnv((void**)&env, JNI_VERSION_1_6);
  384. jclass packetSenderClass = env->GetObjectClass(ref->packetSender);
  385. if(packetSenderClass == NULL)
  386. {
  387. LOGE("Couldn't find class for PacketSender instance");
  388. return -1;
  389. }
  390. jmethodID packetSenderCallbackMethod = lookup.findMethod(packetSenderClass,
  391. "onSendPacketRequested", "(Ljava/net/InetSocketAddress;[B)I");
  392. if(packetSenderCallbackMethod == NULL)
  393. {
  394. LOGE("Couldn't find onSendPacketRequested method");
  395. return -2;
  396. }
  397. jobject addressObj = newInetSocketAddress(env, *address);
  398. jbyteArray bufferObj = env->NewByteArray(bufferSize);
  399. env->SetByteArrayRegion(bufferObj, 0, bufferSize, (jbyte*)buffer);
  400. int retval = env->CallIntMethod(ref->packetSender, packetSenderCallbackMethod, addressObj, bufferObj);
  401. LOGV("JNI Packet Sender returned: %d", retval);
  402. return retval;
  403. }
  404. typedef std::map<uint64_t, JniRef*> NodeMap;
  405. static NodeMap nodeMap;
  406. ZT1_Node* findNode(uint64_t nodeId)
  407. {
  408. NodeMap::iterator found = nodeMap.find(nodeId);
  409. if(found != nodeMap.end())
  410. {
  411. JniRef *ref = found->second;
  412. return ref->node;
  413. }
  414. return NULL;
  415. }
  416. }
  417. JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved)
  418. {
  419. lookup.setJavaVM(vm);
  420. return JNI_VERSION_1_6;
  421. }
  422. JNIEXPORT void JNICALL JNI_OnUnload(JavaVM *vm, void *reserved)
  423. {
  424. }
  425. /*
  426. * Class: com_zerotier_sdk_Node
  427. * Method: node_init
  428. * Signature: (J)Lcom/zerotier/sdk/ResultCode;
  429. */
  430. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_node_1init(
  431. JNIEnv *env, jobject obj, jlong now)
  432. {
  433. LOGV("Creating ZT1_Node struct");
  434. jobject resultObject = createResultObject(env, ZT1_RESULT_OK);
  435. ZT1_Node *node;
  436. JniRef *ref = new JniRef;
  437. ref->id = (uint64_t)now;
  438. env->GetJavaVM(&ref->jvm);
  439. jclass cls = env->GetObjectClass(obj);
  440. jfieldID fid = lookup.findField(
  441. cls, "getListener", "Lcom/zerotier/sdk/DataStoreGetListener;");
  442. if(fid == NULL)
  443. {
  444. return NULL; // exception already thrown
  445. }
  446. jobject tmp = env->GetObjectField(obj, fid);
  447. if(tmp == NULL)
  448. {
  449. return NULL;
  450. }
  451. ref->dataStoreGetListener = env->NewGlobalRef(tmp);
  452. fid = lookup.findField(
  453. cls, "putListener", "Lcom/zerotier/sdk/DataStorePutListener;");
  454. if(fid == NULL)
  455. {
  456. return NULL; // exception already thrown
  457. }
  458. tmp = env->GetObjectField(obj, fid);
  459. if(tmp == NULL)
  460. {
  461. return NULL;
  462. }
  463. ref->dataStorePutListener = env->NewGlobalRef(tmp);
  464. fid = lookup.findField(
  465. cls, "sender", "Lcom/zerotier/sdk/PacketSender;");
  466. if(fid == NULL)
  467. {
  468. return NULL; // exception already thrown
  469. }
  470. tmp = env->GetObjectField(obj, fid);
  471. if(tmp == NULL)
  472. {
  473. return NULL;
  474. }
  475. ref->packetSender = env->NewGlobalRef(tmp);
  476. fid = lookup.findField(
  477. cls, "frameListener", "Lcom/zerotier/sdk/VirtualNetworkFrameListener;");
  478. if(fid == NULL)
  479. {
  480. return NULL; // exception already thrown
  481. }
  482. tmp = env->GetObjectField(obj, fid);
  483. if(tmp == NULL)
  484. {
  485. return NULL;
  486. }
  487. ref->frameListener = env->NewGlobalRef(tmp);
  488. fid = lookup.findField(
  489. cls, "configListener", "Lcom/zerotier/sdk/VirtualNetworkConfigListener;");
  490. if(fid == NULL)
  491. {
  492. return NULL; // exception already thrown
  493. }
  494. tmp = env->GetObjectField(obj, fid);
  495. if(tmp == NULL)
  496. {
  497. return NULL;
  498. }
  499. ref->configListener = env->NewGlobalRef(tmp);
  500. fid = lookup.findField(
  501. cls, "eventListener", "Lcom/zerotier/sdk/EventListener;");
  502. if(fid == NULL)
  503. {
  504. return NULL;
  505. }
  506. tmp = env->GetObjectField(obj, fid);
  507. if(tmp == NULL)
  508. {
  509. return NULL;
  510. }
  511. ref->eventListener = env->NewGlobalRef(tmp);
  512. ZT1_ResultCode rc = ZT1_Node_new(
  513. &node,
  514. ref,
  515. (uint64_t)now,
  516. &DataStoreGetFunction,
  517. &DataStorePutFunction,
  518. &WirePacketSendFunction,
  519. &VirtualNetworkFrameFunctionCallback,
  520. &VirtualNetworkConfigFunctionCallback,
  521. &EventCallback);
  522. if(rc != ZT1_RESULT_OK)
  523. {
  524. LOGE("Error creating Node: %d", rc);
  525. resultObject = createResultObject(env, rc);
  526. if(node)
  527. {
  528. ZT1_Node_delete(node);
  529. node = NULL;
  530. }
  531. delete ref;
  532. ref = NULL;
  533. return resultObject;
  534. }
  535. ref->node = node;
  536. nodeMap.insert(std::make_pair(ref->id, ref));
  537. return resultObject;
  538. }
  539. /*
  540. * Class: com_zerotier_sdk_Node
  541. * Method: node_delete
  542. * Signature: (J)V
  543. */
  544. JNIEXPORT void JNICALL Java_com_zerotier_sdk_Node_node_1delete(
  545. JNIEnv *env, jobject obj, jlong id)
  546. {
  547. LOGV("Destroying ZT1_Node struct");
  548. uint64_t nodeId = (uint64_t)id;
  549. NodeMap::iterator found = nodeMap.find(nodeId);
  550. if(found != nodeMap.end())
  551. {
  552. JniRef *ref = found->second;
  553. nodeMap.erase(found);
  554. ZT1_Node_delete(ref->node);
  555. delete ref;
  556. ref = NULL;
  557. }
  558. else
  559. {
  560. LOGE("Attempted to delete a node that doesn't exist!");
  561. }
  562. }
  563. /*
  564. * Class: com_zerotier_sdk_Node
  565. * Method: processVirtualNetworkFrame
  566. * Signature: (JJJJJII[B[J)Lcom/zerotier/sdk/ResultCode;
  567. */
  568. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_processVirtualNetworkFrame(
  569. JNIEnv *env, jobject obj,
  570. jlong id,
  571. jlong in_now,
  572. jlong in_nwid,
  573. jlong in_sourceMac,
  574. jlong in_destMac,
  575. jint in_etherType,
  576. jint in_vlanId,
  577. jbyteArray in_frameData,
  578. jlongArray out_nextBackgroundTaskDeadline)
  579. {
  580. uint64_t nodeId = (uint64_t) id;
  581. ZT1_Node *node = findNode(nodeId);
  582. if(node == NULL)
  583. {
  584. // cannot find valid node. We should never get here.
  585. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  586. }
  587. unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
  588. if(nbtd_len < 1)
  589. {
  590. // array for next background task length has 0 elements!
  591. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  592. }
  593. uint64_t now = (uint64_t)in_now;
  594. uint64_t nwid = (uint64_t)in_nwid;
  595. uint64_t sourceMac = (uint64_t)in_sourceMac;
  596. uint64_t destMac = (uint64_t)in_destMac;
  597. unsigned int etherType = (unsigned int)in_etherType;
  598. unsigned int vlanId = (unsigned int)in_vlanId;
  599. unsigned int frameLength = env->GetArrayLength(in_frameData);
  600. void *frameData = env->GetPrimitiveArrayCritical(in_frameData, NULL);
  601. void *localData = malloc(frameLength);
  602. memcpy(localData, frameData, frameLength);
  603. env->ReleasePrimitiveArrayCritical(in_frameData, frameData, 0);
  604. uint64_t nextBackgroundTaskDeadline = 0;
  605. ZT1_ResultCode rc = ZT1_Node_processVirtualNetworkFrame(
  606. node,
  607. now,
  608. nwid,
  609. sourceMac,
  610. destMac,
  611. etherType,
  612. vlanId,
  613. (const void*)localData,
  614. frameLength,
  615. &nextBackgroundTaskDeadline);
  616. jlong *outDeadline = (jlong*)env->GetPrimitiveArrayCritical(out_nextBackgroundTaskDeadline, NULL);
  617. outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
  618. env->ReleasePrimitiveArrayCritical(out_nextBackgroundTaskDeadline, outDeadline, 0);
  619. return createResultObject(env, rc);
  620. }
  621. /*
  622. * Class: com_zerotier_sdk_Node
  623. * Method: processWirePacket
  624. * Signature: (JJLjava/net/InetSocketAddress;I[B[J)Lcom/zerotier/sdk/ResultCode;
  625. */
  626. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_processWirePacket(
  627. JNIEnv *env, jobject obj,
  628. jlong id,
  629. jlong in_now,
  630. jobject in_remoteAddress,
  631. jbyteArray in_packetData,
  632. jlongArray out_nextBackgroundTaskDeadline)
  633. {
  634. uint64_t nodeId = (uint64_t) id;
  635. ZT1_Node *node = findNode(nodeId);
  636. if(node == NULL)
  637. {
  638. // cannot find valid node. We should never get here.
  639. LOGE("Couldn't find a valid node!");
  640. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  641. }
  642. unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
  643. if(nbtd_len < 1)
  644. {
  645. LOGE("nbtd_len < 1");
  646. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  647. }
  648. uint64_t now = (uint64_t)in_now;
  649. // get the java.net.InetSocketAddress class and getAddress() method
  650. jclass inetAddressClass = lookup.findClass("java/net/InetAddress");
  651. if(inetAddressClass == NULL)
  652. {
  653. LOGE("Can't find InetAddress class");
  654. // can't find java.net.InetAddress
  655. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  656. }
  657. jmethodID getAddressMethod = lookup.findMethod(
  658. inetAddressClass, "getAddress", "()[B");
  659. if(getAddressMethod == NULL)
  660. {
  661. // cant find InetAddress.getAddres()
  662. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  663. }
  664. jclass InetSocketAddressClass = lookup.findClass("java/net/InetSocketAddress");
  665. if(InetSocketAddressClass == NULL)
  666. {
  667. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  668. }
  669. jmethodID inetSockGetAddressMethod = lookup.findMethod(
  670. InetSocketAddressClass, "getAddress", "()Ljava/net/InetAddress;");
  671. jobject addrObject = env->CallObjectMethod(in_remoteAddress, inetSockGetAddressMethod);
  672. if(addrObject == NULL)
  673. {
  674. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  675. }
  676. jmethodID inetSock_getPort = lookup.findMethod(
  677. InetSocketAddressClass, "getPort", "()I");
  678. if(env->ExceptionCheck() || inetSock_getPort == NULL)
  679. {
  680. LOGE("Couldn't find getPort method on InetSocketAddress");
  681. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  682. }
  683. // call InetSocketAddress.getPort()
  684. int port = env->CallIntMethod(in_remoteAddress, inetSock_getPort);
  685. if(env->ExceptionCheck())
  686. {
  687. LOGE("Exception calling InetSocketAddress.getPort()");
  688. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  689. }
  690. // Call InetAddress.getAddress()
  691. jbyteArray addressArray = (jbyteArray)env->CallObjectMethod(addrObject, getAddressMethod);
  692. if(addressArray == NULL)
  693. {
  694. // unable to call getAddress()
  695. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  696. }
  697. unsigned int addrSize = env->GetArrayLength(addressArray);
  698. // get the address bytes
  699. jbyte *addr = (jbyte*)env->GetPrimitiveArrayCritical(addressArray, NULL);
  700. sockaddr_storage remoteAddress = {};
  701. if(addrSize == 16)
  702. {
  703. // IPV6 address
  704. sockaddr_in6 ipv6 = {};
  705. ipv6.sin6_family = AF_INET6;
  706. ipv6.sin6_port = htons(port);
  707. memcpy(ipv6.sin6_addr.s6_addr, addr, 16);
  708. memcpy(&remoteAddress, &ipv6, sizeof(sockaddr_in6));
  709. }
  710. else if(addrSize == 4)
  711. {
  712. // IPV4 address
  713. sockaddr_in ipv4 = {};
  714. ipv4.sin_family = AF_INET;
  715. ipv4.sin_port = htons(port);
  716. memcpy(&ipv4.sin_addr, addr, 4);
  717. memcpy(&remoteAddress, &ipv4, sizeof(sockaddr_in));
  718. }
  719. else
  720. {
  721. // unknown address type
  722. env->ReleasePrimitiveArrayCritical(addressArray, addr, 0);
  723. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  724. }
  725. env->ReleasePrimitiveArrayCritical(addressArray, addr, 0);
  726. unsigned int packetLength = env->GetArrayLength(in_packetData);
  727. void *packetData = env->GetPrimitiveArrayCritical(in_packetData, NULL);
  728. void *localData = malloc(packetLength);
  729. memcpy(localData, packetData, packetLength);
  730. env->ReleasePrimitiveArrayCritical(in_packetData, packetData, 0);
  731. uint64_t nextBackgroundTaskDeadline = 0;
  732. ZT1_ResultCode rc = ZT1_Node_processWirePacket(
  733. node,
  734. now,
  735. &remoteAddress,
  736. localData,
  737. packetLength,
  738. &nextBackgroundTaskDeadline);
  739. if(rc != ZT1_RESULT_OK)
  740. {
  741. LOGE("ZT1_Node_processWirePacket returned: %d", rc);
  742. }
  743. free(localData);
  744. jlong *outDeadline = (jlong*)env->GetPrimitiveArrayCritical(out_nextBackgroundTaskDeadline, NULL);
  745. outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
  746. env->ReleasePrimitiveArrayCritical(out_nextBackgroundTaskDeadline, outDeadline, 0);
  747. return createResultObject(env, rc);
  748. }
  749. /*
  750. * Class: com_zerotier_sdk_Node
  751. * Method: processBackgroundTasks
  752. * Signature: (JJ[J)Lcom/zerotier/sdk/ResultCode;
  753. */
  754. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_processBackgroundTasks(
  755. JNIEnv *env, jobject obj,
  756. jlong id,
  757. jlong in_now,
  758. jlongArray out_nextBackgroundTaskDeadline)
  759. {
  760. uint64_t nodeId = (uint64_t) id;
  761. ZT1_Node *node = findNode(nodeId);
  762. if(node == NULL)
  763. {
  764. // cannot find valid node. We should never get here.
  765. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  766. }
  767. unsigned int nbtd_len = env->GetArrayLength(out_nextBackgroundTaskDeadline);
  768. if(nbtd_len < 1)
  769. {
  770. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  771. }
  772. uint64_t now = (uint64_t)in_now;
  773. uint64_t nextBackgroundTaskDeadline = 0;
  774. ZT1_ResultCode rc = ZT1_Node_processBackgroundTasks(node, now, &nextBackgroundTaskDeadline);
  775. jlong *outDeadline = (jlong*)env->GetPrimitiveArrayCritical(out_nextBackgroundTaskDeadline, NULL);
  776. outDeadline[0] = (jlong)nextBackgroundTaskDeadline;
  777. env->ReleasePrimitiveArrayCritical(out_nextBackgroundTaskDeadline, outDeadline, 0);
  778. return createResultObject(env, rc);
  779. }
  780. /*
  781. * Class: com_zerotier_sdk_Node
  782. * Method: join
  783. * Signature: (JJ)Lcom/zerotier/sdk/ResultCode;
  784. */
  785. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_join(
  786. JNIEnv *env, jobject obj, jlong id, jlong in_nwid)
  787. {
  788. uint64_t nodeId = (uint64_t) id;
  789. ZT1_Node *node = findNode(nodeId);
  790. if(node == NULL)
  791. {
  792. // cannot find valid node. We should never get here.
  793. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  794. }
  795. uint64_t nwid = (uint64_t)in_nwid;
  796. ZT1_ResultCode rc = ZT1_Node_join(node, nwid);
  797. return createResultObject(env, rc);
  798. }
  799. /*
  800. * Class: com_zerotier_sdk_Node
  801. * Method: leave
  802. * Signature: (JJ)Lcom/zerotier/sdk/ResultCode;
  803. */
  804. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_leave(
  805. JNIEnv *env, jobject obj, jlong id, jlong in_nwid)
  806. {
  807. uint64_t nodeId = (uint64_t) id;
  808. ZT1_Node *node = findNode(nodeId);
  809. if(node == NULL)
  810. {
  811. // cannot find valid node. We should never get here.
  812. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  813. }
  814. uint64_t nwid = (uint64_t)in_nwid;
  815. ZT1_ResultCode rc = ZT1_Node_leave(node, nwid);
  816. return createResultObject(env, rc);
  817. }
  818. /*
  819. * Class: com_zerotier_sdk_Node
  820. * Method: multicastSubscribe
  821. * Signature: (JJJJ)Lcom/zerotier/sdk/ResultCode;
  822. */
  823. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_multicastSubscribe(
  824. JNIEnv *env, jobject obj,
  825. jlong id,
  826. jlong in_nwid,
  827. jlong in_multicastGroup,
  828. jlong in_multicastAdi)
  829. {
  830. uint64_t nodeId = (uint64_t) id;
  831. ZT1_Node *node = findNode(nodeId);
  832. if(node == NULL)
  833. {
  834. // cannot find valid node. We should never get here.
  835. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  836. }
  837. uint64_t nwid = (uint64_t)in_nwid;
  838. uint64_t multicastGroup = (uint64_t)in_multicastGroup;
  839. unsigned long multicastAdi = (unsigned long)in_multicastAdi;
  840. ZT1_ResultCode rc = ZT1_Node_multicastSubscribe(
  841. node, nwid, multicastGroup, multicastAdi);
  842. return createResultObject(env, rc);
  843. }
  844. /*
  845. * Class: com_zerotier_sdk_Node
  846. * Method: multicastUnsubscribe
  847. * Signature: (JJJJ)Lcom/zerotier/sdk/ResultCode;
  848. */
  849. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_multicastUnsubscribe(
  850. JNIEnv *env, jobject obj,
  851. jlong id,
  852. jlong in_nwid,
  853. jlong in_multicastGroup,
  854. jlong in_multicastAdi)
  855. {
  856. uint64_t nodeId = (uint64_t) id;
  857. ZT1_Node *node = findNode(nodeId);
  858. if(node == NULL)
  859. {
  860. // cannot find valid node. We should never get here.
  861. return createResultObject(env, ZT1_RESULT_FATAL_ERROR_INTERNAL);
  862. }
  863. uint64_t nwid = (uint64_t)in_nwid;
  864. uint64_t multicastGroup = (uint64_t)in_multicastGroup;
  865. unsigned long multicastAdi = (unsigned long)in_multicastAdi;
  866. ZT1_ResultCode rc = ZT1_Node_multicastUnsubscribe(
  867. node, nwid, multicastGroup, multicastAdi);
  868. return createResultObject(env, rc);
  869. }
  870. /*
  871. * Class: com_zerotier_sdk_Node
  872. * Method: address
  873. * Signature: (J)J
  874. */
  875. JNIEXPORT jlong JNICALL Java_com_zerotier_sdk_Node_address(
  876. JNIEnv *env , jobject obj, jlong id)
  877. {
  878. uint64_t nodeId = (uint64_t) id;
  879. ZT1_Node *node = findNode(nodeId);
  880. if(node == NULL)
  881. {
  882. // cannot find valid node. We should never get here.
  883. return 0;
  884. }
  885. uint64_t address = ZT1_Node_address(node);
  886. return (jlong)address;
  887. }
  888. /*
  889. * Class: com_zerotier_sdk_Node
  890. * Method: status
  891. * Signature: (J)Lcom/zerotier/sdk/NodeStatus;
  892. */
  893. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_status
  894. (JNIEnv *env, jobject obj, jlong id)
  895. {
  896. uint64_t nodeId = (uint64_t) id;
  897. ZT1_Node *node = findNode(nodeId);
  898. if(node == NULL)
  899. {
  900. // cannot find valid node. We should never get here.
  901. return 0;
  902. }
  903. jclass nodeStatusClass = NULL;
  904. jmethodID nodeStatusConstructor = NULL;
  905. // create a com.zerotier.sdk.NodeStatus object
  906. nodeStatusClass = lookup.findClass("com/zerotier/sdk/NodeStatus");
  907. if(nodeStatusClass == NULL)
  908. {
  909. return NULL;
  910. }
  911. nodeStatusConstructor = lookup.findMethod(
  912. nodeStatusClass, "<init>", "()V");
  913. if(nodeStatusConstructor == NULL)
  914. {
  915. return NULL;
  916. }
  917. jobject nodeStatusObj = env->NewObject(nodeStatusClass, nodeStatusConstructor);
  918. if(nodeStatusObj == NULL)
  919. {
  920. return NULL;
  921. }
  922. ZT1_NodeStatus nodeStatus;
  923. ZT1_Node_status(node, &nodeStatus);
  924. jfieldID addressField = NULL;
  925. jfieldID publicIdentityField = NULL;
  926. jfieldID secretIdentityField = NULL;
  927. jfieldID onlineField = NULL;
  928. addressField = lookup.findField(nodeStatusClass, "address", "J");
  929. if(addressField == NULL)
  930. {
  931. return NULL;
  932. }
  933. publicIdentityField = lookup.findField(nodeStatusClass, "publicIdentity", "Ljava/lang/String;");
  934. if(publicIdentityField == NULL)
  935. {
  936. return NULL;
  937. }
  938. secretIdentityField = lookup.findField(nodeStatusClass, "secretIdentity", "Ljava/lang/String;");
  939. if(secretIdentityField == NULL)
  940. {
  941. return NULL;
  942. }
  943. onlineField = lookup.findField(nodeStatusClass, "online", "Z");
  944. if(onlineField == NULL)
  945. {
  946. return NULL;
  947. }
  948. env->SetLongField(nodeStatusObj, addressField, nodeStatus.address);
  949. jstring pubIdentStr = env->NewStringUTF(nodeStatus.publicIdentity);
  950. if(pubIdentStr == NULL)
  951. {
  952. return NULL; // out of memory
  953. }
  954. env->SetObjectField(nodeStatusObj, publicIdentityField, pubIdentStr);
  955. jstring secIdentStr = env->NewStringUTF(nodeStatus.secretIdentity);
  956. if(secIdentStr == NULL)
  957. {
  958. return NULL; // out of memory
  959. }
  960. env->SetObjectField(nodeStatusObj, secretIdentityField, secIdentStr);
  961. env->SetBooleanField(nodeStatusObj, onlineField, nodeStatus.online);
  962. return nodeStatusObj;
  963. }
  964. /*
  965. * Class: com_zerotier_sdk_Node
  966. * Method: networkConfig
  967. * Signature: (J)Lcom/zerotier/sdk/VirtualNetworkConfig;
  968. */
  969. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_networkConfig(
  970. JNIEnv *env, jobject obj, jlong id, jlong nwid)
  971. {
  972. uint64_t nodeId = (uint64_t) id;
  973. ZT1_Node *node = findNode(nodeId);
  974. if(node == NULL)
  975. {
  976. // cannot find valid node. We should never get here.
  977. return 0;
  978. }
  979. ZT1_VirtualNetworkConfig *vnetConfig = ZT1_Node_networkConfig(node, nwid);
  980. jobject vnetConfigObject = newNetworkConfig(env, *vnetConfig);
  981. ZT1_Node_freeQueryResult(node, vnetConfig);
  982. return vnetConfigObject;
  983. }
  984. /*
  985. * Class: com_zerotier_sdk_Node
  986. * Method: version
  987. * Signature: (J)Lcom/zerotier/sdk/Version;
  988. */
  989. JNIEXPORT jobject JNICALL Java_com_zerotier_sdk_Node_version(
  990. JNIEnv *env, jobject obj)
  991. {
  992. int major = 0;
  993. int minor = 0;
  994. int revision = 0;
  995. unsigned long featureFlags = 0;
  996. ZT1_version(&major, &minor, &revision, &featureFlags);
  997. return newVersion(env, major, minor, revision, featureFlags);
  998. }
  999. /*
  1000. * Class: com_zerotier_sdk_Node
  1001. * Method: peers
  1002. * Signature: (J)[Lcom/zerotier/sdk/Peer;
  1003. */
  1004. JNIEXPORT jobjectArray JNICALL Java_com_zerotier_sdk_Node_peers(
  1005. JNIEnv *env, jobject obj, jlong id)
  1006. {
  1007. uint64_t nodeId = (uint64_t) id;
  1008. ZT1_Node *node = findNode(nodeId);
  1009. if(node == NULL)
  1010. {
  1011. // cannot find valid node. We should never get here.
  1012. return 0;
  1013. }
  1014. ZT1_PeerList *peerList = ZT1_Node_peers(node);
  1015. if(peerList == NULL)
  1016. {
  1017. LOGE("ZT1_Node_peers returned NULL");
  1018. return NULL;
  1019. }
  1020. int peerCount = peerList->peerCount * 100;
  1021. LOGV("Ensure Local Capacity: %d", peerCount);
  1022. if(env->EnsureLocalCapacity(peerCount))
  1023. {
  1024. LOGE("EnsureLocalCapacity failed!!");
  1025. ZT1_Node_freeQueryResult(node, peerList);
  1026. return NULL;
  1027. }
  1028. jclass peerClass = lookup.findClass("com/zerotier/sdk/Peer");
  1029. if(env->ExceptionCheck() || peerClass == NULL)
  1030. {
  1031. LOGE("Error finding Peer class");
  1032. ZT1_Node_freeQueryResult(node, peerList);
  1033. return NULL;
  1034. }
  1035. jobjectArray peerArrayObj = env->NewObjectArray(
  1036. peerList->peerCount, peerClass, NULL);
  1037. if(env->ExceptionCheck() || peerArrayObj == NULL)
  1038. {
  1039. LOGE("Error creating Peer[] array");
  1040. ZT1_Node_freeQueryResult(node, peerList);
  1041. return NULL;
  1042. }
  1043. for(unsigned int i = 0; i < peerList->peerCount; ++i)
  1044. {
  1045. jobject peerObj = newPeer(env, peerList->peers[i]);
  1046. env->SetObjectArrayElement(peerArrayObj, i, peerObj);
  1047. if(env->ExceptionCheck())
  1048. {
  1049. LOGE("Error assigning Peer object to array");
  1050. break;
  1051. }
  1052. }
  1053. ZT1_Node_freeQueryResult(node, peerList);
  1054. peerList = NULL;
  1055. return peerArrayObj;
  1056. }
  1057. /*
  1058. * Class: com_zerotier_sdk_Node
  1059. * Method: networks
  1060. * Signature: (J)[Lcom/zerotier/sdk/VirtualNetworkConfig;
  1061. */
  1062. JNIEXPORT jobjectArray JNICALL Java_com_zerotier_sdk_Node_networks(
  1063. JNIEnv *env, jobject obj, jlong id)
  1064. {
  1065. uint64_t nodeId = (uint64_t) id;
  1066. ZT1_Node *node = findNode(nodeId);
  1067. if(node == NULL)
  1068. {
  1069. // cannot find valid node. We should never get here.
  1070. return 0;
  1071. }
  1072. ZT1_VirtualNetworkList *networkList = ZT1_Node_networks(node);
  1073. if(networkList == NULL)
  1074. {
  1075. return NULL;
  1076. }
  1077. jclass vnetConfigClass = lookup.findClass("com/zerotier/sdk/VirtualNetworkConfig");
  1078. if(env->ExceptionCheck() || vnetConfigClass == NULL)
  1079. {
  1080. LOGE("Error finding VirtualNetworkConfig class");
  1081. ZT1_Node_freeQueryResult(node, networkList);
  1082. return NULL;
  1083. }
  1084. jobjectArray networkListObject = env->NewObjectArray(
  1085. networkList->networkCount, vnetConfigClass, NULL);
  1086. if(env->ExceptionCheck() || networkListObject == NULL)
  1087. {
  1088. LOGE("Error creating VirtualNetworkConfig[] array");
  1089. ZT1_Node_freeQueryResult(node, networkList);
  1090. return NULL;
  1091. }
  1092. for(unsigned int i = 0; i < networkList->networkCount; ++i)
  1093. {
  1094. jobject networkObject = newNetworkConfig(env, networkList->networks[i]);
  1095. env->SetObjectArrayElement(networkListObject, i, networkObject);
  1096. if(env->ExceptionCheck())
  1097. {
  1098. LOGE("Error assigning VirtualNetworkConfig object to array");
  1099. break;
  1100. }
  1101. }
  1102. ZT1_Node_freeQueryResult(node, networkList);
  1103. return networkListObject;
  1104. }
  1105. #ifdef __cplusplus
  1106. } // extern "C"
  1107. #endif