diff options
author | Kenneth R Westerback <krw@cvs.openbsd.org> | 2012-10-30 16:41:29 +0000 |
---|---|---|
committer | Kenneth R Westerback <krw@cvs.openbsd.org> | 2012-10-30 16:41:29 +0000 |
commit | 44ada59129bd8071d04a44d4c39ae658e1791c82 (patch) | |
tree | 896a8aa5392a339d98d138cc1b3151b78e6f1aed /sbin/dhclient | |
parent | 73b501f9e7b72dd509a9b8d60ced65be3db8742b (diff) |
New file holding functions to perform direct interface and routing
maninpulations. Inspired by similar files in bgpd, dvmrpd, ldpd, etc.
Necessary for imminent nuking of dhclient-script.
Not linked into build at the moment.
Diffstat (limited to 'sbin/dhclient')
-rw-r--r-- | sbin/dhclient/kroute.c | 638 |
1 files changed, 638 insertions, 0 deletions
diff --git a/sbin/dhclient/kroute.c b/sbin/dhclient/kroute.c new file mode 100644 index 00000000000..47b7d75a9eb --- /dev/null +++ b/sbin/dhclient/kroute.c @@ -0,0 +1,638 @@ +/* $OpenBSD: kroute.c,v 1.1 2012/10/30 16:41:28 krw Exp $ */ + +/* + * Copyright 2012 Kenneth R Westerback <krw@openbsd.org> + * + * Permission to use, copy, modify, and distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ + +#include <sys/ioctl.h> +#include <sys/param.h> +#include <sys/types.h> +#include <sys/socket.h> +#include <sys/sysctl.h> +#include <sys/uio.h> + +#include <net/if_types.h> +#include <net/if.h> + +#include "dhcpd.h" +#include "privsep.h" + +/* + * Do equivalent of + * + * route -q $rdomain -n flush -inet -iface $interface + * arp -dan + */ +void +flush_routes_and_arp_cache(char *ifname, int rdomain) +{ + size_t len; + struct imsg_hdr hdr; + struct buf *buf; + + hdr.code = IMSG_FLUSH_ROUTES; + hdr.len = sizeof(hdr) + + sizeof(len) + strlen(ifname) + + sizeof(len) + sizeof(rdomain); + + buf = buf_open(hdr.len); + buf_add(buf, &hdr, sizeof(hdr)); + + len = strlen(ifname); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, ifname, len); + + len = sizeof(rdomain); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &rdomain, len); + + buf_close(privfd, buf); +} + +void +priv_flush_routes_and_arp_cache(char *ifname, int rdomain) +{ + struct sockaddr *rti_info[RTAX_MAX]; + int mib[7]; + size_t needed; + char *lim, *buf, *next, *routelabel; + struct rt_msghdr *rtm; + struct sockaddr *sa; + struct sockaddr_dl *sdl; + struct sockaddr_inarp *sin; + struct sockaddr_rtlabel *sa_rl; + int s, seqno = 0, rlen, i; + unsigned int ifi_index; + + ifi_index = if_nametoindex(ifname); + if (ifi_index == 0) + error("No interface index found for '%s'", ifname); + + mib[0] = CTL_NET; + mib[1] = PF_ROUTE; + mib[2] = 0; + mib[3] = 0; + mib[4] = NET_RT_DUMP; + mib[5] = 0; + mib[6] = rdomain; + + if (sysctl(mib, 7, NULL, &needed, NULL, 0) == -1) { + if (rdomain != 0 && errno == EINVAL) + return; + error("could not get routes"); + } + + if (needed == 0) + return; + + if ((buf = malloc(needed)) == NULL) + error("malloc"); + + if (sysctl(mib, 7, buf, &needed, NULL, 0) == -1) + error("sysctl retrieval of routes: %m"); + + if ((s = socket(AF_ROUTE, SOCK_RAW, 0)) == -1) + error("opening socket to flush routes: %m"); + + if (asprintf(&routelabel, "DHCLIENT %d", (int)getpid()) == -1) + error("recreating route label: %m"); + +#define ROUNDUP(a) \ + ((a) > 0 ? (1 + (((a) - 1) | (sizeof(long) - 1))) : sizeof(long)) + + lim = buf + needed; + for (next = buf; next < lim; next += rtm->rtm_msglen) { + rtm = (struct rt_msghdr *)next; + if (rtm->rtm_version != RTM_VERSION) + continue; + + sa = (struct sockaddr *)(next + rtm->rtm_hdrlen); + if (sa->sa_family == AF_KEY) + continue; /* Don't flush SPD */ + + bzero(rti_info, sizeof(rti_info)); + for (i = 0; i < RTAX_MAX; i++) { + if (rtm->rtm_addrs & (1 << i)) { + rti_info[i] = sa; + sa = (struct sockaddr *)((char *)(sa) + + ROUNDUP(sa->sa_len)); + } + } + + sa = (struct sockaddr *)(next + rtm->rtm_hdrlen); + + if (rti_info[RTAX_LABEL]) { + sa_rl = (struct sockaddr_rtlabel *)rti_info[RTAX_LABEL]; + if (strcmp(routelabel, sa_rl->sr_label)) + continue; + } else if (rtm->rtm_flags & RTF_LLINFO) { + if (rtm->rtm_flags & RTF_GATEWAY) + continue; + + /* XXXX Check for AF_INET too? (arp ask for them) */ + /* XXXX Need 'retry' for proxy entries? (arp does) */ + + sin = (struct sockaddr_inarp *)(sa); + sdl = (struct sockaddr_dl *)(ROUNDUP(sin->sin_len) + + (char *)sin); + + if (sdl->sdl_family == AF_LINK) { + switch (sdl->sdl_type) { + case IFT_ETHER: + case IFT_FDDI: + case IFT_ISO88023: + case IFT_ISO88024: + case IFT_ISO88025: + case IFT_CARP: + /* Delete it. */ + ; + default: + continue; + } + } + } else + continue; + + rtm->rtm_type = RTM_DELETE; + rtm->rtm_seq = seqno; + rtm->rtm_tableid = rdomain; + + rlen = write(s, next, rtm->rtm_msglen); + if (rlen == -1) { + if (errno != ESRCH) + error("RTM_DELETE write: %m"); + } else if (rlen < (int)rtm->rtm_msglen) + error("short RTM_DELETE write (%d)\n", rlen); + + seqno++; + } + + close(s); + free(buf); + free(routelabel); +} + +/* + * [priv_]add_default_route is the equivalent of + * + * route -q $rdomain -n flush -inet -iface $interface + * + * and one of + * + * route -q $rdomain add default -iface $router + * route -q $rdomain add default $router + * + * depending on the contents of the gateway parameter. + */ +void +add_default_route(char *ifname, int rdomain, struct iaddr addr, + struct iaddr gateway) +{ + size_t len; + struct imsg_hdr hdr; + struct buf *buf; + + hdr.code = IMSG_ADD_DEFAULT_ROUTE; + hdr.len = sizeof(hdr) + + sizeof(len) + strlen(ifname) + + sizeof(len) + sizeof(rdomain) + + sizeof(len) + sizeof(addr) + + sizeof(len) + sizeof(gateway); + + buf = buf_open(hdr.len); + buf_add(buf, &hdr, sizeof(hdr)); + + len = strlen(ifname); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, ifname, len); + + len = sizeof(rdomain); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &rdomain, len); + + len = sizeof(addr); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &addr, len); + + len = sizeof(gateway); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &gateway, len); + + buf_close(privfd, buf); +} + +void +priv_add_default_route(char *ifname, int rdomain, struct iaddr addr, + struct iaddr router) +{ + struct rt_msghdr rtm; + struct sockaddr_in dest, gateway, mask; + struct sockaddr_rtlabel label; + struct iovec iov[5]; + int s, len, i, iovcnt = 0; + + /* + * Add a default route via the specified address. + */ + + if ((s = socket(AF_ROUTE, SOCK_RAW, 0)) == -1) + error("Routing Socket open failed: %m"); + + /* Build RTM header */ + + bzero(&rtm, sizeof(rtm)); + + rtm.rtm_version = RTM_VERSION; + rtm.rtm_type = RTM_ADD; + rtm.rtm_tableid = rdomain; + rtm.rtm_priority = 0; + rtm.rtm_msglen = sizeof(rtm); + + iov[iovcnt].iov_base = &rtm; + iov[iovcnt++].iov_len = sizeof(rtm); + + /* Set destination address of all zeros. */ + + bzero(&dest, sizeof(dest)); + + dest.sin_len = sizeof(dest); + dest.sin_family = AF_INET; + + rtm.rtm_addrs |= RTA_DST; + rtm.rtm_msglen += sizeof(dest); + + iov[iovcnt].iov_base = &dest; + iov[iovcnt++].iov_len = sizeof(dest); + + /* + * Set gateway address if and only if non-zero addr supplied. A + * gateway address of 0 implies '-iface'. + */ + + bzero(&gateway, sizeof(gateway)); + if (bcmp(&router, &addr, sizeof(addr)) != 0) { + gateway.sin_len = sizeof(gateway); + gateway.sin_family = AF_INET; + gateway.sin_addr.s_addr = inet_addr(piaddr(router)); + + rtm.rtm_flags |= RTF_GATEWAY | RTF_STATIC; + rtm.rtm_addrs |= RTA_GATEWAY; + rtm.rtm_msglen += sizeof(gateway); + + iov[iovcnt].iov_base = &gateway; + iov[iovcnt++].iov_len = sizeof(gateway); + } + + /* Add netmask of 0. */ + bzero(&mask, sizeof(mask)); + + mask.sin_len = sizeof(mask); + mask.sin_family = AF_INET; + + rtm.rtm_addrs |= RTA_NETMASK; + rtm.rtm_msglen += sizeof(mask); + + iov[iovcnt].iov_base = &mask; + iov[iovcnt++].iov_len = sizeof(mask); + + /* Add our label so we can identify the route as our creation. */ + bzero(&label, sizeof(label)); + label.sr_len = sizeof(label); + label.sr_family = AF_UNSPEC; + len = snprintf(label.sr_label, sizeof(label.sr_label), "DHCLIENT %d", + getpid()); + if (len == -1) + error("writing label for default route: %m"); + if (len >= sizeof(label.sr_label)) + error("label for default route too long (%zd)", + sizeof(label.sr_label)); + + rtm.rtm_addrs |= RTA_LABEL; + rtm.rtm_msglen += sizeof(label); + + iov[iovcnt].iov_base = &label; + iov[iovcnt++].iov_len = sizeof(label); + + /* Check for EEXIST since other dhclient may not be done. */ + for (i = 0; i < 5; i++) { + if (writev(s, iov, iovcnt) != -1) + break; + if (errno != EEXIST) + error("failed to add default route: %m"); + sleep(1); + } + + close(s); +} + +/* + * [priv_]delete_old_address is the equivalent of + * + * ifconfig <ifname> inet <addr> delete + * route -q <rdomain> delete <addr> 127.0.0.1 + */ +void +delete_old_address(char *ifname, int rdomain, struct iaddr addr) +{ + size_t len; + struct imsg_hdr hdr; + struct buf *buf; + + /* Note the address we are deleting for RTM_DELADDR filtering! */ + iaddr_deleting = addr; + + hdr.code = IMSG_DELETE_ADDRESS; + hdr.len = sizeof(hdr) + + sizeof(len) + strlen(ifname) + + sizeof(len) + sizeof(rdomain) + + sizeof(len) + sizeof(addr); + + buf = buf_open(hdr.len); + buf_add(buf, &hdr, sizeof(hdr)); + + len = strlen(ifname); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, ifname, len); + + len = sizeof(rdomain); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &rdomain, len); + + len = sizeof(addr); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &addr, len); + + buf_close(privfd, buf); +} + +void +priv_delete_old_address(char *ifname, int rdomain, struct iaddr addr) +{ + struct ifaliasreq ifaliasreq; + struct rt_msghdr rtm; + struct sockaddr_in dest, gateway; + struct iovec iov[3]; + struct sockaddr_in *in; + int s, iovcnt = 0; + + /* + * Delete specified address on specified interface. + */ + + if ((s = socket(AF_INET, SOCK_STREAM, 0)) == -1) + error("socket open failed: %m"); + + bzero(&ifaliasreq, sizeof(ifaliasreq)); + strncpy(ifaliasreq.ifra_name, ifname, sizeof(ifaliasreq.ifra_name)); + + in = (struct sockaddr_in *) &ifaliasreq.ifra_addr; + in->sin_family = AF_INET; + in->sin_len = sizeof(ifaliasreq.ifra_addr); + + in->sin_addr.s_addr = inet_addr(piaddr(addr)); + + /* SIOCDIFADDR will result in a RTM_DELADDR message we must catch! */ + if (ioctl(s, SIOCDIFADDR, &ifaliasreq) == -1) { + warning("SIOCDIFADDR failed (%s): %m", piaddr(addr)); + close(s); + return; + } + + close(s); + + /* + * Delete the 127.0.0.1 route for the specified address. + */ + + if ((s = socket(AF_ROUTE, SOCK_RAW, 0)) == -1) + error("Routing Socket open failed: %m"); + + /* Build RTM header */ + + bzero(&rtm, sizeof(rtm)); + + rtm.rtm_version = RTM_VERSION; + rtm.rtm_type = RTM_DELETE; + rtm.rtm_tableid = rdomain; + rtm.rtm_priority = 0; + rtm.rtm_msglen = sizeof(rtm); + + iov[iovcnt].iov_base = &rtm; + iov[iovcnt++].iov_len = sizeof(rtm); + + /* Set destination address */ + + bzero(&dest, sizeof(dest)); + + dest.sin_len = sizeof(dest); + dest.sin_family = AF_INET; + dest.sin_addr.s_addr = inet_addr(piaddr(addr)); + + rtm.rtm_addrs |= RTA_DST; + rtm.rtm_msglen += sizeof(dest); + + iov[iovcnt].iov_base = &dest; + iov[iovcnt++].iov_len = sizeof(dest); + + /* Set gateway address */ + + bzero(&gateway, sizeof(gateway)); + + gateway.sin_len = sizeof(gateway); + gateway.sin_family = AF_INET; + gateway.sin_addr.s_addr = inet_addr("127.0.0.1"); + + rtm.rtm_flags |= RTF_GATEWAY; + rtm.rtm_addrs |= RTA_GATEWAY; + rtm.rtm_msglen += sizeof(gateway); + + iov[iovcnt].iov_base = &gateway; + iov[iovcnt++].iov_len = sizeof(gateway); + + /* ESRCH means the route does not exist to delete. */ + if ((writev(s, iov, iovcnt) == -1) && (errno != ESRCH)) + error("failed to delete 127.0.0.1: %m"); +} + +/* + * [priv_]add_new_address is the equivalent of + * + * ifconfig <if> inet <addr> netmask <mask> broadcast <addr> + * route -q <rdomain> add <addr> 127.0.0.1 + */ +void +add_new_address(char *ifname, int rdomain, struct iaddr addr, in_addr_t *mask) +{ + in_addr_t nomask; + struct buf *buf; + size_t len; + struct imsg_hdr hdr; + + iaddr_adding = addr; + + hdr.code = IMSG_ADD_ADDRESS; + hdr.len = sizeof(hdr) + + sizeof(len) + strlen(ifname) + + sizeof(len) + sizeof(rdomain) + + sizeof(len) + sizeof(addr) + + sizeof(len) + sizeof(*mask); + + buf = buf_open(hdr.len); + buf_add(buf, &hdr, sizeof(hdr)); + + len = strlen(ifname); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, ifname, len); + + len = sizeof(rdomain); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &rdomain, len); + + len = sizeof(addr); + buf_add(buf, &len, sizeof(len)); + buf_add(buf, &addr, len); + + len = sizeof(*mask); + buf_add(buf, &len, sizeof(len)); + if (mask) + buf_add(buf, mask, len); + else { + bzero(&nomask, sizeof(nomask)); + buf_add(buf, &nomask, len); + } + + buf_close(privfd, buf); +} + +void +priv_add_new_address(char *ifname, int rdomain, struct iaddr addr, + in_addr_t mask) +{ + struct ifaliasreq ifaliasreq; + struct rt_msghdr rtm; + struct sockaddr_in dest, gateway; + struct sockaddr_rtlabel label; + struct iovec iov[4]; + struct sockaddr_in *in; + int s, len, i, iovcnt = 0; + + /* + * Add specified address on specified interface. + */ + + if ((s = socket(AF_INET, SOCK_STREAM, 0)) == -1) + error("socket open failed: %m"); + + bzero(&ifaliasreq, sizeof(ifaliasreq)); + strncpy(ifaliasreq.ifra_name, ifname, sizeof(ifaliasreq.ifra_name)); + + /* The actual address in ifra_addr. */ + in = (struct sockaddr_in *) &ifaliasreq.ifra_addr; + in->sin_family = AF_INET; + in->sin_len = sizeof(ifaliasreq.ifra_addr); + in->sin_addr.s_addr = inet_addr(piaddr(addr)); + + /* And the netmask in ifra_mask. */ + in = (struct sockaddr_in *) &ifaliasreq.ifra_mask; + in->sin_family = AF_INET; + in->sin_len = sizeof(ifaliasreq.ifra_mask); + in->sin_addr.s_addr = mask; + + /* No need to set broadcast address. Kernel can figure it out. */ + + if (ioctl(s, SIOCAIFADDR, &ifaliasreq) == -1) + warning("SIOCAIFADDR failed (%s): %m", piaddr(addr)); + + close(s); + + /* + * Add the 127.0.0.1 route for the specified address. + */ + + if ((s = socket(AF_ROUTE, SOCK_RAW, 0)) == -1) + error("Routing Socket open failed: %m"); + + /* Build RTM header */ + + bzero(&rtm, sizeof(rtm)); + + rtm.rtm_version = RTM_VERSION; + rtm.rtm_type = RTM_ADD; + rtm.rtm_tableid = rdomain; + rtm.rtm_priority = 0; + rtm.rtm_msglen = sizeof(rtm); + + iov[iovcnt].iov_base = &rtm; + iov[iovcnt++].iov_len = sizeof(rtm); + + /* Set destination address */ + + bzero(&dest, sizeof(dest)); + + dest.sin_len = sizeof(dest); + dest.sin_family = AF_INET; + dest.sin_addr.s_addr = inet_addr(piaddr(addr)); + + rtm.rtm_addrs |= RTA_DST; + rtm.rtm_msglen += sizeof(dest); + + iov[iovcnt].iov_base = &dest; + iov[iovcnt++].iov_len = sizeof(dest); + + /* Set gateway address */ + + bzero(&gateway, sizeof(gateway)); + + gateway.sin_len = sizeof(gateway); + gateway.sin_family = AF_INET; + gateway.sin_addr.s_addr = inet_addr("127.0.0.1"); + + rtm.rtm_flags |= RTF_GATEWAY; + rtm.rtm_addrs |= RTA_GATEWAY; + rtm.rtm_msglen += sizeof(gateway); + + iov[iovcnt].iov_base = &gateway; + iov[iovcnt++].iov_len = sizeof(gateway); + + /* Add our label so we can identify the route as our creation. */ + bzero(&label, sizeof(label)); + + label.sr_len = sizeof(label); + label.sr_family = AF_UNSPEC; + len = snprintf(label.sr_label, sizeof(label.sr_label), "DHCLIENT %d", + getpid()); + if (len == -1) + error("writing label for host route: %m"); + if (len >= sizeof(label.sr_label)) + error("label for host route too long (%zd)", + sizeof(label.sr_label)); + + rtm.rtm_addrs |= RTA_LABEL; + rtm.rtm_msglen += sizeof(label); + + iov[iovcnt].iov_base = &label; + iov[iovcnt++].iov_len = sizeof(label); + + /* Check for EEXIST since other dhclient may not be done. */ + for (i = 0; i < 5; i++) { + if (writev(s, iov, iovcnt) != -1) + break; + if (errno != EEXIST) + error("failed to add 127.0.0.1 route: %m"); + sleep(1); + } + + close(s); +} |