1 #include <stdio.h> 2 #include <stdlib.h> 3 #include <string.h> 4 #include <err.h> 5 #include <errno.h> 6 7 #include <sys/bitcount.h> 8 #include <sys/param.h> 9 #include <sys/linker.h> 10 #include <sys/module.h> 11 #include <sys/socket.h> 12 #include <sys/sysctl.h> 13 #include <sys/time.h> 14 #include <sys/types.h> 15 16 #include <netinet/in.h> 17 #include <arpa/inet.h> 18 19 #include <net/ethernet.h> 20 #include <net/if.h> 21 #include <net/if_dl.h> 22 #include <net/if_types.h> 23 #include <netlink/netlink.h> 24 #include <netlink/netlink_route.h> 25 #include <netlink/netlink_snl.h> 26 #include <netlink/netlink_snl_route.h> 27 #include <netlink/netlink_snl_route_compat.h> 28 #include <netlink/netlink_snl_route_parsers.h> 29 30 const char *routename(struct sockaddr *); 31 const char *netname(struct sockaddr *); 32 void printb(int, const char *); 33 extern const char routeflags[]; 34 extern int verbose, debugonly; 35 36 int rtmsg_nl(int cmd, int rtm_flags, int fib, struct sockaddr_storage *so, 37 struct rt_metrics *rt_metrics); 38 int flushroutes_fib_nl(int fib, int af); 39 void monitor_nl(int fib); 40 41 struct nl_helper; 42 static void print_getmsg(struct nl_helper *h, struct nlmsghdr *hdr, struct sockaddr *dst); 43 static void print_nlmsg(struct nl_helper *h, struct nlmsghdr *hdr); 44 45 #define s6_addr32 __u6_addr.__u6_addr32 46 #define bitcount32(x) __bitcount32((uint32_t)(x)) 47 static int 48 inet6_get_plen(const struct in6_addr *addr) 49 { 50 51 return (bitcount32(addr->s6_addr32[0]) + bitcount32(addr->s6_addr32[1]) + 52 bitcount32(addr->s6_addr32[2]) + bitcount32(addr->s6_addr32[3])); 53 } 54 55 static void 56 ip6_writemask(struct in6_addr *addr6, uint8_t mask) 57 { 58 uint32_t *cp; 59 60 for (cp = (uint32_t *)addr6; mask >= 32; mask -= 32) 61 *cp++ = 0xFFFFFFFF; 62 if (mask > 0) 63 *cp = htonl(mask ? ~((1 << (32 - mask)) - 1) : 0); 64 } 65 66 static struct sockaddr * 67 get_netmask(struct snl_state *ss, int family, int plen) 68 { 69 if (family == AF_INET) { 70 if (plen == 32) 71 return (NULL); 72 73 struct sockaddr_in *sin = snl_allocz(ss, sizeof(*sin)); 74 75 sin->sin_len = sizeof(*sin); 76 sin->sin_family = family; 77 sin->sin_addr.s_addr = htonl(plen ? ~((1 << (32 - plen)) - 1) : 0); 78 79 return (struct sockaddr *)sin; 80 } else if (family == AF_INET6) { 81 if (plen == 128) 82 return (NULL); 83 84 struct sockaddr_in6 *sin6 = snl_allocz(ss, sizeof(*sin6)); 85 86 sin6->sin6_len = sizeof(*sin6); 87 sin6->sin6_family = family; 88 ip6_writemask(&sin6->sin6_addr, plen); 89 90 return (struct sockaddr *)sin6; 91 } 92 return (NULL); 93 } 94 95 static void 96 nl_init_socket(struct snl_state *ss) 97 { 98 if (snl_init(ss, NETLINK_ROUTE)) 99 return; 100 101 if (modfind("netlink") == -1 && errno == ENOENT) { 102 /* Try to load */ 103 if (kldload("netlink") == -1) 104 err(1, "netlink is not loaded and load attempt failed"); 105 if (snl_init(ss, NETLINK_ROUTE)) 106 return; 107 } 108 109 err(1, "unable to open netlink socket"); 110 } 111 112 struct nl_helper { 113 struct snl_state ss_cmd; 114 }; 115 116 static void 117 nl_helper_init(struct nl_helper *h) 118 { 119 nl_init_socket(&h->ss_cmd); 120 } 121 122 static void 123 nl_helper_free(struct nl_helper *h) 124 { 125 snl_free(&h->ss_cmd); 126 } 127 128 static int 129 rtmsg_nl_int(struct nl_helper *h, int cmd, int rtm_flags, int fib, 130 struct sockaddr_storage *so, struct rt_metrics *rt_metrics) 131 { 132 struct snl_state *ss = &h->ss_cmd; 133 struct snl_writer nw; 134 int nl_type = 0, nl_flags = 0; 135 136 snl_init_writer(ss, &nw); 137 138 switch (cmd) { 139 case RTSOCK_RTM_ADD: 140 nl_type = RTM_NEWROUTE; 141 nl_flags = NLM_F_CREATE | NLM_F_APPEND; /* Do append by default */ 142 break; 143 case RTSOCK_RTM_CHANGE: 144 nl_type = RTM_NEWROUTE; 145 nl_flags = NLM_F_REPLACE; 146 break; 147 case RTSOCK_RTM_DELETE: 148 nl_type = RTM_DELROUTE; 149 break; 150 case RTSOCK_RTM_GET: 151 nl_type = RTM_GETROUTE; 152 break; 153 default: 154 exit(1); 155 } 156 157 struct sockaddr *dst = (struct sockaddr *)&so[RTAX_DST]; 158 struct sockaddr *mask = (struct sockaddr *)&so[RTAX_NETMASK]; 159 struct sockaddr *gw = (struct sockaddr *)&so[RTAX_GATEWAY]; 160 161 if (dst == NULL) 162 return (EINVAL); 163 164 struct nlmsghdr *hdr = snl_create_msg_request(&nw, nl_type); 165 hdr->nlmsg_flags |= nl_flags; 166 167 int plen = 0; 168 int rtm_type = RTN_UNICAST; 169 170 switch (dst->sa_family) { 171 case AF_INET: 172 { 173 struct sockaddr_in *mask4 = (struct sockaddr_in *)mask; 174 175 if ((rtm_flags & RTF_HOST) == 0 && mask4 != NULL) 176 plen = bitcount32(mask4->sin_addr.s_addr); 177 else 178 plen = 32; 179 break; 180 } 181 case AF_INET6: 182 { 183 struct sockaddr_in6 *mask6 = (struct sockaddr_in6 *)mask; 184 185 if ((rtm_flags & RTF_HOST) == 0 && mask6 != NULL) 186 plen = inet6_get_plen(&mask6->sin6_addr); 187 else 188 plen = 128; 189 break; 190 } 191 default: 192 return (ENOTSUP); 193 } 194 195 if (rtm_flags & RTF_REJECT) 196 rtm_type = RTN_PROHIBIT; 197 else if (rtm_flags & RTF_BLACKHOLE) 198 rtm_type = RTN_BLACKHOLE; 199 200 struct rtmsg *rtm = snl_reserve_msg_object(&nw, struct rtmsg); 201 rtm->rtm_family = dst->sa_family; 202 rtm->rtm_protocol = RTPROT_STATIC; 203 rtm->rtm_type = rtm_type; 204 rtm->rtm_dst_len = plen; 205 206 snl_add_msg_attr_ip(&nw, RTA_DST, dst); 207 snl_add_msg_attr_u32(&nw, RTA_TABLE, fib); 208 209 if (rtm_flags & RTF_GATEWAY) { 210 if (gw->sa_family == dst->sa_family) 211 snl_add_msg_attr_ip(&nw, RTA_GATEWAY, gw); 212 else 213 snl_add_msg_attr_ipvia(&nw, RTA_VIA, gw); 214 } else if (gw != NULL) { 215 /* Should be AF_LINK */ 216 struct sockaddr_dl *sdl = (struct sockaddr_dl *)gw; 217 if (sdl->sdl_index != 0) 218 snl_add_msg_attr_u32(&nw, RTA_OIF, sdl->sdl_index); 219 } 220 221 if (rtm_flags != 0) 222 snl_add_msg_attr_u32(&nw, NL_RTA_RTFLAGS, rtm_flags); 223 224 if (rt_metrics->rmx_mtu > 0) { 225 int off = snl_add_msg_attr_nested(&nw, RTA_METRICS); 226 snl_add_msg_attr_u32(&nw, RTAX_MTU, rt_metrics->rmx_mtu); 227 snl_end_attr_nested(&nw, off); 228 } 229 230 if (rt_metrics->rmx_weight > 0) 231 snl_add_msg_attr_u32(&nw, NL_RTA_WEIGHT, rt_metrics->rmx_weight); 232 233 if (snl_finalize_msg(&nw) && snl_send_message(ss, hdr)) { 234 struct snl_errmsg_data e = {}; 235 236 hdr = snl_read_reply(ss, hdr->nlmsg_seq); 237 if (nl_type == NL_RTM_GETROUTE) { 238 if (hdr->nlmsg_type == NL_RTM_NEWROUTE) 239 print_getmsg(h, hdr, dst); 240 else { 241 snl_parse_errmsg(ss, hdr, &e); 242 if (e.error == ESRCH) 243 warn("route has not been found"); 244 else 245 warn("message indicates error %d", e.error); 246 } 247 248 return (0); 249 } 250 251 if (snl_parse_errmsg(ss, hdr, &e)) 252 return (e.error); 253 } 254 return (EINVAL); 255 } 256 257 int 258 rtmsg_nl(int cmd, int rtm_flags, int fib, struct sockaddr_storage *so, 259 struct rt_metrics *rt_metrics) 260 { 261 struct nl_helper h = {}; 262 263 nl_helper_init(&h); 264 int error = rtmsg_nl_int(&h, cmd, rtm_flags, fib, so, rt_metrics); 265 nl_helper_free(&h); 266 267 return (error); 268 } 269 270 static void 271 get_ifdata(struct nl_helper *h, uint32_t ifindex, struct snl_parsed_link_simple *link) 272 { 273 struct snl_state *ss = &h->ss_cmd; 274 struct snl_writer nw; 275 276 snl_init_writer(ss, &nw); 277 struct nlmsghdr *hdr = snl_create_msg_request(&nw, NL_RTM_GETLINK); 278 struct ifinfomsg *ifmsg = snl_reserve_msg_object(&nw, struct ifinfomsg); 279 if (ifmsg != NULL) 280 ifmsg->ifi_index = ifindex; 281 if (!snl_finalize_msg(&nw) || !snl_send_message(ss, hdr)) 282 return; 283 284 hdr = snl_read_reply(ss, hdr->nlmsg_seq); 285 286 if (hdr != NULL && hdr->nlmsg_type == RTM_NEWLINK) { 287 snl_parse_nlmsg(ss, hdr, &snl_rtm_link_parser_simple, link); 288 } 289 290 if (link->ifla_ifname == NULL) { 291 char ifname[16]; 292 293 snprintf(ifname, sizeof(ifname), "if#%u", ifindex); 294 int len = strlen(ifname); 295 char *buf = snl_allocz(ss, len + 1); 296 strlcpy(buf, ifname, len + 1); 297 link->ifla_ifname = buf; 298 } 299 } 300 301 static void 302 print_getmsg(struct nl_helper *h, struct nlmsghdr *hdr, struct sockaddr *dst) 303 { 304 struct snl_state *ss = &h->ss_cmd; 305 struct timespec ts; 306 struct snl_parsed_route r = { .rtax_weight = RT_DEFAULT_WEIGHT }; 307 308 if (!snl_parse_nlmsg(ss, hdr, &snl_rtm_route_parser, &r)) 309 return; 310 311 struct snl_parsed_link_simple link = {}; 312 get_ifdata(h, r.rta_oif, &link); 313 314 if (r.rtax_mtu == 0) 315 r.rtax_mtu = link.ifla_mtu; 316 r.rta_rtflags |= (RTF_UP | RTF_DONE); 317 318 (void)printf(" route to: %s\n", routename(dst)); 319 320 if (r.rta_dst) 321 (void)printf("destination: %s\n", routename(r.rta_dst)); 322 struct sockaddr *mask = get_netmask(ss, r.rtm_family, r.rtm_dst_len); 323 if (mask) 324 (void)printf(" mask: %s\n", routename(mask)); 325 if (r.rta_gw && (r.rta_rtflags & RTF_GATEWAY)) 326 (void)printf(" gateway: %s\n", routename(r.rta_gw)); 327 (void)printf(" fib: %u\n", (unsigned int)r.rta_table); 328 if (link.ifla_ifname) 329 (void)printf(" interface: %s\n", link.ifla_ifname); 330 (void)printf(" flags: "); 331 printb(r.rta_rtflags, routeflags); 332 333 struct rt_metrics rmx = { 334 .rmx_mtu = r.rtax_mtu, 335 .rmx_weight = r.rtax_weight, 336 .rmx_expire = r.rta_expire, 337 }; 338 339 printf("\n%9s %9s %9s %9s %9s %10s %9s\n", "recvpipe", 340 "sendpipe", "ssthresh", "rtt,msec", "mtu ", "weight", "expire"); 341 printf("%8lu ", rmx.rmx_recvpipe); 342 printf("%8lu ", rmx.rmx_sendpipe); 343 printf("%8lu ", rmx.rmx_ssthresh); 344 printf("%8lu ", 0UL); 345 printf("%8lu ", rmx.rmx_mtu); 346 printf("%8lu ", rmx.rmx_weight); 347 if (rmx.rmx_expire > 0) 348 clock_gettime(CLOCK_REALTIME_FAST, &ts); 349 else 350 ts.tv_sec = 0; 351 printf("%8ld \n", (long)(rmx.rmx_expire - ts.tv_sec)); 352 } 353 354 static void 355 print_prefix(struct nl_helper *h, char *buf, int bufsize, struct sockaddr *sa, int plen) 356 { 357 int sz = 0; 358 359 if (sa == NULL) { 360 snprintf(buf, bufsize, "<NULL>"); 361 return; 362 } 363 364 switch (sa->sa_family) { 365 case AF_INET: 366 { 367 struct sockaddr_in *sin = (struct sockaddr_in *)sa; 368 char abuf[INET_ADDRSTRLEN]; 369 370 inet_ntop(AF_INET, &sin->sin_addr, abuf, sizeof(abuf)); 371 sz = snprintf(buf, bufsize, "%s", abuf); 372 break; 373 } 374 case AF_INET6: 375 { 376 struct sockaddr_in6 *sin6 = (struct sockaddr_in6 *)sa; 377 char abuf[INET6_ADDRSTRLEN]; 378 char *ifname = NULL; 379 380 inet_ntop(AF_INET6, &sin6->sin6_addr, abuf, sizeof(abuf)); 381 if (IN6_IS_ADDR_LINKLOCAL(&sin6->sin6_addr)) { 382 struct snl_parsed_link_simple link = {}; 383 384 if (sin6->sin6_scope_id != 0) { 385 get_ifdata(h, sin6->sin6_scope_id, &link); 386 ifname = link.ifla_ifname; 387 } 388 } 389 if (ifname == NULL) 390 sz = snprintf(buf, bufsize, "%s", abuf); 391 else 392 sz = snprintf(buf, bufsize, "%s%%%s", abuf, ifname); 393 break; 394 } 395 default: 396 snprintf(buf, bufsize, "unknown_af#%d", sa->sa_family); 397 plen = -1; 398 } 399 400 if (plen >= 0) 401 snprintf(buf + sz, bufsize - sz, "/%d", plen); 402 } 403 404 405 static int 406 print_line_prefix(const char *cmd, const char *name) 407 { 408 struct timespec tp; 409 struct tm tm; 410 char buf[32]; 411 412 clock_gettime(CLOCK_REALTIME, &tp); 413 localtime_r(&tp.tv_sec, &tm); 414 415 strftime(buf, sizeof(buf), "%T", &tm); 416 int len = printf("%s.%03ld %s %s ", buf, tp.tv_nsec / 1000000, cmd, name); 417 418 return (len); 419 } 420 421 static const char * 422 get_action_name(struct nlmsghdr *hdr, int new_cmd) 423 { 424 if (hdr->nlmsg_type == new_cmd) { 425 //return ((hdr->nlmsg_flags & NLM_F_REPLACE) ? "replace" : "add"); 426 return ("add/repl"); 427 } else 428 return ("delete"); 429 } 430 431 static void 432 print_nlmsg_route_nhop(struct nl_helper *h, struct snl_parsed_route *r, 433 struct rta_mpath_nh *nh, bool first) 434 { 435 // gw 10.0.0.1 ifp vtnet0 mtu 1500 table inet.0 436 if (nh->gw != NULL) { 437 char gwbuf[128]; 438 print_prefix(h, gwbuf, sizeof(gwbuf), nh->gw, -1); 439 printf("gw %s ", gwbuf); 440 } 441 442 if (nh->ifindex != 0) { 443 struct snl_parsed_link_simple link = {}; 444 445 get_ifdata(h, nh->ifindex, &link); 446 if (nh->rtax_mtu == 0) 447 nh->rtax_mtu = link.ifla_mtu; 448 printf("iface %s ", link.ifla_ifname); 449 if (nh->rtax_mtu != 0) 450 printf("mtu %d ", nh->rtax_mtu); 451 } 452 453 if (first) { 454 switch (r->rtm_family) { 455 case AF_INET: 456 printf("table inet.%d", r->rta_table); 457 break; 458 case AF_INET6: 459 printf("table inet6.%d", r->rta_table); 460 break; 461 } 462 } 463 464 printf("\n"); 465 } 466 467 static void 468 print_nlmsg_route(struct nl_helper *h, struct nlmsghdr *hdr) 469 { 470 struct snl_parsed_route r = { .rtax_weight = RT_DEFAULT_WEIGHT }; 471 struct snl_state *ss = &h->ss_cmd; 472 473 if (!snl_parse_nlmsg(ss, hdr, &snl_rtm_route_parser, &r)) 474 return; 475 476 // 20:19:41.333 add route 10.0.0.0/24 gw 10.0.0.1 ifp vtnet0 mtu 1500 table inet.0 477 478 const char *cmd = get_action_name(hdr, RTM_NEWROUTE); 479 int len = print_line_prefix(cmd, "route"); 480 481 char buf[128]; 482 print_prefix(h, buf, sizeof(buf), r.rta_dst, r.rtm_dst_len); 483 len += strlen(buf) + 1; 484 printf("%s ", buf); 485 486 switch (r.rtm_type) { 487 case RTN_BLACKHOLE: 488 printf("blackhole\n"); 489 return; 490 case RTN_UNREACHABLE: 491 printf("unreach(reject)\n"); 492 return; 493 case RTN_PROHIBIT: 494 printf("prohibit(reject)\n"); 495 return; 496 } 497 498 if (r.rta_multipath != NULL) { 499 bool first = true; 500 501 memset(buf, ' ', sizeof(buf)); 502 buf[len] = '\0'; 503 504 for (int i = 0; i < r.rta_multipath->num_nhops; i++) { 505 struct rta_mpath_nh *nh = &r.rta_multipath->nhops[i]; 506 507 if (!first) 508 printf("%s", buf); 509 print_nlmsg_route_nhop(h, &r, nh, first); 510 first = false; 511 } 512 } else { 513 struct rta_mpath_nh nh = { 514 .gw = r.rta_gw, 515 .ifindex = r.rta_oif, 516 .rtax_mtu = r.rtax_mtu, 517 }; 518 519 print_nlmsg_route_nhop(h, &r, &nh, true); 520 } 521 } 522 523 static const char *operstate[] = { 524 "UNKNOWN", /* 0, IF_OPER_UNKNOWN */ 525 "NOTPRESENT", /* 1, IF_OPER_NOTPRESENT */ 526 "DOWN", /* 2, IF_OPER_DOWN */ 527 "LLDOWN", /* 3, IF_OPER_LOWERLAYERDOWN */ 528 "TESTING", /* 4, IF_OPER_TESTING */ 529 "DORMANT", /* 5, IF_OPER_DORMANT */ 530 "UP", /* 6, IF_OPER_UP */ 531 }; 532 533 static void 534 print_nlmsg_link(struct nl_helper *h, struct nlmsghdr *hdr) 535 { 536 struct snl_parsed_link l = {}; 537 struct snl_state *ss = &h->ss_cmd; 538 539 if (!snl_parse_nlmsg(ss, hdr, &snl_rtm_link_parser, &l)) 540 return; 541 542 // 20:19:41.333 add iface#3 vtnet0 admin UP oper UP mtu 1500 table inet.0 543 const char *cmd = get_action_name(hdr, RTM_NEWLINK); 544 print_line_prefix(cmd, "iface"); 545 546 printf("iface#%u %s ", l.ifi_index, l.ifla_ifname); 547 printf("admin %s ", (l.ifi_flags & IFF_UP) ? "UP" : "DOWN"); 548 if (l.ifla_operstate < NL_ARRAY_LEN(operstate)) 549 printf("oper %s ", operstate[l.ifla_operstate]); 550 if (l.ifla_mtu > 0) 551 printf("mtu %u ", l.ifla_mtu); 552 553 printf("\n"); 554 } 555 556 static void 557 print_nlmsg_addr(struct nl_helper *h, struct nlmsghdr *hdr) 558 { 559 struct snl_parsed_addr attrs = {}; 560 struct snl_state *ss = &h->ss_cmd; 561 562 if (!snl_parse_nlmsg(ss, hdr, &snl_rtm_addr_parser, &attrs)) 563 return; 564 565 // add addr 192.168.1.1/24 iface vtnet0 566 const char *cmd = get_action_name(hdr, RTM_NEWADDR); 567 print_line_prefix(cmd, "addr"); 568 569 char buf[128]; 570 struct sockaddr *addr = attrs.ifa_local ? attrs.ifa_local : attrs.ifa_address; 571 print_prefix(h, buf, sizeof(buf), addr, attrs.ifa_prefixlen); 572 printf("%s ", buf); 573 574 struct snl_parsed_link_simple link = {}; 575 get_ifdata(h, attrs.ifa_index, &link); 576 577 if (link.ifi_flags & IFF_POINTOPOINT) { 578 char buf[64]; 579 print_prefix(h, buf, sizeof(buf), attrs.ifa_address, -1); 580 printf("-> %s ", buf); 581 } 582 583 printf("iface %s ", link.ifla_ifname); 584 585 printf("\n"); 586 } 587 588 static const char *nudstate[] = { 589 "INCOMPLETE", /* 0x01(0) */ 590 "REACHABLE", /* 0x02(1) */ 591 "STALE", /* 0x04(2) */ 592 "DELAY", /* 0x08(3) */ 593 "PROBE", /* 0x10(4) */ 594 "FAILED", /* 0x20(5) */ 595 }; 596 597 #define NUD_INCOMPLETE 0x01 /* No lladdr, address resolution in progress */ 598 #define NUD_REACHABLE 0x02 /* reachable & recently resolved */ 599 #define NUD_STALE 0x04 /* has lladdr but it's stale */ 600 #define NUD_DELAY 0x08 /* has lladdr, is stale, probes delayed */ 601 #define NUD_PROBE 0x10 /* has lladdr, is stale, probes sent */ 602 #define NUD_FAILED 0x20 /* unused */ 603 604 605 static void 606 print_nlmsg_neigh(struct nl_helper *h, struct nlmsghdr *hdr) 607 { 608 struct snl_parsed_neigh attrs = {}; 609 struct snl_state *ss = &h->ss_cmd; 610 611 if (!snl_parse_nlmsg(ss, hdr, &snl_rtm_neigh_parser, &attrs)) 612 return; 613 614 // add addr 192.168.1.1 state %s lladdr %s iface vtnet0 615 const char *cmd = get_action_name(hdr, RTM_NEWNEIGH); 616 print_line_prefix(cmd, "neigh"); 617 618 char buf[128]; 619 print_prefix(h, buf, sizeof(buf), attrs.nda_dst, -1); 620 printf("%s ", buf); 621 622 struct snl_parsed_link_simple link = {}; 623 get_ifdata(h, attrs.nda_ifindex, &link); 624 625 for (unsigned int i = 0; i < NL_ARRAY_LEN(nudstate); i++) { 626 if ((1 << i) & attrs.ndm_state) { 627 printf("state %s ", nudstate[i]); 628 break; 629 } 630 } 631 632 if (attrs.nda_lladdr != NULL) { 633 int if_type = link.ifi_type; 634 635 if ((if_type == IFT_ETHER || if_type == IFT_L2VLAN || if_type == IFT_BRIDGE) && 636 NLA_DATA_LEN(attrs.nda_lladdr) == ETHER_ADDR_LEN) { 637 struct ether_addr *ll; 638 639 ll = (struct ether_addr *)NLA_DATA(attrs.nda_lladdr); 640 printf("lladdr %s ", ether_ntoa(ll)); 641 } else { 642 struct sockaddr_dl sdl = { 643 .sdl_len = sizeof(sdl), 644 .sdl_family = AF_LINK, 645 .sdl_index = attrs.nda_ifindex, 646 .sdl_type = if_type, 647 .sdl_alen = NLA_DATA_LEN(attrs.nda_lladdr), 648 }; 649 if (sdl.sdl_alen < sizeof(sdl.sdl_data)) { 650 void *ll = NLA_DATA(attrs.nda_lladdr); 651 652 memcpy(sdl.sdl_data, ll, sdl.sdl_alen); 653 printf("lladdr %s ", link_ntoa(&sdl)); 654 } 655 } 656 } 657 658 if (link.ifla_ifname != NULL) 659 printf("iface %s ", link.ifla_ifname); 660 printf("\n"); 661 } 662 663 static void 664 print_nlmsg_generic(struct nl_helper *h, struct nlmsghdr *hdr) 665 { 666 } 667 668 static void 669 print_nlmsg(struct nl_helper *h, struct nlmsghdr *hdr) 670 { 671 switch (hdr->nlmsg_type) { 672 case RTM_NEWLINK: 673 case RTM_DELLINK: 674 print_nlmsg_link(h, hdr); 675 break; 676 case RTM_NEWADDR: 677 case RTM_DELADDR: 678 print_nlmsg_addr(h, hdr); 679 break; 680 case RTM_NEWROUTE: 681 case RTM_DELROUTE: 682 print_nlmsg_route(h, hdr); 683 break; 684 case RTM_NEWNEIGH: 685 case RTM_DELNEIGH: 686 print_nlmsg_neigh(h, hdr); 687 break; 688 default: 689 print_nlmsg_generic(h, hdr); 690 } 691 692 snl_clear_lb(&h->ss_cmd); 693 } 694 695 void 696 monitor_nl(int fib) 697 { 698 struct snl_state ss_event = {}; 699 struct nl_helper h; 700 701 nl_init_socket(&ss_event); 702 nl_helper_init(&h); 703 704 int groups[] = { 705 RTNLGRP_LINK, 706 RTNLGRP_NEIGH, 707 RTNLGRP_NEXTHOP, 708 #ifdef INET 709 RTNLGRP_IPV4_IFADDR, 710 RTNLGRP_IPV4_ROUTE, 711 #endif 712 #ifdef INET6 713 RTNLGRP_IPV6_IFADDR, 714 RTNLGRP_IPV6_ROUTE, 715 #endif 716 }; 717 718 for (unsigned int i = 0; i < NL_ARRAY_LEN(groups); i++) { 719 int error; 720 int optval = groups[i]; 721 socklen_t optlen = sizeof(optval); 722 error = setsockopt(ss_event.fd, SOL_NETLINK, 723 NETLINK_ADD_MEMBERSHIP, &optval, optlen); 724 if (error != 0) 725 warn("Unable to subscribe to group %d", optval); 726 } 727 728 struct nlmsghdr *hdr; 729 while ((hdr = snl_read_message(&ss_event)) != NULL) 730 { 731 // printf("-- MSG type %d--\n", hdr->nlmsg_type); 732 print_nlmsg(&h, hdr); 733 snl_clear_lb(&h.ss_cmd); 734 snl_clear_lb(&ss_event); 735 } 736 737 snl_free(&ss_event); 738 nl_helper_free(&h); 739 exit(0); 740 } 741 742 static void 743 print_flushed_route(struct snl_parsed_route *r, struct sockaddr *gw) 744 { 745 struct sockaddr *sa = r->rta_dst; 746 747 printf("%-20.20s ", r->rta_rtflags & RTF_HOST ? 748 routename(sa) : netname(sa)); 749 sa = gw; 750 printf("%-20.20s ", routename(sa)); 751 if (r->rta_table >= 0) 752 printf("-fib %-3d ", r->rta_table); 753 printf("done\n"); 754 } 755 756 static int 757 flushroute_one(struct nl_helper *h, struct snl_parsed_route *r) 758 { 759 struct snl_state *ss = &h->ss_cmd; 760 struct snl_errmsg_data e = {}; 761 struct snl_writer nw; 762 763 snl_init_writer(ss, &nw); 764 765 struct nlmsghdr *hdr = snl_create_msg_request(&nw, NL_RTM_DELROUTE); 766 struct rtmsg *rtm = snl_reserve_msg_object(&nw, struct rtmsg); 767 rtm->rtm_family = r->rtm_family; 768 rtm->rtm_dst_len = r->rtm_dst_len; 769 770 snl_add_msg_attr_u32(&nw, RTA_TABLE, r->rta_table); 771 snl_add_msg_attr_ip(&nw, RTA_DST, r->rta_dst); 772 773 if (!snl_finalize_msg(&nw) || !snl_send_message(ss, hdr)) 774 return (ENOMEM); 775 776 if (!snl_read_reply_code(ss, hdr->nlmsg_seq, &e)) { 777 return (e.error); 778 if (e.error == EPERM) 779 errc(1, e.error, "RTM_DELROUTE failed:"); 780 else 781 warnc(e.error, "RTM_DELROUTE failed:"); 782 return (true); 783 }; 784 785 if (verbose) 786 print_nlmsg(h, hdr); 787 else { 788 if (r->rta_multipath != NULL) { 789 for (int i = 0; i < r->rta_multipath->num_nhops; i++) { 790 struct rta_mpath_nh *nh = &r->rta_multipath->nhops[i]; 791 792 print_flushed_route(r, nh->gw); 793 } 794 795 } else 796 print_flushed_route(r, r->rta_gw); 797 } 798 799 return (0); 800 } 801 802 int 803 flushroutes_fib_nl(int fib, int af) 804 { 805 struct snl_state ss = {}; 806 struct snl_writer nw; 807 struct nl_helper h = {}; 808 809 nl_init_socket(&ss); 810 snl_init_writer(&ss, &nw); 811 812 struct nlmsghdr *hdr = snl_create_msg_request(&nw, NL_RTM_GETROUTE); 813 hdr->nlmsg_flags |= NLM_F_DUMP; 814 struct rtmsg *rtm = snl_reserve_msg_object(&nw, struct rtmsg); 815 rtm->rtm_family = af; 816 snl_add_msg_attr_u32(&nw, RTA_TABLE, fib); 817 818 if (!snl_finalize_msg(&nw) || !snl_send_message(&ss, hdr)) { 819 snl_free(&ss); 820 return (EINVAL); 821 } 822 823 struct snl_errmsg_data e = {}; 824 uint32_t nlm_seq = hdr->nlmsg_seq; 825 826 nl_helper_init(&h); 827 828 while ((hdr = snl_read_reply_multi(&ss, nlm_seq, &e)) != NULL) { 829 struct snl_parsed_route r = { .rtax_weight = RT_DEFAULT_WEIGHT }; 830 int error; 831 832 if (!snl_parse_nlmsg(&ss, hdr, &snl_rtm_route_parser, &r)) 833 continue; 834 if (verbose) 835 print_nlmsg(&h, hdr); 836 if (r.rta_table != (uint32_t)fib || r.rtm_family != af) 837 continue; 838 if ((r.rta_rtflags & RTF_GATEWAY) == 0) 839 continue; 840 if (debugonly) 841 continue; 842 843 if ((error = flushroute_one(&h, &r)) != 0) { 844 if (error == EPERM) 845 errc(1, error, "RTM_DELROUTE failed:"); 846 else 847 warnc(error, "RTM_DELROUTE failed:"); 848 } 849 snl_clear_lb(&h.ss_cmd); 850 } 851 852 snl_free(&ss); 853 nl_helper_free(&h); 854 855 return (e.error); 856 } 857 858