/* route.c - Display/edit network routing table. * * Copyright 2012 Ranjan Kumar * Copyright 2013 Kyungwan Han * * No Standard * * TODO: autodetect -net -host target dev -A (but complain) * route add -net target 10.0.0.0 netmask 255.0.0.0 dev eth0 * route del delete * delete net route, must match netmask, informative error message * * mod dyn reinstate metric netmask gw mss window irtt dev USE_ROUTE(NEWTOY(route, "?neA:", TOYFLAG_SBIN)) config ROUTE bool "route" default n help usage: route [-ne] [-A [inet|inet6]] [add|del TARGET [OPTIONS]] Display, add or delete network routes in the "Forwarding Information Base". -n Show numerical addresses (no DNS lookups) -e display netstat fields Routing means sending packets out a network interface to an address. The kernel can tell where to send packets one hop away by examining each interface's address and netmask, so the most common use of this command is to identify a "gateway" that forwards other traffic. Assigning an address to an interface automatically creates an appropriate network route ("ifconfig eth0 10.0.2.15/8" does "route add 10.0.0.0/8 eth0" for you), although some devices (such as loopback) won't show it in the table. For machines more than one hop away, you need to specify a gateway (ala "route add default gw 10.0.2.2"). The address "default" is a wildcard address (0.0.0.0/0) matching all packets without a more specific route. Available OPTIONS include: reject - blocking route (force match failure) dev NAME - force packets out this interface (ala "eth0") netmask - old way of saying things like ADDR/24 gw ADDR - forward packets to gateway ADDR */ #define FOR_route #include "toys.h" #include #include GLOBALS( char *family; ) #define DEFAULT_PREFIXLEN 128 #define INVALID_ADDR 0xffffffffUL #define IPV6_ADDR_LEN 40 //32 + 7 (':') + 1 ('\0') struct _arglist { char *arg; int action; }; static struct _arglist arglist1[] = { { "add", 1 }, { "del", 2 }, { "delete", 2 }, { NULL, 0 } }; static struct _arglist arglist2[] = { { "-net", 1 }, { "-host", 2 }, { NULL, 0 } }; void xsend(int sockfd, void *buf, size_t len) { if (send(sockfd, buf, len, 0) != len) exit(EXIT_FAILURE); } int xrecv(int sockfd, void *buf, size_t len) { int msg_len = recv(sockfd, buf, len, 0); if (msg_len < 0) exit(EXIT_FAILURE); return msg_len; } void send_nlrtmsg(int fd, int type, int flags, struct rtmsg *rt) { struct { struct nlmsghdr nl; struct rtmsg rt; } req; memset(&req, 0, sizeof(req)); req.nl.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg)); req.nl.nlmsg_type = type; req.nl.nlmsg_flags = flags; req.nl.nlmsg_pid = getpid(); req.nl.nlmsg_seq = 1; req.rt = *rt; xsend(fd, &req, sizeof(req)); } // to get the host name from the given ip. static int get_hostname(char *ipstr, struct sockaddr_in *sockin) { struct hostent *host; sockin->sin_family = AF_INET; sockin->sin_port = 0; if (!strcmp(ipstr, "default")) { sockin->sin_addr.s_addr = INADDR_ANY; return 1; } if (inet_aton(ipstr, &sockin->sin_addr)) return 0; if (!(host = gethostbyname(ipstr))) perror_exit("resolving '%s'", ipstr); memcpy(&sockin->sin_addr, host->h_addr_list[0], sizeof(struct in_addr)); return 0; } // used to extract the address info from the given ip. static int get_addrinfo(char *ip, struct sockaddr_in6 *sock_in6) { struct addrinfo hints, *result; int status = 0; memset(&hints, 0, sizeof(struct addrinfo)); hints.ai_family = AF_INET6; if ((status = getaddrinfo(ip, NULL, &hints, &result))) { perror_msg("getaddrinfo: %s", gai_strerror(status)); return -1; } if (result) { memcpy(sock_in6, result->ai_addr, sizeof(*sock_in6)); freeaddrinfo(result); } return 0; } static void display_routes(sa_family_t family) { int fd, msg_hdr_len, route_protocol; struct nlmsghdr buf[8192 / sizeof(struct nlmsghdr)]; struct nlmsghdr *msg_hdr_ptr; struct rtmsg req; struct rtmsg *route_entry; struct rtattr *route_attribute; fd = xsocket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE); memset(&req, 0, sizeof(req)); req.rtm_family = family; req.rtm_table = RT_TABLE_MAIN; send_nlrtmsg(fd, RTM_GETROUTE, NLM_F_REQUEST | NLM_F_DUMP, &req); if (family == AF_INET) { xprintf("Kernel IP routing table\n" "Destination Gateway Genmask Flags %s Iface\n", (toys.optflags & FLAG_e) ? " MSS Window irtt" : "Metric Ref Use"); } else { xprintf("Kernel IPv6 routing table\n" "%-31s%-26s Flag Met Ref Use If\n", "Destination", "Next Hop"); } msg_hdr_len = xrecv(fd, buf, sizeof(buf)); msg_hdr_ptr = (struct nlmsghdr *) buf; while (msg_hdr_ptr->nlmsg_type != NLMSG_DONE) { while (NLMSG_OK(msg_hdr_ptr, msg_hdr_len)) { route_entry = (struct rtmsg *) NLMSG_DATA(msg_hdr_ptr); route_protocol = route_entry->rtm_protocol; // Annoyingly NLM_F_MATCH is not yet implemented so even if we pass in // RT_TABLE_MAIN with RTM_GETROUTE it still returns everything so we // have to filter here. if (route_entry->rtm_table == RT_TABLE_MAIN) { struct in_addr netmask_addr; char destip[INET6_ADDRSTRLEN]; char gateip[INET6_ADDRSTRLEN]; char netmask[32]; char flags[10] = "U"; uint32_t priority = 0; uint32_t mss = 0; uint32_t win = 0; uint32_t irtt = 0; uint32_t refcount = 0; //todo(emolitor): implement char if_name[IF_NAMESIZE] = "-"; uint32_t route_netmask; struct rtattr *metric; uint32_t metric_len; if (family == AF_INET) { if (!(toys.optflags & FLAG_n)) strcpy(destip, "default"); else strcpy(destip, "0.0.0.0"); if (!(toys.optflags & FLAG_n)) strcpy(gateip, "*"); else strcpy(gateip, "0.0.0.0"); strcpy(netmask, "0.0.0.0"); } else { strcpy(destip, "::"); strcpy(gateip, "::"); } route_netmask = route_entry->rtm_dst_len; if (route_netmask == 0) { netmask_addr.s_addr = ~((in_addr_t) -1); } else { netmask_addr.s_addr = htonl(~((1 << (32 - route_netmask)) - 1)); } inet_ntop(AF_INET, &netmask_addr, netmask, sizeof(netmask)); route_attribute = RTM_RTA(route_entry); int route_attribute_len = RTM_PAYLOAD(msg_hdr_ptr); while (RTA_OK(route_attribute, route_attribute_len)) { switch (route_attribute->rta_type) { case RTA_DST: inet_ntop(family, RTA_DATA(route_attribute), destip, INET6_ADDRSTRLEN); break; case RTA_GATEWAY: inet_ntop(family, RTA_DATA(route_attribute), gateip, INET6_ADDRSTRLEN); strcat(flags, "G"); break; case RTA_PRIORITY: priority = *(uint32_t *) RTA_DATA(route_attribute); break; case RTA_OIF: if_indextoname(*((int *) RTA_DATA(route_attribute)), if_name); break; case RTA_METRICS: metric_len = RTA_PAYLOAD(route_attribute); for (metric = RTA_DATA(route_attribute); RTA_OK(metric, metric_len); metric=RTA_NEXT(metric, metric_len)) { if (metric->rta_type == RTAX_ADVMSS) { mss = *(uint32_t *) RTA_DATA(metric); } else if (metric->rta_type == RTAX_WINDOW) { win = *(uint32_t *) RTA_DATA(metric); } else if (metric->rta_type == RTAX_RTT) { irtt = (*(uint32_t *) RTA_DATA(metric)) / 8; } else { printf("Unknown metric->rta_type %u\n", metric->rta_type); } } break; case RTA_CACHEINFO: // todo(emolitor): implement for AF_INET6 break; case RTA_PREF: // todo(emolitor): implement for AF_INET6 break; } route_attribute = RTA_NEXT(route_attribute, route_attribute_len); } // Set/Update flags, rtnetlink.h note RTPROT_REDIRECT is not used if (route_entry->rtm_type == RTN_UNREACHABLE) flags[0] = '!'; if (route_netmask == 32) strcat(flags, "H"); if (route_protocol == RTPROT_REDIRECT) strcat(flags, "D"); if (family == AF_INET) { // Ref is not used by the kernel so hard coding to 0 // IPv4 caching is disabled so hard coding Use to 0 xprintf("%-15.15s %-15.15s %-16s%-6s", destip, gateip, netmask, flags); if (toys.optflags & FLAG_e) { xprintf("%5d %-5d %6d %s\n", mss, win, irtt, if_name); } else xprintf("%-6d %-2d %7d %s\n", priority, refcount, 0, if_name); } else {; char dest_with_mask[INET6_ADDRSTRLEN + 4]; snprintf(dest_with_mask, INET6_ADDRSTRLEN + 4, "%s/%u", destip, route_netmask); xprintf("%-30s %-26s %-4s %-3d %-4d %2d %-8s\n", dest_with_mask, gateip, flags, priority, refcount, 0, if_name); } } msg_hdr_ptr = NLMSG_NEXT(msg_hdr_ptr, msg_hdr_len); } msg_hdr_len = xrecv(fd, buf, sizeof(buf)); msg_hdr_ptr = (struct nlmsghdr *) buf; } xclose(fd); } /* * find the given parameter in list like add/del/net/host. * and if match found return the appropriate action. */ static int get_action(char ***argv, struct _arglist *list) { struct _arglist *alist; if (!**argv) return 0; for (alist = list; alist->arg; alist++) { //find the given parameter in list if (!strcmp(**argv, alist->arg)) { *argv += 1; return alist->action; } } return 0; } /* * used to get the params like: metric, netmask, gw, mss, window, irtt, dev and their values. * additionally set the flag values for reject, mod, dyn and reinstate. */ static void get_next_params(char **argv, struct rtentry *rt, char **netmask) { for (;*argv;argv++) { if (!strcmp(*argv, "reject")) rt->rt_flags |= RTF_REJECT; else if (!strcmp(*argv, "mod")) rt->rt_flags |= RTF_MODIFIED; else if (!strcmp(*argv, "dyn")) rt->rt_flags |= RTF_DYNAMIC; else if (!strcmp(*argv, "reinstate")) rt->rt_flags |= RTF_REINSTATE; else { if (!argv[1]) help_exit(0); //set the metric field in the routing table. if (!strcmp(*argv, "metric")) rt->rt_metric = atolx_range(argv[1], 0, ULONG_MAX) + 1; else if (!strcmp(*argv, "netmask")) { //when adding a network route, the netmask to be used. struct sockaddr sock; unsigned int addr_mask = (((struct sockaddr_in *)&((rt)->rt_genmask))->sin_addr.s_addr); if (addr_mask) help_exit("dup netmask"); *netmask = argv[1]; get_hostname(*netmask, (struct sockaddr_in *) &sock); rt->rt_genmask = sock; } else if (!strcmp(*argv, "gw")) { //route packets via a gateway. if (!(rt->rt_flags & RTF_GATEWAY)) { if (!get_hostname(argv[1], (struct sockaddr_in *) &rt->rt_gateway)) rt->rt_flags |= RTF_GATEWAY; else perror_exit("gateway '%s' is a NETWORK", argv[1]); } else help_exit("dup gw"); } else if (!strcmp(*argv, "mss")) { //set the TCP Maximum Segment Size for connections over this route. rt->rt_mtu = atolx_range(argv[1], 64, 65536); rt->rt_flags |= RTF_MSS; } else if (!strcmp(*argv, "window")) { //set the TCP window size for connections over this route to W bytes. rt->rt_window = atolx_range(argv[1], 128, INT_MAX); //win low rt->rt_flags |= RTF_WINDOW; } else if (!strcmp(*argv, "irtt")) { rt->rt_irtt = atolx_range(argv[1], 0, INT_MAX); rt->rt_flags |= RTF_IRTT; } else if (!strcmp(*argv, "dev") && !rt->rt_dev) rt->rt_dev = argv[1]; else help_exit("no '%s'", *argv); argv++; } } if (!rt->rt_dev && (rt->rt_flags & RTF_REJECT)) rt->rt_dev = (char *)"lo"; } // verify the netmask and conflict in netmask and route address. static void verify_netmask(struct rtentry *rt, char *netmask) { unsigned int addr_mask = (((struct sockaddr_in *)&((rt)->rt_genmask))->sin_addr.s_addr); unsigned int router_addr = ~(unsigned int)(((struct sockaddr_in *)&((rt)->rt_dst))->sin_addr.s_addr); if (addr_mask) { addr_mask = ~ntohl(addr_mask); if ((rt->rt_flags & RTF_HOST) && addr_mask != INVALID_ADDR) perror_exit("conflicting netmask and host route"); if (addr_mask & (addr_mask + 1)) perror_exit("wrong netmask '%s'", netmask); addr_mask = ((struct sockaddr_in *) &rt->rt_dst)->sin_addr.s_addr; if (addr_mask & router_addr) perror_exit("conflicting netmask and route address"); } } // add/del a route. static void setroute(char **argv) { struct rtentry rt; char *netmask, *targetip; int is_net_or_host = 0, sokfd, arg2_action; int action = get_action(&argv, arglist1); //verify the arg for add/del. if (!action || !*argv) help_exit("setroute"); arg2_action = get_action(&argv, arglist2); //verify the arg for -net or -host if (!*argv) help_exit("setroute"); memset(&rt, 0, sizeof(struct rtentry)); targetip = *argv++; netmask = strchr(targetip, '/'); if (netmask) { *netmask++ = 0; //used to verify the netmask and route conflict. (((struct sockaddr_in *)&rt.rt_genmask)->sin_addr.s_addr) = htonl((1<<(32-atolx_range(netmask, 0, 32)))-1); rt.rt_genmask.sa_family = AF_INET; netmask = 0; } else netmask = "default"; is_net_or_host = get_hostname(targetip, (void *)&rt.rt_dst); if (arg2_action) is_net_or_host = arg2_action & 1; rt.rt_flags = ((is_net_or_host) ? RTF_UP : (RTF_UP | RTF_HOST)); get_next_params(argv, &rt, (char **)&netmask); verify_netmask(&rt, (char *)netmask); if ((action == 1) && (rt.rt_flags & RTF_HOST)) (((struct sockaddr_in *)&((rt).rt_genmask))->sin_addr.s_addr) = INVALID_ADDR; sokfd = xsocket(AF_INET, SOCK_DGRAM, 0); if (action == 1) xioctl(sokfd, SIOCADDRT, &rt); else xioctl(sokfd, SIOCDELRT, &rt); xclose(sokfd); } /* * get prefix len (if any) and remove the prefix from target ip. * if no prefix then set default prefix len. */ static void is_prefix_inet6(char **tip, struct in6_rtmsg *rt) { unsigned long plen; char *prefix = strchr(*tip, '/'); if (prefix) { *prefix = '\0'; plen = atolx_range(prefix + 1, 0, 128); //DEFAULT_PREFIXLEN); } else plen = DEFAULT_PREFIXLEN; rt->rtmsg_flags = (plen == DEFAULT_PREFIXLEN) ? (RTF_UP | RTF_HOST) : RTF_UP; rt->rtmsg_dst_len = plen; } /* * used to get the params like: metric, gw, dev and their values. * additionally set the flag values for mod and dyn. */ static void get_next_params_inet6(char **argv, struct sockaddr_in6 *sock_in6, struct in6_rtmsg *rt, char **dev_name) { for (;*argv;argv++) { if (!strcmp(*argv, "mod")) rt->rtmsg_flags |= RTF_MODIFIED; else if (!strcmp(*argv, "dyn")) rt->rtmsg_flags |= RTF_DYNAMIC; else { if (!argv[1]) help_exit(0); if (!strcmp(*argv, "metric")) rt->rtmsg_metric = atolx_range(argv[1], 0, ULONG_MAX); else if (!strcmp(*argv, "gw")) { //route packets via a gateway. if (!(rt->rtmsg_flags & RTF_GATEWAY)) { if (!get_addrinfo(argv[1], (struct sockaddr_in6 *) &sock_in6)) { memcpy(&rt->rtmsg_gateway, sock_in6->sin6_addr.s6_addr, sizeof(struct in6_addr)); rt->rtmsg_flags |= RTF_GATEWAY; } else perror_exit("resolving '%s'", argv[1]); } else help_exit(0); } else if (!strcmp(*argv, "dev")) { if (!*dev_name) *dev_name = argv[1]; } else help_exit(0); argv++; } } } // add/del a route. static void setroute_inet6(char **argv) { struct sockaddr_in6 sock_in6; struct in6_rtmsg rt; char *targetip, *dev_name = 0; int sockfd, action = get_action(&argv, arglist1); if (!action || !*argv) help_exit(0); memset(&sock_in6, 0, sizeof(struct sockaddr_in6)); memset(&rt, 0, sizeof(struct in6_rtmsg)); targetip = *argv++; if (!*argv) help_exit(0); if (!strcmp(targetip, "default")) { rt.rtmsg_flags = RTF_UP; rt.rtmsg_dst_len = 0; } else { is_prefix_inet6((char **)&targetip, &rt); if (get_addrinfo(targetip, (struct sockaddr_in6 *) &sock_in6)) perror_exit("resolving '%s'", targetip); } rt.rtmsg_metric = 1; //default metric. memcpy(&rt.rtmsg_dst, sock_in6.sin6_addr.s6_addr, sizeof(struct in6_addr)); get_next_params_inet6(argv, &sock_in6, &rt, (char **)&dev_name); sockfd = xsocket(AF_INET6, SOCK_DGRAM, 0); if (dev_name) { char ifre_buf[sizeof(struct ifreq)] = {0,}; struct ifreq *ifre = (struct ifreq*)ifre_buf; xstrncpy(ifre->ifr_name, dev_name, IFNAMSIZ); xioctl(sockfd, SIOGIFINDEX, ifre); rt.rtmsg_ifindex = ifre->ifr_ifindex; } if (action == 1) xioctl(sockfd, SIOCADDRT, &rt); else xioctl(sockfd, SIOCDELRT, &rt); xclose(sockfd); } void route_main(void) { if (!TT.family) TT.family = "inet"; if (!*toys.optargs) { if (!strcmp(TT.family, "inet")) display_routes(AF_INET); else if (!strcmp(TT.family, "inet6")) display_routes(AF_INET6); else help_exit(0); } else { if (!strcmp(TT.family, "inet6")) setroute_inet6(toys.optargs); else setroute(toys.optargs); } }