NCDInterfaceMonitor.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440
  1. /**
  2. * @file NCDInterfaceMonitor.c
  3. * @author Ambroz Bizjak <ambrop7@gmail.com>
  4. *
  5. * @section LICENSE
  6. *
  7. * Redistribution and use in source and binary forms, with or without
  8. * modification, are permitted provided that the following conditions are met:
  9. * 1. Redistributions of source code must retain the above copyright
  10. * notice, this list of conditions and the following disclaimer.
  11. * 2. Redistributions in binary form must reproduce the above copyright
  12. * notice, this list of conditions and the following disclaimer in the
  13. * documentation and/or other materials provided with the distribution.
  14. * 3. Neither the name of the author nor the
  15. * names of its contributors may be used to endorse or promote products
  16. * derived from this software without specific prior written permission.
  17. *
  18. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  19. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  20. * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  21. * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
  22. * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  23. * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  24. * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  25. * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  26. * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  27. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  28. */
  29. #include <string.h>
  30. #include <unistd.h>
  31. #include <stdint.h>
  32. #include <sys/socket.h>
  33. #include <net/if.h>
  34. #include <linux/netlink.h>
  35. #include <linux/rtnetlink.h>
  36. #include <asm/types.h>
  37. #include <asm/types.h>
  38. #include <misc/debug.h>
  39. #include <misc/nonblocking.h>
  40. #include <base/BLog.h>
  41. #include <ncd/NCDInterfaceMonitor.h>
  42. #include <generated/blog_channel_NCDInterfaceMonitor.h>
  43. #define IFA_RTA(r) ((struct rtattr*)(((char*)(r)) + NLMSG_ALIGN(sizeof(struct ifaddrmsg))))
  44. static int send_wilddump_request (NCDInterfaceMonitor *o, int fd, uint32_t seq, int family, int type);
  45. static int get_attr (int type, struct rtattr *rta, int rta_len, void **out_attr, int *out_attr_len);
  46. static void report_error (NCDInterfaceMonitor *o);
  47. static int send_next_dump_request (NCDInterfaceMonitor *o);
  48. static void netlink_fd_handler (NCDInterfaceMonitor *o, int events);
  49. static void process_buffer (NCDInterfaceMonitor *o);
  50. static void more_job_handler (NCDInterfaceMonitor *o);
  51. static int send_wilddump_request (NCDInterfaceMonitor *o, int fd, uint32_t seq, int family, int type)
  52. {
  53. struct {
  54. struct nlmsghdr nlh;
  55. struct rtgenmsg g;
  56. } req;
  57. memset(&req, 0, sizeof(req));
  58. req.nlh.nlmsg_len = sizeof(req);
  59. req.nlh.nlmsg_type = type;
  60. req.nlh.nlmsg_flags = NLM_F_ROOT|NLM_F_MATCH|NLM_F_REQUEST;
  61. req.nlh.nlmsg_pid = 0;
  62. req.nlh.nlmsg_seq = seq;
  63. req.g.rtgen_family = family;
  64. int res = write(fd, &req, sizeof(req));
  65. if (res < 0) {
  66. BLog(BLOG_ERROR, "write failed");
  67. return 0;
  68. }
  69. if (res != sizeof(req)) {
  70. BLog(BLOG_ERROR, "write short");
  71. return 0;
  72. }
  73. return 1;
  74. }
  75. static int get_attr (int type, struct rtattr *rta, int rta_len, void **out_attr, int *out_attr_len)
  76. {
  77. for (; RTA_OK(rta, rta_len); rta = RTA_NEXT(rta, rta_len)) {
  78. uint8_t *attr = RTA_DATA(rta);
  79. int attr_len = RTA_PAYLOAD(rta);
  80. if (rta->rta_type == type) {
  81. *out_attr = attr;
  82. *out_attr_len = attr_len;
  83. return 1;
  84. }
  85. }
  86. return 0;
  87. }
  88. static void report_error (NCDInterfaceMonitor *o)
  89. {
  90. DEBUGERROR(&o->d_err, o->handler_error(o->user))
  91. }
  92. static int send_next_dump_request (NCDInterfaceMonitor *o)
  93. {
  94. ASSERT(o->dump_queue)
  95. if (o->dump_queue & NCDIFMONITOR_WATCH_LINK) {
  96. o->dump_queue &= ~NCDIFMONITOR_WATCH_LINK;
  97. return send_wilddump_request(o, o->netlink_fd, o->dump_seq, 0, RTM_GETLINK);
  98. }
  99. else if (o->dump_queue & NCDIFMONITOR_WATCH_IPV4_ADDR) {
  100. o->dump_queue &= ~NCDIFMONITOR_WATCH_IPV4_ADDR;
  101. return send_wilddump_request(o, o->netlink_fd, o->dump_seq, AF_INET, RTM_GETADDR);
  102. }
  103. else if (o->dump_queue & NCDIFMONITOR_WATCH_IPV6_ADDR) {
  104. o->dump_queue &= ~NCDIFMONITOR_WATCH_IPV6_ADDR;
  105. return send_wilddump_request(o, o->netlink_fd, o->dump_seq, AF_INET6, RTM_GETADDR);
  106. }
  107. ASSERT(0)
  108. return 0;
  109. }
  110. void netlink_fd_handler (NCDInterfaceMonitor *o, int events)
  111. {
  112. DebugObject_Access(&o->d_obj);
  113. ASSERT(o->buf_left == -1)
  114. ASSERT(o->have_bfd)
  115. // read from netlink fd
  116. int len = read(o->netlink_fd, o->buf.buf, sizeof(o->buf));
  117. if (len < 0) {
  118. BLog(BLOG_ERROR, "read failed");
  119. goto fail;
  120. }
  121. // stop receiving fd events
  122. BReactor_SetFileDescriptorEvents(o->reactor, &o->bfd, 0);
  123. // set buffer
  124. o->buf_nh = &o->buf.nlh;
  125. o->buf_left = len;
  126. // process buffer
  127. process_buffer(o);
  128. return;
  129. fail:
  130. report_error(o);
  131. }
  132. void process_buffer (NCDInterfaceMonitor *o)
  133. {
  134. ASSERT(o->buf_left >= 0)
  135. ASSERT(o->have_bfd)
  136. int done = 0;
  137. for (; NLMSG_OK(o->buf_nh, o->buf_left); o->buf_nh = NLMSG_NEXT(o->buf_nh, o->buf_left)) {
  138. if (o->buf_nh->nlmsg_type == NLMSG_DONE) {
  139. done = 1;
  140. break;
  141. }
  142. struct nlmsghdr *buf = o->buf_nh;
  143. void *pl = NLMSG_DATA(buf);
  144. int pl_len = NLMSG_PAYLOAD(buf, 0);
  145. struct NCDInterfaceMonitor_event ev;
  146. switch (buf->nlmsg_type) {
  147. case RTM_NEWLINK: { // not RTM_DELLINK! who knows what these mean...
  148. if (pl_len < sizeof(struct ifinfomsg)) {
  149. BLog(BLOG_ERROR, "ifinfomsg too short");
  150. goto fail;
  151. }
  152. struct ifinfomsg *msg = pl;
  153. if (msg->ifi_index == o->ifindex && (o->watch_events & NCDIFMONITOR_WATCH_LINK)) {
  154. ev.event = (buf->nlmsg_type == RTM_NEWLINK && (msg->ifi_flags & IFF_RUNNING)) ? NCDIFMONITOR_EVENT_LINK_UP : NCDIFMONITOR_EVENT_LINK_DOWN;
  155. goto dispatch;
  156. }
  157. } break;
  158. case RTM_NEWADDR:
  159. case RTM_DELADDR: {
  160. if (pl_len < sizeof(struct ifaddrmsg)) {
  161. BLog(BLOG_ERROR, "ifaddrmsg too short");
  162. goto fail;
  163. }
  164. struct ifaddrmsg *msg = pl;
  165. void *addr;
  166. int addr_len;
  167. if (!get_attr(IFA_ADDRESS, IFA_RTA(msg), buf->nlmsg_len - NLMSG_LENGTH(sizeof(*msg)), &addr, &addr_len)) {
  168. break;
  169. }
  170. if (msg->ifa_index == o->ifindex && msg->ifa_family == AF_INET && (o->watch_events & NCDIFMONITOR_WATCH_IPV4_ADDR)) {
  171. if (addr_len != 4 || msg->ifa_prefixlen > 32) {
  172. BLog(BLOG_ERROR, "bad ipv4 ifaddrmsg");
  173. goto fail;
  174. }
  175. ev.event = (buf->nlmsg_type == RTM_NEWADDR) ? NCDIFMONITOR_EVENT_IPV4_ADDR_ADDED : NCDIFMONITOR_EVENT_IPV4_ADDR_REMOVED;
  176. ev.u.ipv4_addr.addr.addr = ((struct in_addr *)addr)->s_addr;
  177. ev.u.ipv4_addr.addr.prefix = msg->ifa_prefixlen;
  178. goto dispatch;
  179. }
  180. if (msg->ifa_index == o->ifindex && msg->ifa_family == AF_INET6 && (o->watch_events & NCDIFMONITOR_WATCH_IPV6_ADDR)) {
  181. if (addr_len != 16 || msg->ifa_prefixlen > 128) {
  182. BLog(BLOG_ERROR, "bad ipv6 ifaddrmsg");
  183. goto fail;
  184. }
  185. ev.event = (buf->nlmsg_type == RTM_NEWADDR) ? NCDIFMONITOR_EVENT_IPV6_ADDR_ADDED : NCDIFMONITOR_EVENT_IPV6_ADDR_REMOVED;
  186. memcpy(ev.u.ipv6_addr.addr.addr, ((struct in6_addr *)addr)->s6_addr, 16);
  187. ev.u.ipv6_addr.addr.prefix = msg->ifa_prefixlen;
  188. ev.u.ipv6_addr.addr.scope = msg->ifa_scope;
  189. ev.u.ipv6_addr.addr_flags = 0;
  190. if (!(msg->ifa_flags & IFA_F_PERMANENT)) {
  191. ev.u.ipv6_addr.addr_flags |= NCDIFMONITOR_ADDR_FLAG_DYNAMIC;
  192. }
  193. goto dispatch;
  194. }
  195. } break;
  196. }
  197. continue;
  198. dispatch:
  199. // move to next message
  200. o->buf_nh = NLMSG_NEXT(o->buf_nh, o->buf_left);
  201. // schedule more job
  202. BPending_Set(&o->more_job);
  203. // dispatch event
  204. o->handler(o->user, ev);
  205. return;
  206. }
  207. if (done) {
  208. if (o->dump_queue) {
  209. // increment dump request sequence number
  210. o->dump_seq++;
  211. // send next dump request
  212. if (!send_next_dump_request(o)) {
  213. goto fail;
  214. }
  215. }
  216. else if (o->event_netlink_fd >= 0) {
  217. // stop watching dump fd
  218. BReactor_RemoveFileDescriptor(o->reactor, &o->bfd);
  219. o->have_bfd = 0;
  220. // close dump fd, make event fd current
  221. close(o->netlink_fd);
  222. o->netlink_fd = o->event_netlink_fd;
  223. o->event_netlink_fd = -1;
  224. // start watching event fd
  225. BFileDescriptor_Init(&o->bfd, o->netlink_fd, (BFileDescriptor_handler)netlink_fd_handler, o);
  226. if (!BReactor_AddFileDescriptor(o->reactor, &o->bfd)) {
  227. BLog(BLOG_ERROR, "BReactor_AddFileDescriptor failed");
  228. goto fail;
  229. }
  230. o->have_bfd = 1;
  231. }
  232. }
  233. // set no buffer
  234. o->buf_left = -1;
  235. // continue receiving fd events
  236. BReactor_SetFileDescriptorEvents(o->reactor, &o->bfd, BREACTOR_READ);
  237. return;
  238. fail:
  239. report_error(o);
  240. }
  241. void more_job_handler (NCDInterfaceMonitor *o)
  242. {
  243. DebugObject_Access(&o->d_obj);
  244. ASSERT(o->buf_left >= 0)
  245. // process buffer
  246. process_buffer(o);
  247. return;
  248. }
  249. int NCDInterfaceMonitor_Init (NCDInterfaceMonitor *o, int ifindex, int watch_events, BReactor *reactor, void *user,
  250. NCDInterfaceMonitor_handler handler,
  251. NCDInterfaceMonitor_handler_error handler_error)
  252. {
  253. ASSERT(watch_events)
  254. ASSERT((watch_events&~(NCDIFMONITOR_WATCH_LINK|NCDIFMONITOR_WATCH_IPV4_ADDR|NCDIFMONITOR_WATCH_IPV6_ADDR)) == 0)
  255. ASSERT(handler)
  256. ASSERT(handler_error)
  257. BNetwork_Assert();
  258. // init arguments
  259. o->ifindex = ifindex;
  260. o->watch_events = watch_events;
  261. o->reactor = reactor;
  262. o->user = user;
  263. o->handler = handler;
  264. o->handler_error = handler_error;
  265. // init dump netlink fd
  266. if ((o->netlink_fd = socket(AF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE)) < 0) {
  267. BLog(BLOG_ERROR, "socket failed");
  268. goto fail0;
  269. }
  270. if (!badvpn_set_nonblocking(o->netlink_fd)) {
  271. BLog(BLOG_ERROR, "badvpn_set_nonblocking failed");
  272. goto fail1;
  273. }
  274. // init event netlink fd
  275. if ((o->event_netlink_fd = socket(AF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE)) < 0) {
  276. BLog(BLOG_ERROR, "socket failed");
  277. goto fail1;
  278. }
  279. if (!badvpn_set_nonblocking(o->event_netlink_fd)) {
  280. BLog(BLOG_ERROR, "badvpn_set_nonblocking failed");
  281. goto fail2;
  282. }
  283. // build bind address
  284. struct sockaddr_nl sa;
  285. memset(&sa, 0, sizeof(sa));
  286. sa.nl_family = AF_NETLINK;
  287. sa.nl_groups = 0;
  288. if (watch_events & NCDIFMONITOR_WATCH_LINK) sa.nl_groups |= RTMGRP_LINK;
  289. if (watch_events & NCDIFMONITOR_WATCH_IPV4_ADDR) sa.nl_groups |= RTMGRP_IPV4_IFADDR;
  290. if (watch_events & NCDIFMONITOR_WATCH_IPV6_ADDR) sa.nl_groups |= RTMGRP_IPV6_IFADDR;
  291. // bind event netlink fd
  292. if (bind(o->event_netlink_fd, (void *)&sa, sizeof(sa)) < 0) {
  293. BLog(BLOG_ERROR, "bind failed");
  294. goto fail2;
  295. }
  296. // set dump state
  297. o->dump_queue = watch_events;
  298. o->dump_seq = 0;
  299. // init BFileDescriptor
  300. BFileDescriptor_Init(&o->bfd, o->netlink_fd, (BFileDescriptor_handler)netlink_fd_handler, o);
  301. if (!BReactor_AddFileDescriptor(reactor, &o->bfd)) {
  302. BLog(BLOG_ERROR, "BReactor_AddFileDescriptor failed");
  303. goto fail2;
  304. }
  305. o->have_bfd = 1;
  306. // set nothing in buffer
  307. o->buf_left = -1;
  308. // init more job
  309. BPending_Init(&o->more_job, BReactor_PendingGroup(reactor), (BPending_handler)more_job_handler, o);
  310. // send first dump request
  311. if (!send_next_dump_request(o)) {
  312. goto fail3;
  313. }
  314. // wait for reading fd
  315. BReactor_SetFileDescriptorEvents(reactor, &o->bfd, BREACTOR_READ);
  316. DebugError_Init(&o->d_err, BReactor_PendingGroup(reactor));
  317. DebugObject_Init(&o->d_obj);
  318. return 1;
  319. fail3:
  320. BPending_Free(&o->more_job);
  321. BReactor_RemoveFileDescriptor(o->reactor, &o->bfd);
  322. fail2:
  323. close(o->event_netlink_fd);
  324. fail1:
  325. close(o->netlink_fd);
  326. fail0:
  327. return 0;
  328. }
  329. void NCDInterfaceMonitor_Free (NCDInterfaceMonitor *o)
  330. {
  331. DebugObject_Free(&o->d_obj);
  332. DebugError_Free(&o->d_err);
  333. // free more job
  334. BPending_Free(&o->more_job);
  335. // free BFileDescriptor
  336. if (o->have_bfd) {
  337. BReactor_RemoveFileDescriptor(o->reactor, &o->bfd);
  338. }
  339. // close event fd, in case we're still dumping
  340. if (o->event_netlink_fd >= 0) {
  341. close(o->event_netlink_fd);
  342. }
  343. // close fd
  344. close(o->netlink_fd);
  345. }
  346. void NCDInterfaceMonitor_Pause (NCDInterfaceMonitor *o)
  347. {
  348. DebugObject_Access(&o->d_obj);
  349. DebugError_AssertNoError(&o->d_err);
  350. ASSERT(o->have_bfd)
  351. if (o->buf_left >= 0) {
  352. BPending_Unset(&o->more_job);
  353. } else {
  354. BReactor_SetFileDescriptorEvents(o->reactor, &o->bfd, 0);
  355. }
  356. }
  357. void NCDInterfaceMonitor_Continue (NCDInterfaceMonitor *o)
  358. {
  359. DebugObject_Access(&o->d_obj);
  360. DebugError_AssertNoError(&o->d_err);
  361. ASSERT(o->have_bfd)
  362. if (o->buf_left >= 0) {
  363. BPending_Set(&o->more_job);
  364. } else {
  365. BReactor_SetFileDescriptorEvents(o->reactor, &o->bfd, BREACTOR_READ);
  366. }
  367. }