ZT_jniutils.cpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518
  1. /*
  2. * ZeroTier One - Network Virtualization Everywhere
  3. * Copyright (C) 2011-2016 ZeroTier, Inc. https://www.zerotier.com/
  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. #include "ZT_jniutils.h"
  19. #include "ZT_jnicache.h"
  20. #include <string>
  21. #include <cassert>
  22. #include <arpa/inet.h>
  23. #include <netinet/in.h>
  24. #include <sys/types.h>
  25. #include <sys/socket.h>
  26. #define LOG_TAG "Utils"
  27. jobject createResultObject(JNIEnv *env, ZT_ResultCode code)
  28. {
  29. jobject resultObject = env->CallStaticObjectMethod(ResultCode_class, ResultCode_fromInt_method, code);
  30. if(env->ExceptionCheck() || resultObject == NULL) {
  31. LOGE("Error creating ResultCode object");
  32. return NULL;
  33. }
  34. return resultObject;
  35. }
  36. jobject createVirtualNetworkStatus(JNIEnv *env, ZT_VirtualNetworkStatus status)
  37. {
  38. jobject statusObject = env->CallStaticObjectMethod(VirtualNetworkStatus_class, VirtualNetworkStatus_fromInt_method, status);
  39. if (env->ExceptionCheck() || statusObject == NULL) {
  40. LOGE("Error creating VirtualNetworkStatus object");
  41. return NULL;
  42. }
  43. return statusObject;
  44. }
  45. jobject createEvent(JNIEnv *env, ZT_Event event)
  46. {
  47. jobject eventObject = env->CallStaticObjectMethod(Event_class, Event_fromInt_method, event);
  48. if (env->ExceptionCheck() || eventObject == NULL) {
  49. LOGE("Error creating Event object");
  50. return NULL;
  51. }
  52. return eventObject;
  53. }
  54. jobject createPeerRole(JNIEnv *env, ZT_PeerRole role)
  55. {
  56. jobject peerRoleObject = env->CallStaticObjectMethod(PeerRole_class, PeerRole_fromInt_method, role);
  57. if (env->ExceptionCheck() || peerRoleObject == NULL) {
  58. LOGE("Error creating PeerRole object");
  59. return NULL;
  60. }
  61. return peerRoleObject;
  62. }
  63. jobject createVirtualNetworkType(JNIEnv *env, ZT_VirtualNetworkType type)
  64. {
  65. jobject vntypeObject = env->CallStaticObjectMethod(VirtualNetworkType_class, VirtualNetworkType_fromInt_method, type);
  66. if (env->ExceptionCheck() || vntypeObject == NULL) {
  67. LOGE("Error creating VirtualNetworkType object");
  68. return NULL;
  69. }
  70. return vntypeObject;
  71. }
  72. jobject createVirtualNetworkConfigOperation(JNIEnv *env, ZT_VirtualNetworkConfigOperation op)
  73. {
  74. jobject vnetConfigOpObject = env->CallStaticObjectMethod(VirtualNetworkConfigOperation_class, VirtualNetworkConfigOperation_fromInt_method, op);
  75. if (env->ExceptionCheck() || vnetConfigOpObject == NULL) {
  76. LOGE("Error creating VirtualNetworkConfigOperation object");
  77. return NULL;
  78. }
  79. return vnetConfigOpObject;
  80. }
  81. jobject newInetAddress(JNIEnv *env, const sockaddr_storage &addr)
  82. {
  83. LOGV("newInetAddress");
  84. jobject inetAddressObj = NULL;
  85. switch(addr.ss_family)
  86. {
  87. case AF_INET6:
  88. {
  89. sockaddr_in6 *ipv6 = (sockaddr_in6*)&addr;
  90. const unsigned char *bytes = reinterpret_cast<const unsigned char *>(&ipv6->sin6_addr.s6_addr);
  91. jbyteArray buff = newByteArray(env, bytes, 16);
  92. if(env->ExceptionCheck() || buff == NULL)
  93. {
  94. return NULL;
  95. }
  96. inetAddressObj = env->CallStaticObjectMethod(
  97. InetAddress_class, InetAddress_getByAddress_method, buff);
  98. }
  99. break;
  100. case AF_INET:
  101. {
  102. sockaddr_in *ipv4 = (sockaddr_in*)&addr;
  103. const unsigned char *bytes = reinterpret_cast<const unsigned char *>(&ipv4->sin_addr.s_addr);
  104. jbyteArray buff = newByteArray(env, bytes, 4);
  105. if(env->ExceptionCheck() || buff == NULL)
  106. {
  107. return NULL;
  108. }
  109. inetAddressObj = env->CallStaticObjectMethod(
  110. InetAddress_class, InetAddress_getByAddress_method, buff);
  111. }
  112. break;
  113. }
  114. if(env->ExceptionCheck() || inetAddressObj == NULL) {
  115. LOGE("Error creating InetAddress object");
  116. return NULL;
  117. }
  118. return inetAddressObj;
  119. }
  120. jobject newInetSocketAddress(JNIEnv *env, const sockaddr_storage &addr)
  121. {
  122. LOGV("newInetSocketAddress Called");
  123. jobject inetAddressObject = NULL;
  124. if(addr.ss_family != 0)
  125. {
  126. inetAddressObject = newInetAddress(env, addr);
  127. if(env->ExceptionCheck() || inetAddressObject == NULL)
  128. {
  129. LOGE("Error creating new inet address");
  130. return NULL;
  131. }
  132. }
  133. else
  134. {
  135. return NULL;
  136. }
  137. int port = 0;
  138. switch(addr.ss_family)
  139. {
  140. case AF_INET6:
  141. {
  142. LOGV("IPV6 Address");
  143. sockaddr_in6 *ipv6 = (sockaddr_in6*)&addr;
  144. port = ntohs(ipv6->sin6_port);
  145. LOGV("Port %d", port);
  146. }
  147. break;
  148. case AF_INET:
  149. {
  150. LOGV("IPV4 Address");
  151. sockaddr_in *ipv4 = (sockaddr_in*)&addr;
  152. port = ntohs(ipv4->sin_port);
  153. LOGV("Port: %d", port);
  154. }
  155. break;
  156. default:
  157. {
  158. break;
  159. }
  160. }
  161. jobject inetSocketAddressObject = env->NewObject(InetSocketAddress_class, InetSocketAddress_ctor, inetAddressObject, port);
  162. if(env->ExceptionCheck() || inetSocketAddressObject == NULL) {
  163. LOGE("Error creating InetSocketAddress object");
  164. }
  165. return inetSocketAddressObject;
  166. }
  167. jobject newPeerPhysicalPath(JNIEnv *env, const ZT_PeerPhysicalPath &ppp)
  168. {
  169. LOGV("newPeerPhysicalPath Called");
  170. jobject addressObject = newInetSocketAddress(env, ppp.address);
  171. if(env->ExceptionCheck() || addressObject == NULL) {
  172. LOGE("Error creating InetSocketAddress object");
  173. return NULL;
  174. }
  175. jobject pppObject = env->NewObject(
  176. PeerPhysicalPath_class,
  177. PeerPhysicalPath_ctor,
  178. addressObject,
  179. ppp.lastSend,
  180. ppp.lastReceive,
  181. ppp.preferred);
  182. if(env->ExceptionCheck() || pppObject == NULL)
  183. {
  184. LOGE("Error creating PPP object");
  185. return NULL;
  186. }
  187. return pppObject;
  188. }
  189. jobject newPeer(JNIEnv *env, const ZT_Peer &peer)
  190. {
  191. LOGV("newPeer called");
  192. jobject peerRoleObj = createPeerRole(env, peer.role);
  193. if(env->ExceptionCheck() || peerRoleObj == NULL)
  194. {
  195. return NULL; // out of memory
  196. }
  197. jobjectArray arrayObject = newPeerPhysicalPathArray(env, peer.paths, peer.pathCount);
  198. if (env->ExceptionCheck() || arrayObject == NULL) {
  199. return NULL;
  200. }
  201. jobject peerObject = env->NewObject(
  202. Peer_class,
  203. Peer_ctor,
  204. peer.address,
  205. peer.versionMajor,
  206. peer.versionMinor,
  207. peer.versionRev,
  208. peer.latency,
  209. peerRoleObj,
  210. arrayObject);
  211. if(env->ExceptionCheck() || peerObject == NULL)
  212. {
  213. LOGE("Error creating Peer object");
  214. return NULL;
  215. }
  216. return peerObject;
  217. }
  218. jobject newNetworkConfig(JNIEnv *env, const ZT_VirtualNetworkConfig &vnetConfig)
  219. {
  220. jstring nameStr = env->NewStringUTF(vnetConfig.name);
  221. if(env->ExceptionCheck() || nameStr == NULL)
  222. {
  223. LOGE("Exception creating new string");
  224. return NULL; // out of memory
  225. }
  226. jobject statusObject = createVirtualNetworkStatus(env, vnetConfig.status);
  227. if(env->ExceptionCheck() || statusObject == NULL)
  228. {
  229. return NULL;
  230. }
  231. jobject typeObject = createVirtualNetworkType(env, vnetConfig.type);
  232. if(env->ExceptionCheck() || typeObject == NULL)
  233. {
  234. return NULL;
  235. }
  236. jobjectArray assignedAddrArrayObj = newInetSocketAddressArray(env, vnetConfig.assignedAddresses, vnetConfig.assignedAddressCount);
  237. if (env->ExceptionCheck() || assignedAddrArrayObj == NULL) {
  238. return NULL;
  239. }
  240. jobjectArray routesArrayObj = newVirtualNetworkRouteArray(env, vnetConfig.routes, vnetConfig.routeCount);
  241. if (env->ExceptionCheck() || routesArrayObj == NULL) {
  242. return NULL;
  243. }
  244. //
  245. // may be NULL
  246. //
  247. jobject dnsObj = newVirtualNetworkDNS(env, vnetConfig.dns);
  248. if(env->ExceptionCheck()) {
  249. return NULL;
  250. }
  251. jobject vnetConfigObj = env->NewObject(
  252. VirtualNetworkConfig_class,
  253. VirtualNetworkConfig_ctor,
  254. vnetConfig.nwid,
  255. vnetConfig.mac,
  256. nameStr,
  257. statusObject,
  258. typeObject,
  259. vnetConfig.mtu,
  260. vnetConfig.dhcp,
  261. vnetConfig.bridge,
  262. vnetConfig.broadcastEnabled,
  263. vnetConfig.portError,
  264. vnetConfig.netconfRevision,
  265. assignedAddrArrayObj,
  266. routesArrayObj,
  267. dnsObj);
  268. if(env->ExceptionCheck() || vnetConfigObj == NULL)
  269. {
  270. LOGE("Error creating new VirtualNetworkConfig object");
  271. return NULL;
  272. }
  273. return vnetConfigObj;
  274. }
  275. jobject newVersion(JNIEnv *env, int major, int minor, int rev)
  276. {
  277. // create a com.zerotier.sdk.Version object
  278. jobject versionObj = env->NewObject(Version_class, Version_ctor, major, minor, rev);
  279. if(env->ExceptionCheck() || versionObj == NULL)
  280. {
  281. LOGE("Error creating new Version object");
  282. return NULL;
  283. }
  284. return versionObj;
  285. }
  286. jobject newVirtualNetworkRoute(JNIEnv *env, const ZT_VirtualNetworkRoute &route)
  287. {
  288. jobject routeObj = env->NewObject(VirtualNetworkRoute_class, VirtualNetworkRoute_ctor);
  289. if(env->ExceptionCheck() || routeObj == NULL)
  290. {
  291. return NULL;
  292. }
  293. jobject targetObj = newInetSocketAddress(env, route.target);
  294. jobject viaObj = newInetSocketAddress(env, route.via);
  295. env->SetObjectField(routeObj, VirtualNetworkRoute_target_field, targetObj);
  296. env->SetObjectField(routeObj, VirtualNetworkRoute_via_field, viaObj);
  297. env->SetIntField(routeObj, VirtualNetworkRoute_flags_field, (jint)route.flags);
  298. env->SetIntField(routeObj, VirtualNetworkRoute_metric_field, (jint)route.metric);
  299. return routeObj;
  300. }
  301. //
  302. // may return NULL
  303. //
  304. jobject newVirtualNetworkDNS(JNIEnv *env, const ZT_VirtualNetworkDNS &dns)
  305. {
  306. if (strlen(dns.domain) == 0) {
  307. LOGD("dns.domain is empty; returning NULL");
  308. return NULL;
  309. }
  310. jstring domain = env->NewStringUTF(dns.domain);
  311. if (env->ExceptionCheck() || domain == NULL) {
  312. LOGE("Exception creating new string");
  313. return NULL;
  314. }
  315. jobject addrList = env->NewObject(ArrayList_class, ArrayList_ctor, 0);
  316. if (env->ExceptionCheck() || addrList == NULL) {
  317. LOGE("Exception creating new ArrayList");
  318. return NULL;
  319. }
  320. for (int i = 0; i < ZT_MAX_DNS_SERVERS; ++i) { //NOLINT
  321. struct sockaddr_storage tmp = dns.server_addr[i];
  322. //
  323. // may be NULL
  324. //
  325. jobject addr = newInetSocketAddress(env, tmp);
  326. if (env->ExceptionCheck()) {
  327. return NULL;
  328. }
  329. if (addr == NULL) {
  330. continue;
  331. }
  332. env->CallBooleanMethod(addrList, ArrayList_add_method, addr);
  333. if(env->ExceptionCheck())
  334. {
  335. LOGE("Exception calling add");
  336. return NULL;
  337. }
  338. env->DeleteLocalRef(addr);
  339. }
  340. jobject dnsObj = env->NewObject(
  341. VirtualNetworkDNS_class,
  342. VirtualNetworkDNS_ctor,
  343. domain,
  344. addrList);
  345. if (env->ExceptionCheck() || dnsObj == NULL) {
  346. LOGE("Exception creating new VirtualNetworkDNS");
  347. return NULL;
  348. }
  349. return dnsObj;
  350. }
  351. jobject newNodeStatus(JNIEnv *env, const ZT_NodeStatus &status) {
  352. jstring pubIdentStr = env->NewStringUTF(status.publicIdentity);
  353. if(env->ExceptionCheck() || pubIdentStr == NULL)
  354. {
  355. LOGE("Exception creating new string");
  356. return NULL;
  357. }
  358. jstring secIdentStr = env->NewStringUTF(status.secretIdentity);
  359. if(env->ExceptionCheck() || secIdentStr == NULL)
  360. {
  361. LOGE("Exception creating new string");
  362. return NULL;
  363. }
  364. jobject nodeStatusObj = env->NewObject(
  365. NodeStatus_class,
  366. NodeStatus_ctor,
  367. status.address,
  368. pubIdentStr,
  369. secIdentStr,
  370. status.online);
  371. if(env->ExceptionCheck() || nodeStatusObj == NULL) {
  372. LOGE("Exception creating new NodeStatus");
  373. return NULL;
  374. }
  375. return nodeStatusObj;
  376. }
  377. jobjectArray newPeerArray(JNIEnv *env, const ZT_Peer *peers, size_t count) {
  378. return newArrayObject<ZT_Peer, newPeer>(env, peers, count, Peer_class);
  379. }
  380. jobjectArray newVirtualNetworkConfigArray(JNIEnv *env, const ZT_VirtualNetworkConfig *networks, size_t count) {
  381. return newArrayObject<ZT_VirtualNetworkConfig, newNetworkConfig>(env, networks, count, VirtualNetworkConfig_class);
  382. }
  383. jobjectArray newPeerPhysicalPathArray(JNIEnv *env, const ZT_PeerPhysicalPath *paths, size_t count) {
  384. return newArrayObject<ZT_PeerPhysicalPath, newPeerPhysicalPath>(env, paths, count, PeerPhysicalPath_class);
  385. }
  386. jobjectArray newInetSocketAddressArray(JNIEnv *env, const sockaddr_storage *addresses, size_t count) {
  387. return newArrayObject<sockaddr_storage, newInetSocketAddress>(env, addresses, count, InetSocketAddress_class);
  388. }
  389. jobjectArray newVirtualNetworkRouteArray(JNIEnv *env, const ZT_VirtualNetworkRoute *routes, size_t count) {
  390. return newArrayObject<ZT_VirtualNetworkRoute, newVirtualNetworkRoute>(env, routes, count, VirtualNetworkRoute_class);
  391. }
  392. void newArrayObject_logCount(size_t count) {
  393. LOGE("count > JSIZE_MAX: %zu", count);
  394. }
  395. void newArrayObject_log(const char *msg) {
  396. LOGE("%s", msg);
  397. }
  398. jbyteArray newByteArray(JNIEnv *env, const unsigned char *bytes, size_t count) {
  399. if (count > JSIZE_MAX) {
  400. LOGE("count > JSIZE_MAX: %zu", count);
  401. return NULL;
  402. }
  403. jsize jCount = static_cast<jsize>(count);
  404. const jbyte *jBytes = reinterpret_cast<const jbyte *>(bytes);
  405. jbyteArray byteArrayObj = env->NewByteArray(jCount);
  406. if(byteArrayObj == NULL)
  407. {
  408. LOGE("NewByteArray returned NULL");
  409. return NULL;
  410. }
  411. env->SetByteArrayRegion(byteArrayObj, 0, jCount, jBytes);
  412. if (env->ExceptionCheck()) {
  413. LOGE("Exception when calling SetByteArrayRegion");
  414. return NULL;
  415. }
  416. return byteArrayObj;
  417. }
  418. jbyteArray newByteArray(JNIEnv *env, size_t count) {
  419. if (count > JSIZE_MAX) {
  420. LOGE("count > JSIZE_MAX: %zu", count);
  421. return NULL;
  422. }
  423. jsize jCount = static_cast<jsize>(count);
  424. jbyteArray byteArrayObj = env->NewByteArray(jCount);
  425. if(byteArrayObj == NULL)
  426. {
  427. LOGE("NewByteArray returned NULL");
  428. return NULL;
  429. }
  430. return byteArrayObj;
  431. }