com_zerotierone_sdk_Node.cpp 37 KB

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