/* Pfinet option parsing Copyright (C) 1996, 1997, 2000, 2001, 2006, 2007 Free Software Foundation, Inc. Written by Miles Bader <miles@gnu.org> This file is part of the GNU Hurd. The GNU Hurd is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. The GNU Hurd is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include <stdlib.h> #include <string.h> #include <hurd.h> #include <argp.h> #include <argz.h> #include <error.h> #include <netinet/in.h> #include <arpa/inet.h> #include "pfinet.h" #include <linux/netdevice.h> #include <linux/inetdevice.h> #include <linux/ip.h> #include <linux/route.h> #include <linux/rtnetlink.h> #include <net/route.h> #include <net/sock.h> #include <net/ip_fib.h> #include <net/ip6_fib.h> #include <net/ip6_route.h> #include <net/addrconf.h> /* Our interface to the set of devices. */ extern error_t find_device (char *name, struct device **device); extern error_t enumerate_devices (error_t (*fun) (struct device *dev)); /* devinet.c */ extern error_t configure_device (struct device *dev, uint32_t addr, uint32_t netmask, uint32_t peer, uint32_t broadcast); extern void inquire_device (struct device *dev, uint32_t *addr, uint32_t *netmask, uint32_t *peer, uint32_t *broadcast); /* addrconf.c */ extern struct inet6_dev *ipv6_find_idev (struct device *dev); extern int inet6_addr_add (int ifindex, struct in6_addr *pfx, int plen); extern int inet6_addr_del (int ifindex, struct in6_addr *pfx, int plen); #ifdef CONFIG_IPV6 static struct rt6_info * ipv6_get_dflt_router (void); #endif /* Pfinet options. Used for both startup and runtime. */ static const struct argp_option options[] = { {"interface", 'i', "DEVICE", 0, "Network interface to use", 1}, {0,0,0,0,"These apply to a given interface:", 2}, {"address", 'a', "ADDRESS", OPTION_ARG_OPTIONAL, "Set the network address"}, {"netmask", 'm', "MASK", OPTION_ARG_OPTIONAL, "Set the netmask"}, {"peer", 'p', "ADDRESS", OPTION_ARG_OPTIONAL, "Set the peer address"}, {"gateway", 'g', "ADDRESS", OPTION_ARG_OPTIONAL, "Set the default gateway"}, {"ipv4", '4', "NAME", 0, "Put active IPv4 translator on NAME"}, #ifdef CONFIG_IPV6 {"ipv6", '6', "NAME", 0, "Put active IPv6 translator on NAME"}, {"address6", 'A', "ADDR/LEN", OPTION_ARG_OPTIONAL, "Set the global IPv6 address"}, {"gateway6", 'G', "ADDRESS", OPTION_ARG_OPTIONAL, "Set the IPv6 default gateway"}, #endif {0} }; static const char doc[] = "Interface-specific options before the first \ interface specification apply to the first following interface; otherwise \ they apply to the previously specified interface."; /* Used to describe a particular interface during argument parsing. */ struct parse_interface { /* The network interface in question. */ struct device *device; /* New values to apply to it. (IPv4) */ uint32_t address, netmask, peer, gateway; #ifdef CONFIG_IPV6 /* New IPv6 configuration to apply. */ struct inet6_ifaddr address6; struct in6_addr gateway6; #endif }; /* Used to hold data during argument parsing. */ struct parse_hook { /* A list of specified interfaces and their corresponding options. */ struct parse_interface *interfaces; size_t num_interfaces; /* Interface to which options apply. If the device field isn't filled in then it should be by the next --interface option. */ struct parse_interface *curint; }; static void parse_interface_copy_device(struct device *src, struct parse_interface *dst) { uint32_t broad; struct rt_key key = { 0 }; struct inet6_dev *idev = NULL; struct fib_result res; inquire_device (src, &dst->address, &dst->netmask, &dst->peer, &broad); /* Get gateway */ dst->gateway = INADDR_NONE; key.oif = src->ifindex; if (! main_table->tb_lookup (main_table, &key, &res) && FIB_RES_GW(res) != INADDR_ANY) dst->gateway = FIB_RES_GW (res); #ifdef CONFIG_IPV6 if (pfinet_protid_portclasses[PORTCLASS_INET6] != MACH_PORT_NULL) idev = ipv6_find_idev(src); if (idev) { struct inet6_ifaddr *ifa = idev->addr_list; /* Look for IPv6 default router and add it to the interface, * if it belongs to it. */ struct rt6_info *rt6i = ipv6_get_dflt_router(); if (rt6i->rt6i_dev == src) memcpy (&dst->gateway6, &rt6i->rt6i_gateway, sizeof (struct in6_addr)); /* Search for global address and set it in dst */ do { if (!IN6_IS_ADDR_LINKLOCAL (&ifa->addr)) { memcpy (&dst->address6, ifa, sizeof (struct inet6_ifaddr)); break; } } while ((ifa = ifa->if_next)); } #endif } /* Adds an empty interface slot to H, and sets H's current interface to it, or returns an error. */ static error_t parse_hook_add_interface (struct parse_hook *h) { struct parse_interface *new = realloc (h->interfaces, (h->num_interfaces + 1) * sizeof (struct parse_interface)); if (! new) return ENOMEM; h->interfaces = new; h->num_interfaces++; h->curint = new + h->num_interfaces - 1; h->curint->device = 0; h->curint->address = INADDR_NONE; h->curint->netmask = INADDR_NONE; h->curint->peer = INADDR_NONE; h->curint->gateway = INADDR_NONE; #ifdef CONFIG_IPV6 memset (&h->curint->address6, 0, sizeof (struct inet6_ifaddr)); memset (&h->curint->gateway6, 0, sizeof (struct in6_addr)); #endif return 0; } #ifdef CONFIG_IPV6 static struct rt6_info * ipv6_get_dflt_router (void) { struct in6_addr daddr = { 0 }; struct fib6_node *fib = fib6_lookup (&ip6_routing_table, &daddr, NULL); return fib->leaf; } #endif /* CONFIG_IPV6 */ static error_t parse_opt (int opt, char *arg, struct argp_state *state) { error_t err = 0; struct parse_hook *h = state->hook; /* Return _ERR from this routine, and in the special case of OPT being ARGP_KEY_SUCCESS, remember to free H first. */ #define RETURN(_err) \ do { if (opt == ARGP_KEY_SUCCESS) \ { err = (_err); goto free_hook; } \ else \ return _err; } while (0) /* Print a parsing error message and (if exiting is turned off) return the error code ERR. */ #define PERR(err, fmt, args...) \ do { argp_error (state, fmt , ##args); RETURN (err); } while (0) /* Like PERR but for non-parsing errors. */ #define FAIL(rerr, status, perr, fmt, args...) \ do{ argp_failure (state, status, perr, fmt , ##args); RETURN (rerr); } while(0) /* Parse STR and return the corresponding internet address. If STR is not a valid internet address, signal an error mentioned TYPE. */ #undef ADDR #define ADDR(str, type) \ ({ unsigned long addr = inet_addr (str); \ if (addr == INADDR_NONE) PERR (EINVAL, "Malformed %s", type); \ addr; }) if (!arg && state->next < state->argc && (*state->argv[state->next] != '-')) { arg = state->argv[state->next]; state->next ++; } switch (opt) { struct parse_interface *in, *gw4_in; #ifdef CONFIG_IPV6 struct parse_interface *gw6_in; char *ptr; #endif case 'i': /* An interface. */ err = 0; if (h->curint->device) /* The current interface slot is not available. */ { /* First see if a previously specified one is being re-specified. */ for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) if (strcmp (in->device->name, arg) == 0) /* Re-use an old slot. */ { h->curint = in; return 0; } /* Add a new interface entry. */ err = parse_hook_add_interface (h); } in = h->curint; if (! err) err = find_device (arg, &in->device); if (err) FAIL (err, 10, err, "%s", arg); /* Set old interface values */ parse_interface_copy_device (in->device, in); break; case 'a': if (arg) { h->curint->address = ADDR (arg, "address"); if (!IN_CLASSA (ntohl (h->curint->address)) && !IN_CLASSB (ntohl (h->curint->address)) && !IN_CLASSC (ntohl (h->curint->address))) { if (IN_MULTICAST (ntohl (h->curint->address))) FAIL (EINVAL, 1, 0, "%s: Cannot set interface address to multicast address", arg); else FAIL (EINVAL, 1, 0, "%s: Illegal or undefined network address", arg); } } else { h->curint->address = ADDR ("0.0.0.0", "address"); h->curint->netmask = ADDR ("255.0.0.0", "netmask"); h->curint->gateway = INADDR_NONE; } break; case 'm': if (arg) h->curint->netmask = ADDR (arg, "netmask"); else h->curint->netmask = INADDR_NONE; break; case 'p': if (arg) h->curint->peer = ADDR (arg, "peer"); else h->curint->peer = INADDR_NONE; break; case 'g': if (arg) { /* Remove an possible other default gateway */ for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) in->gateway = INADDR_NONE; h->curint->gateway = ADDR (arg, "gateway"); } else h->curint->gateway = INADDR_NONE; break; case '4': pfinet_bind (PORTCLASS_INET, arg); /* Install IPv6 port class on bootstrap port. */ pfinet_bootstrap_portclass = PORTCLASS_INET6; break; #ifdef CONFIG_IPV6 case '6': pfinet_bind (PORTCLASS_INET6, arg); break; case 'A': if (arg) { if ((ptr = strchr (arg, '/'))) { h->curint->address6.prefix_len = atoi (ptr + 1); if (h->curint->address6.prefix_len > 128) FAIL (EINVAL, 1, 0, "%s: The prefix-length is invalid", arg); *ptr = 0; } else { h->curint->address6.prefix_len = 64; fprintf (stderr, "No prefix-length given, " "defaulting to %s/64.\n", arg); } if (inet_pton (AF_INET6, arg, &h->curint->address6.addr) <= 0) PERR (EINVAL, "Malformed address"); if (IN6_IS_ADDR_MULTICAST (&h->curint->address6.addr)) FAIL (EINVAL, 1, 0, "%s: Cannot set interface address to " "multicast address", arg); } else memset (&h->curint->address6, 0, sizeof (struct inet6_ifaddr)); break; case 'G': if (arg) { if (inet_pton (AF_INET6, arg, &h->curint->gateway6) <= 0) PERR (EINVAL, "Malformed gateway"); if (IN6_IS_ADDR_MULTICAST (&h->curint->gateway6)) FAIL (EINVAL, 1, 0, "%s: Cannot set gateway to " "multicast address", arg); } else memset (&h->curint->gateway6, 0, sizeof (struct in6_addr)); break; #endif /* CONFIG_IPV6 */ case ARGP_KEY_INIT: /* Initialize our parsing state. */ h = malloc (sizeof (struct parse_hook)); if (! h) FAIL (ENOMEM, 11, ENOMEM, "option parsing"); h->interfaces = 0; h->num_interfaces = 0; err = parse_hook_add_interface (h); if (err) FAIL (err, 12, err, "option parsing"); state->hook = h; break; case ARGP_KEY_SUCCESS: in = h->curint; if (! in->device) /* No specific interface specified; is that ok? */ if (in->address != INADDR_NONE || in->netmask != INADDR_NONE || in->gateway != INADDR_NONE) /* Some options were specified, so we need an interface. See if there's a single extant interface to use as a default. */ { err = find_device (0, &in->device); if (err) FAIL (err, 13, 0, "No default interface"); } #if 0 /* XXX what does this mean??? */ /* Check for bogus option combinations. */ for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) if (in->netmask != INADDR_NONE && in->address == INADDR_NONE && in->device->pa_addr == 0) /* Specifying a netmask for an address-less interface is a no-no. */ FAIL (EDESTADDRREQ, 14, 0, "Cannot set netmask"); #endif #ifdef CONFIG_IPV6 gw6_in = NULL; #endif gw4_in = NULL; for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) { /* delete interface if it doesn't match the actual netmask */ if (! ( (h->curint->address & h->curint->netmask) == (h->curint->gateway & h->curint->netmask))) h->curint->gateway = INADDR_NONE; if (in->gateway != INADDR_NONE) gw4_in = in; #ifdef CONFIG_IPV6 if (!IN6_IS_ADDR_UNSPECIFIED (&in->gateway6)) { if (gw6_in != NULL) FAIL (err, 15, 0, "Cannot have multiple IPv6 " "default gateways"); gw6_in = in; } #endif } /* Successfully finished parsing, return a result. */ pthread_mutex_lock (&global_lock); for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) { #ifdef CONFIG_IPV6 struct inet6_dev *idev = NULL; if (pfinet_protid_portclasses[PORTCLASS_INET6] != MACH_PORT_NULL && in->device) idev = ipv6_find_idev(in->device); #endif if (in->address == INADDR_NONE && in->netmask == INADDR_NONE) { h->curint->address = ADDR ("0.0.0.0", "address"); h->curint->netmask = ADDR ("255.0.0.0", "netmask"); } if (in->device) err = configure_device (in->device, in->address, in->netmask, in->peer, INADDR_NONE); if (err) { pthread_mutex_unlock (&global_lock); FAIL (err, 16, 0, "cannot configure interface"); } #ifdef CONFIG_IPV6 if (!idev) continue; /* First let's remove all non-local addresses. */ struct inet6_ifaddr *ifa = idev->addr_list; while (ifa) { struct inet6_ifaddr *c_ifa = ifa; ifa = ifa->if_next; if (!IN6_IS_ADDR_UNSPECIFIED (&in->address6.addr) && IN6_ARE_ADDR_EQUAL (&c_ifa->addr, &in->address6.addr)) memset (&in->address6, 0, sizeof (struct inet6_ifaddr)); else if (!IN6_IS_ADDR_LINKLOCAL (&c_ifa->addr) && !IN6_IS_ADDR_SITELOCAL (&c_ifa->addr)) inet6_addr_del (in->device->ifindex, &c_ifa->addr, c_ifa->prefix_len); } if (!IN6_IS_ADDR_UNSPECIFIED (&in->address6.addr)) { /* Now assign the new address */ inet6_addr_add (in->device->ifindex, &in->address6.addr, in->address6.prefix_len); } #endif /* CONFIG_IPV6 */ } /* Set the default gateway. This code is cobbled together from what the SIOCADDRT ioctl code does, and from the apparent functionality of the "netlink" layer from perusing a little. */ { struct kern_rta rta; struct { struct nlmsghdr nlh; struct rtmsg rtm; } req; struct fib_table *tb; req.nlh.nlmsg_pid = 0; req.nlh.nlmsg_seq = 0; req.nlh.nlmsg_len = NLMSG_LENGTH (sizeof req.rtm); memset (&req.rtm, 0, sizeof req.rtm); memset (&rta, 0, sizeof rta); req.rtm.rtm_scope = RT_SCOPE_UNIVERSE; req.rtm.rtm_type = RTN_UNICAST; req.rtm.rtm_protocol = RTPROT_STATIC; if (!gw4_in) { /* Delete any existing default route on configured devices */ for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) { req.nlh.nlmsg_type = RTM_DELROUTE; req.nlh.nlmsg_flags = 0; rta.rta_oif = &in->device->ifindex; tb = fib_get_table (req.rtm.rtm_table); if (tb) { err = - (*tb->tb_delete) (tb, &req.rtm, &rta, &req.nlh, 0); if (err && err != ESRCH) { pthread_mutex_unlock (&global_lock); FAIL (err, 17, 0, "cannot remove old default gateway"); } err = 0; } } } else { /* Add a default route, replacing any existing one. */ rta.rta_oif = &gw4_in->device->ifindex; rta.rta_gw = &gw4_in->gateway; req.nlh.nlmsg_type = RTM_NEWROUTE; req.nlh.nlmsg_flags = NLM_F_REQUEST | NLM_F_CREATE | NLM_F_REPLACE; tb = fib_new_table (req.rtm.rtm_table); err = (!tb ? ENOBUFS : - (*tb->tb_insert) (tb, &req.rtm, &rta, &req.nlh, 0)); if (err) { pthread_mutex_unlock (&global_lock); FAIL (err, 17, 0, "cannot set default gateway"); } } } /* Set IPv6 default router. */ #ifdef CONFIG_IPV6 if (pfinet_protid_portclasses[PORTCLASS_INET6] != MACH_PORT_NULL) { struct rt6_info *rt6i = ipv6_get_dflt_router (); if (!gw6_in || rt6i->rt6i_dev != gw6_in->device || !IN6_ARE_ADDR_EQUAL (&rt6i->rt6i_gateway, &gw6_in->gateway6)) { /* Delete any existing default route on configured devices */ for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) if (rt6i->rt6i_dev == in->device || gw6_in ) rt6_purge_dflt_routers (0); if (gw6_in) rt6_add_dflt_router (&gw6_in->gateway6, gw6_in->device); } } #endif /* Setup the routing required for DHCP. */ for (in = h->interfaces; in < h->interfaces + h->num_interfaces; in++) { struct kern_rta rta; struct { struct nlmsghdr nlh; struct rtmsg rtm; } req; struct fib_table *tb; struct rtentry route; struct sockaddr_in *dst; struct device *dev; if (!in->device) continue; dst = (struct sockaddr_in *) &route.rt_dst; if (!in->device->name) { pthread_mutex_unlock (&global_lock); FAIL (ENODEV, 17, 0, "unknown device"); } dev = dev_get (in->device->name); if (!dev) { pthread_mutex_unlock (&global_lock); FAIL (ENODEV, 17, 0, "unknown device"); } /* Simulate the SIOCADDRT behavior. */ memset (&route, 0, sizeof (struct rtentry)); memset (&req.rtm, 0, sizeof req.rtm); memset (&rta, 0, sizeof rta); req.nlh.nlmsg_type = RTM_NEWROUTE; /* Append this routing for 0.0.0.0. By this way we can send always dhcp messages (e.g dhcp renew). */ req.nlh.nlmsg_flags = NLM_F_REQUEST | NLM_F_CREATE | NLM_F_APPEND; req.rtm.rtm_protocol = RTPROT_BOOT; req.rtm.rtm_scope = RT_SCOPE_LINK; req.rtm.rtm_type = RTN_UNICAST; rta.rta_dst = &dst->sin_addr.s_addr; rta.rta_oif = &dev->ifindex; tb = fib_new_table (req.rtm.rtm_table); if (tb) err = tb->tb_insert (tb, &req.rtm, &rta, &req.nlh, NULL); else err = ENOBUFS; if (err) { pthread_mutex_unlock (&global_lock); FAIL (err, 17, 0, "cannot add route"); } } pthread_mutex_unlock (&global_lock); /* Fall through to free hook. */ case ARGP_KEY_ERROR: /* Parsing error occurred, free everything. */ free_hook: free (h->interfaces); free (h); break; default: return ARGP_ERR_UNKNOWN; } return err; } struct argp pfinet_argp = { options, parse_opt, 0, doc }; struct argp *trivfs_runtime_argp = &pfinet_argp; error_t trivfs_append_args (struct trivfs_control *fsys, char **argz, size_t *argz_len) { error_t add_dev_opts (struct device *dev) { error_t err = 0; uint32_t addr, mask, peer, broad; struct rt_key key = { 0 }; struct fib_result res; inquire_device (dev, &addr, &mask, &peer, &broad); #define ADD_OPT(fmt, args...) \ do { char buf[100]; \ if (! err) { \ snprintf (buf, sizeof buf, fmt , ##args); \ err = argz_add (argz, argz_len, buf); } } while (0) #define ADD_ADDR_OPT(name, addr) \ do { struct in_addr i; \ i.s_addr = (addr); \ ADD_OPT ("--%s=%s", name, inet_ntoa (i)); } while (0) ADD_OPT ("--interface=%s", dev->name); if (addr != INADDR_NONE) ADD_ADDR_OPT ("address", addr); if (mask != INADDR_NONE) ADD_ADDR_OPT ("netmask", mask); if (peer != addr) ADD_ADDR_OPT ("peer", peer); key.oif = dev->ifindex; if (! main_table->tb_lookup (main_table, &key, &res) && FIB_RES_GW(res) != INADDR_ANY) ADD_ADDR_OPT ("gateway", FIB_RES_GW (res)); #undef ADD_ADDR_OPT #ifdef CONFIG_IPV6 struct inet6_dev *idev = NULL; if (pfinet_protid_portclasses[PORTCLASS_INET6] != MACH_PORT_NULL) idev = ipv6_find_idev(dev); if (idev) { struct inet6_ifaddr *ifa; static char addr_buf[INET6_ADDRSTRLEN]; /* Push all IPv6 addresses assigned to the interface. */ for (ifa = idev->addr_list; ifa; ifa = ifa->if_next) { inet_ntop (AF_INET6, &ifa->addr, addr_buf, INET6_ADDRSTRLEN); ADD_OPT ("--address6=%s/%d", addr_buf, ifa->prefix_len); } /* Last not least push --gateway6 option. */ struct rt6_info *rt6i = ipv6_get_dflt_router (); if(rt6i->rt6i_dev == dev) { inet_ntop (AF_INET6, &rt6i->rt6i_gateway, addr_buf, INET6_ADDRSTRLEN); ADD_OPT ("--gateway6=%s", addr_buf); } } #endif /* CONFIG_IPV6 */ #undef ADD_OPT return err; } return enumerate_devices (add_dev_opts); }