LinuxNetLink.cpp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. /*
  2. * ZeroTier One - Network Virtualization Everywhere
  3. * Copyright (C) 2011-2018 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. * --
  19. *
  20. * You can be released from the requirements of the license by purchasing
  21. * a commercial license. Buying such a license is mandatory as soon as you
  22. * develop commercial closed-source software that incorporates or links
  23. * directly against ZeroTier software without disclosing the source code
  24. * of your own application.
  25. */
  26. #include "LinuxNetLink.hpp"
  27. #include <unistd.h>
  28. namespace ZeroTier {
  29. LinuxNetLink::LinuxNetLink()
  30. : _t()
  31. , _running(false)
  32. , _routes_ipv4()
  33. , _routes_ipv6()
  34. , _fd(socket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE))
  35. , _la({0})
  36. , _pa({0})
  37. , _msg({0})
  38. , _iov({0})
  39. , _rtn(0)
  40. , _nlp(NULL)
  41. , _nll(0)
  42. , _rtp(NULL)
  43. , _rtl(0)
  44. , _rtap(NULL)
  45. , _ifip(NULL)
  46. , _ifil(0)
  47. , _ifap(NULL)
  48. , _ifal(0)
  49. {
  50. memset(_buf, 0, sizeof(_buf));
  51. // set socket timeout to 1 sec so we're not permablocking recv() calls
  52. struct timeval tv;
  53. tv.tv_sec = 1;
  54. tv.tv_usec = 0;
  55. if(setsockopt(_fd, SOL_SOCKET, SO_RCVTIMEO, (const char*)&tv, sizeof(tv)) != 0) {
  56. fprintf(stderr, "setsockopt failed: %s\n", strerror(errno));
  57. }
  58. _la.nl_family = AF_NETLINK;
  59. _la.nl_pid = getpid();
  60. _la.nl_groups = RTMGRP_LINK|RTMGRP_IPV4_IFADDR|RTMGRP_IPV6_IFADDR|RTMGRP_IPV4_ROUTE|RTMGRP_IPV6_ROUTE;
  61. if (bind(_fd, (struct sockaddr*)&_la, sizeof(_la))) {
  62. fprintf(stderr, "Error connecting to RTNETLINK: %s\n", strerror(errno));
  63. ::exit(1);
  64. }
  65. _running = true;
  66. _t = Thread::start(this);
  67. fprintf(stderr, "Requesting IPV4 Routes\n");
  68. _requestIPv4Routes();
  69. Thread::sleep(10);
  70. fprintf(stderr, "Requesting IPV6 Routes\n");
  71. _requestIPv6Routes();
  72. }
  73. LinuxNetLink::~LinuxNetLink()
  74. {
  75. _running = false;
  76. Thread::join(_t);
  77. ::close(_fd);
  78. }
  79. void LinuxNetLink::threadMain() throw()
  80. {
  81. char *p;
  82. p = _buf;
  83. _nll = 0;
  84. while(_running) {
  85. _rtn = recv(_fd, p, sizeof(_buf) - _nll, 0);
  86. if (_rtn > 0) {
  87. _nlp = (struct nlmsghdr *) p;
  88. if(_nlp->nlmsg_type == NLMSG_ERROR && (_nlp->nlmsg_flags & NLM_F_ACK) != NLM_F_ACK) {
  89. fprintf(stderr, "NLMSG_ERROR\n");
  90. struct nlmsgerr *err = (struct nlmsgerr*)NLMSG_DATA(_nlp);
  91. if (err->error != 0) {
  92. fprintf(stderr, "rtnetlink error: %s\n", strerror(-(err->error)));
  93. }
  94. p = _buf;
  95. _nll = 0;
  96. continue;
  97. }
  98. if (_nlp->nlmsg_type == NLMSG_NOOP) {
  99. fprintf(stderr, "noop\n");
  100. continue;
  101. }
  102. if( (_nlp->nlmsg_flags & NLM_F_MULTI) == NLM_F_MULTI || (_nlp->nlmsg_type == NLMSG_DONE))
  103. {
  104. if (_nlp->nlmsg_type == NLMSG_DONE) {
  105. _processMessage();
  106. p = _buf;
  107. _nll = 0;
  108. continue;
  109. }
  110. p += _rtn;
  111. _nll += _rtn;
  112. }
  113. if (_nlp->nlmsg_type == NLMSG_OVERRUN) {
  114. fprintf(stderr, "NLMSG_OVERRUN: Data lost\n");
  115. p = _buf;
  116. _nll = 0;
  117. continue;
  118. }
  119. _nll += _rtn;
  120. _processMessage();
  121. p = _buf;
  122. _nll = 0;
  123. }
  124. else {
  125. Thread::sleep(100);
  126. continue;
  127. }
  128. }
  129. }
  130. void LinuxNetLink::_processMessage()
  131. {
  132. for(_nlp = (struct nlmsghdr *)_buf; NLMSG_OK(_nlp, _nll); _nlp=NLMSG_NEXT(_nlp, _nll))
  133. {
  134. switch(_nlp->nlmsg_type)
  135. {
  136. case RTM_NEWLINK:
  137. _linkAdded();
  138. break;
  139. case RTM_DELLINK:
  140. _linkDeleted();
  141. break;
  142. case RTM_GETLINK:
  143. fprintf(stderr, "Get Link\n");
  144. break;
  145. case RTM_SETLINK:
  146. fprintf(stderr, "Set Link\n");
  147. break;
  148. case RTM_NEWADDR:
  149. _ipAddressAdded();
  150. break;
  151. case RTM_DELADDR:
  152. _ipAddressDeleted();
  153. break;
  154. case RTM_GETADDR:
  155. fprintf(stderr, "Get IP Address\n");
  156. break;
  157. case RTM_NEWROUTE:
  158. _routeAdded();
  159. break;
  160. case RTM_DELROUTE:
  161. _routeDeleted();
  162. break;
  163. case RTM_GETROUTE:
  164. break;
  165. default:
  166. fprintf(stderr, "ignore msgtype %d...\n", _nlp->nlmsg_type);
  167. }
  168. }
  169. _nlp = NULL;
  170. _nll = 0;
  171. _rtp = NULL;
  172. _rtl = 0;
  173. _ifip = NULL;
  174. _ifil = 0;
  175. _ifap = NULL;
  176. _ifal = 0;
  177. }
  178. void LinuxNetLink::_ipAddressAdded()
  179. {
  180. _ifap = (struct ifaddrmsg *)NLMSG_DATA(_nlp);
  181. _rtap = (struct rtattr *)IFA_RTA(_ifap);
  182. _ifal = IFA_PAYLOAD(_nlp);
  183. char addr[40] = {0};
  184. char local[40] = {0};
  185. char label[40] = {0};
  186. char bcast[40] = {0};
  187. for(;RTA_OK(_rtap, _ifal); _rtap=RTA_NEXT(_rtap,_ifal))
  188. {
  189. switch(_rtap->rta_type) {
  190. case IFA_ADDRESS:
  191. inet_ntop(_ifap->ifa_family, RTA_DATA(_rtap), addr, 40);
  192. break;
  193. case IFA_LOCAL:
  194. inet_ntop(_ifap->ifa_family, RTA_DATA(_rtap), local, 40);
  195. break;
  196. case IFA_LABEL:
  197. memcpy(label, RTA_DATA(_rtap), 40);
  198. break;
  199. case IFA_BROADCAST:
  200. inet_ntop(_ifap->ifa_family, RTA_DATA(_rtap), bcast, 40);
  201. break;
  202. }
  203. }
  204. fprintf(stderr, "Added IP Address %s local: %s label: %s broadcast: %s\n", addr, local, label, bcast);
  205. }
  206. void LinuxNetLink::_ipAddressDeleted()
  207. {
  208. _ifap = (struct ifaddrmsg *)NLMSG_DATA(_nlp);
  209. _rtap = (struct rtattr *)IFA_RTA(_ifap);
  210. _ifal = IFA_PAYLOAD(_nlp);
  211. char addr[40] = {0};
  212. char local[40] = {0};
  213. char label[40] = {0};
  214. char bcast[40] = {0};
  215. for(;RTA_OK(_rtap, _ifal); _rtap=RTA_NEXT(_rtap,_ifal))
  216. {
  217. switch(_rtap->rta_type) {
  218. case IFA_ADDRESS:
  219. inet_ntop(_ifap->ifa_family, RTA_DATA(_rtap), addr, 40);
  220. break;
  221. case IFA_LOCAL:
  222. inet_ntop(_ifap->ifa_family, RTA_DATA(_rtap), local, 40);
  223. break;
  224. case IFA_LABEL:
  225. memcpy(label, RTA_DATA(_rtap), 40);
  226. break;
  227. case IFA_BROADCAST:
  228. inet_ntop(_ifap->ifa_family, RTA_DATA(_rtap), bcast, 40);
  229. break;
  230. }
  231. }
  232. fprintf(stderr, "Removed IP Address %s local: %s label: %s broadcast: %s\n", addr, local, label, bcast);
  233. }
  234. void LinuxNetLink::_routeAdded()
  235. {
  236. char dsts[40] = {0};
  237. char gws[40] = {0};
  238. char ifs[16] = {0};
  239. char ms[24] = {0};
  240. _rtp = (struct rtmsg *) NLMSG_DATA(_nlp);
  241. _rtap = (struct rtattr *)RTM_RTA(_rtp);
  242. _rtl = RTM_PAYLOAD(_nlp);
  243. for(;RTA_OK(_rtap, _rtl); _rtap=RTA_NEXT(_rtap, _rtl))
  244. {
  245. switch(_rtap->rta_type)
  246. {
  247. case RTA_DST:
  248. inet_ntop(_rtp->rtm_family, RTA_DATA(_rtap), dsts, _rtp->rtm_family == AF_INET ? 24 : 40);
  249. break;
  250. case RTA_GATEWAY:
  251. inet_ntop(_rtp->rtm_family, RTA_DATA(_rtap), gws, _rtp->rtm_family == AF_INET ? 24 : 40);
  252. break;
  253. case RTA_OIF:
  254. sprintf(ifs, "%d", *((int*)RTA_DATA(_rtap)));
  255. break;
  256. }
  257. }
  258. sprintf(ms, "%d", _rtp->rtm_dst_len);
  259. fprintf(stderr, "Route Added: dst %s/%s gw %s if %s\n", dsts, ms, gws, ifs);
  260. }
  261. void LinuxNetLink::_routeDeleted()
  262. {
  263. char dsts[40] = {0};
  264. char gws[40] = {0};
  265. char ifs[16] = {0};
  266. char ms[24] = {0};
  267. _rtp = (struct rtmsg *) NLMSG_DATA(_nlp);
  268. _rtap = (struct rtattr *)RTM_RTA(_rtp);
  269. _rtl = RTM_PAYLOAD(_nlp);
  270. for(;RTA_OK(_rtap, _rtl); _rtap=RTA_NEXT(_rtap, _rtl))
  271. {
  272. switch(_rtap->rta_type)
  273. {
  274. case RTA_DST:
  275. inet_ntop(_rtp->rtm_family, RTA_DATA(_rtap), dsts, _rtp->rtm_family == AF_INET ? 24 : 40);
  276. break;
  277. case RTA_GATEWAY:
  278. inet_ntop(_rtp->rtm_family, RTA_DATA(_rtap), gws, _rtp->rtm_family == AF_INET ? 24 : 40);
  279. break;
  280. case RTA_OIF:
  281. sprintf(ifs, "%d", *((int*)RTA_DATA(_rtap)));
  282. break;
  283. }
  284. }
  285. sprintf(ms, "%d", _rtp->rtm_dst_len);
  286. fprintf(stderr, "Route Deleted: dst %s/%s gw %s if %s\n", dsts, ms, gws, ifs);
  287. }
  288. void LinuxNetLink::_linkAdded()
  289. {
  290. char mac[20] = {0};
  291. unsigned int mtu = 0;
  292. char ifname[40] = {0};
  293. _ifip = (struct ifinfomsg *)NLMSG_DATA(_nlp);
  294. _rtap = (struct rtattr *)IFLA_RTA(_ifip);
  295. _ifil = RTM_PAYLOAD(_nlp);
  296. const char *ptr;
  297. unsigned char *ptr2;
  298. for(;RTA_OK(_rtap, _ifil);_rtap=RTA_NEXT(_rtap, _ifil))
  299. {
  300. switch(_rtap->rta_type) {
  301. case IFLA_ADDRESS:
  302. ptr2 = (unsigned char*)RTA_DATA(_rtap);
  303. snprintf(mac, 20, "%02x:%02x:%02x:%02x:%02x:%02x",
  304. ptr2[0], ptr2[1], ptr2[2], ptr2[3], ptr2[4], ptr2[5]);
  305. break;
  306. case IFLA_IFNAME:
  307. ptr = (const char*)RTA_DATA(_rtap);
  308. memcpy(ifname, ptr, strlen(ptr));
  309. break;
  310. case IFLA_MTU:
  311. memcpy(&mtu, RTA_DATA(_rtap), sizeof(unsigned int));
  312. break;
  313. }
  314. }
  315. fprintf(stderr, "Link Added: %s mac: %s, mtu: %d\n", ifname, mac, mtu);
  316. }
  317. void LinuxNetLink::_linkDeleted()
  318. {
  319. char mac[20] = {0};
  320. unsigned int mtu = 0;
  321. char ifname[40] = {0};
  322. _ifip = (struct ifinfomsg *)NLMSG_DATA(_nlp);
  323. _rtap = (struct rtattr *)IFLA_RTA(_ifip);
  324. _ifil = RTM_PAYLOAD(_nlp);
  325. const char *ptr;
  326. unsigned char *ptr2;
  327. for(;RTA_OK(_rtap, _ifil);_rtap=RTA_NEXT(_rtap, _ifil))
  328. {
  329. switch(_rtap->rta_type) {
  330. case IFLA_ADDRESS:
  331. ptr2 = (unsigned char*)RTA_DATA(_rtap);
  332. snprintf(mac, 20, "%02x:%02x:%02x:%02x:%02x:%02x",
  333. ptr2[0], ptr2[1], ptr2[2], ptr2[3], ptr2[4], ptr2[5]);
  334. break;
  335. case IFLA_IFNAME:
  336. ptr = (const char*)RTA_DATA(_rtap);
  337. memcpy(ifname, ptr, strlen(ptr));
  338. break;
  339. case IFLA_MTU:
  340. memcpy(&mtu, RTA_DATA(_rtap), sizeof(unsigned int));
  341. break;
  342. }
  343. }
  344. fprintf(stderr, "Link Deleted: %s mac: %s, mtu: %d\n", ifname, mac, mtu);
  345. }
  346. void LinuxNetLink::_requestIPv4Routes()
  347. {
  348. struct nl_req req;
  349. bzero(&req, sizeof(req));
  350. req.nl.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
  351. req.nl.nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP;
  352. req.nl.nlmsg_type = RTM_GETROUTE;
  353. req.rt.rtm_family = AF_INET;
  354. req.rt.rtm_table = RT_TABLE_MAIN;
  355. bzero(&_pa, sizeof(_pa));
  356. _pa.nl_family = AF_NETLINK;
  357. bzero(&_msg, sizeof(_msg));
  358. _msg.msg_name = (void*)&_pa;
  359. _msg.msg_namelen = sizeof(_pa);
  360. _iov.iov_base = (void*)&req.nl;
  361. _iov.iov_len = req.nl.nlmsg_len;
  362. _msg.msg_iov = &_iov;
  363. _msg.msg_iovlen = 1;
  364. _rtn = sendmsg(_fd, &_msg, 0);
  365. }
  366. void LinuxNetLink::_requestIPv6Routes()
  367. {
  368. struct nl_req req;
  369. bzero(&req, sizeof(req));
  370. req.nl.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
  371. req.nl.nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP;
  372. req.nl.nlmsg_type = RTM_GETROUTE;
  373. req.rt.rtm_family = AF_INET6;
  374. req.rt.rtm_table = RT_TABLE_MAIN;
  375. bzero(&_pa, sizeof(_pa));
  376. _pa.nl_family = AF_NETLINK;
  377. bzero(&_msg, sizeof(_msg));
  378. _msg.msg_name = (void*)&_pa;
  379. _msg.msg_namelen = sizeof(_pa);
  380. _iov.iov_base = (void*)&req.nl;
  381. _iov.iov_len = req.nl.nlmsg_len;
  382. _msg.msg_iov = &_iov;
  383. _msg.msg_iovlen = 1;
  384. while((_rtn = sendmsg(_fd, &_msg, 0)) == -1) {
  385. fprintf(stderr, "ipv6 waiting...");
  386. Thread::sleep(100);
  387. }
  388. }
  389. void LinuxNetLink::addRoute(const InetAddress &target, const InetAddress &via, const char *ifaceName)
  390. {
  391. }
  392. void LinuxNetLink::delRoute(const InetAddress &target, const InetAddress &via, const char *ifaceName)
  393. {
  394. }
  395. void LinuxNetLink::addInterface(const char *iface)
  396. {
  397. }
  398. void LinuxNetLink::addAddress(const InetAddress &addr, const char *iface)
  399. {
  400. }
  401. RouteList LinuxNetLink::getIPV4Routes() const
  402. {
  403. return _routes_ipv4;
  404. }
  405. RouteList LinuxNetLink::getIPV6Routes() const
  406. {
  407. return _routes_ipv6;
  408. }
  409. } // namespace ZeroTier