#include <sys/types.h>
#include <sys/ioctl.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <net/if.h>
#include <net/if_dl.h>
#include <net/if_media.h>
#include <netinet/in.h>
#include <errno.h>
#include <ifaddrs.h>
#include <limits.h>
#include <poll.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
#include <time.h>
#include <unistd.h>
#include "dhcp.h"
#include "tree.h"
#include "dhcpd.h"
#include "log.h"
#include "sync.h"
extern int rdomain;
struct interface_info *interfaces;
struct protocol *protocols;
struct dhcpd_timeout *timeouts;
static int interfaces_invalidated;
static int interface_status(struct interface_info *ifinfo);
void
discover_interfaces(void)
{
struct interface_info *tmp;
struct interface_info *last, *next;
struct subnet *subnet;
struct shared_network *share;
struct sockaddr_in foo;
int ir;
struct ifreq *tif;
struct ifaddrs *ifap, *ifa;
if (getifaddrs(&ifap) != 0)
fatalx("getifaddrs failed");
ir = (interfaces != NULL);
for (ifa = ifap; ifa != NULL; ifa = ifa->ifa_next) {
if ((ifa->ifa_flags & IFF_LOOPBACK) ||
(ifa->ifa_flags & IFF_POINTOPOINT) ||
(!(ifa->ifa_flags & IFF_BROADCAST)))
continue;
for (tmp = interfaces; tmp; tmp = tmp->next)
if (!strcmp(tmp->name, ifa->ifa_name))
break;
if (tmp == NULL && ir)
continue;
if (tmp == NULL) {
tmp = calloc(1, sizeof *tmp);
if (!tmp)
fatalx("Insufficient memory to %s %s",
"record interface", ifa->ifa_name);
strlcpy(tmp->name, ifa->ifa_name, sizeof(tmp->name));
tmp->next = interfaces;
tmp->noifmedia = tmp->dead = tmp->errors = 0;
interfaces = tmp;
}
if (ifa->ifa_addr->sa_family == AF_LINK) {
struct if_data *ifi = ifa->ifa_data;
struct sockaddr_dl *sdl;
if (rdomain != ifi->ifi_rdomain)
continue;
sdl = (struct sockaddr_dl *)ifa->ifa_addr;
tmp->index = sdl->sdl_index;
tmp->hw_address.hlen = sdl->sdl_alen;
tmp->hw_address.htype = HTYPE_ETHER;
memcpy(tmp->hw_address.haddr,
LLADDR(sdl), sdl->sdl_alen);
} else if (ifa->ifa_addr->sa_family == AF_INET) {
memcpy(&foo, ifa->ifa_addr, sizeof(foo));
if (foo.sin_addr.s_addr == htonl (INADDR_LOOPBACK))
continue;
if (!tmp->ifp) {
int len = (IFNAMSIZ + ifa->ifa_addr->sa_len);
tif = malloc(len);
if (!tif)
fatalx("no space to remember ifp.");
strlcpy(tif->ifr_name, ifa->ifa_name,
IFNAMSIZ);
memcpy(&tif->ifr_addr, ifa->ifa_addr,
ifa->ifa_addr->sa_len);
tmp->ifp = tif;
tmp->primary_address = foo.sin_addr;
}
}
}
last = NULL;
for (tmp = interfaces; tmp; tmp = next) {
struct iaddr addr;
next = tmp->next;
if (tmp->index == 0) {
log_warnx("Can't listen on %s - wrong rdomain",
tmp->name);
if (!last)
interfaces = interfaces->next;
else
last->next = tmp->next;
continue;
}
if (!tmp->ifp) {
log_warnx("Can't listen on %s - it has no IP address.",
tmp->name);
if (!last)
interfaces = interfaces->next;
else
last->next = tmp->next;
continue;
}
memcpy(&foo, &tmp->ifp->ifr_addr, sizeof tmp->ifp->ifr_addr);
addr.len = 4;
memcpy(addr.iabuf, &foo.sin_addr.s_addr, addr.len);
if ((subnet = find_subnet(addr))) {
if (!subnet->interface) {
subnet->interface = tmp;
subnet->interface_address = addr;
} else if (subnet->interface != tmp) {
log_warnx("Multiple %s %s: %s %s",
"interfaces match the",
"same subnet",
subnet->interface->name,
tmp->name);
}
share = subnet->shared_network;
if (tmp->shared_network &&
tmp->shared_network != share) {
log_warnx("Interface %s matches %s",
tmp->name,
"multiple shared networks");
} else {
tmp->shared_network = share;
}
if (!share->interface) {
share->interface = tmp;
} else if (share->interface != tmp) {
log_warnx("Multiple %s %s: %s %s",
"interfaces match the",
"same shared network",
share->interface->name,
tmp->name);
}
}
if (!tmp->shared_network) {
log_warnx("Can't listen on %s - dhcpd.conf has no "
"subnet declaration for %s.", tmp->name,
inet_ntoa(foo.sin_addr));
if (!last)
interfaces = interfaces->next;
else
last->next = tmp->next;
continue;
}
last = tmp;
for (subnet = (tmp->shared_network ?
tmp->shared_network->subnets : NULL); subnet;
subnet = subnet->next_sibling) {
if (!subnet->interface_address.len) {
subnet->interface_address.len = 4;
memcpy(subnet->interface_address.iabuf,
&foo.sin_addr.s_addr, 4);
}
}
if_register_receive(tmp);
if_register_send(tmp);
log_info("Listening on %s (%s).", tmp->name,
inet_ntoa(foo.sin_addr));
}
if (interfaces == NULL)
fatalx("No interfaces to listen on.");
for (tmp = interfaces; tmp; tmp = tmp->next)
add_protocol(tmp->name, tmp->rfdesc, got_one, tmp);
freeifaddrs(ifap);
}
void
dispatch(void)
{
int nfds, i, to_msec;
struct protocol *l, *next;
static struct pollfd *fds;
static int nfds_max;
time_t howlong;
int nifaces;
for (nfds = 0, l = protocols; l; l = l->next)
nfds++;
if (nfds > nfds_max) {
fds = reallocarray(fds, nfds, sizeof(struct pollfd));
if (fds == NULL)
fatalx("Can't allocate poll structures.");
nfds_max = nfds;
}
for (;;) {
time(&cur_time);
another:
if (timeouts) {
if (timeouts->when <= cur_time) {
struct dhcpd_timeout *t = timeouts;
timeouts = timeouts->next;
t->func(t->what);
free(t);
goto another;
}
howlong = timeouts->when - cur_time;
if (howlong > INT_MAX / 1000)
howlong = INT_MAX / 1000;
to_msec = howlong * 1000;
} else
to_msec = -1;
nifaces = 0;
for (i = 0, l = protocols; l; l = l->next) {
if (l->handler == got_one) {
struct interface_info *ip = l->local;
if (ip->dead) {
l->pfd = -1;
continue;
} else
nifaces++;
}
fds[i].fd = l->fd;
fds[i].events = POLLIN;
l->pfd = i++;
}
if (nifaces == 0)
fatalx("No live interfaces to poll on - exiting.");
switch (poll(fds, nfds, to_msec)) {
case -1:
if (errno != EAGAIN && errno != EINTR)
fatal("poll");
case 0:
continue;
}
time(&cur_time);
for (l = protocols; l; l = next) {
next = l->next;
i = l->pfd;
if (i == -1)
continue;
if (fds[i].revents & (POLLIN | POLLHUP))
l->handler(l);
}
interfaces_invalidated = 0;
}
}
void
got_one(struct protocol *l)
{
struct sockaddr_in from;
struct hardware hfrom;
struct iaddr ifrom;
ssize_t result;
union {
unsigned char packbuf[4095];
struct dhcp_packet packet;
} u;
struct interface_info *ip = l->local;
memset(&u, 0, sizeof(u));
if ((result = receive_packet(ip, u.packbuf, sizeof u,
&from, &hfrom)) == -1) {
log_warn("receive_packet failed on %s", ip->name);
ip->errors++;
if ((!interface_status(ip)) ||
(ip->noifmedia && ip->errors > 20)) {
log_warnx("Interface %s no longer appears valid.",
ip->name);
ip->dead = 1;
interfaces_invalidated = 1;
close(l->fd);
remove_protocol(l);
free(ip);
}
return;
}
if (result == 0)
return;
ifrom.len = 4;
memcpy(ifrom.iabuf, &from.sin_addr, ifrom.len);
do_packet(ip, &u.packet, result, from.sin_port, ifrom, &hfrom);
}
int
interface_status(struct interface_info *ifinfo)
{
char * ifname = ifinfo->name;
int ifsock = ifinfo->rfdesc;
struct ifreq ifr;
struct ifmediareq ifmr;
memset(&ifr, 0, sizeof(ifr));
strlcpy(ifr.ifr_name, ifname, sizeof(ifr.ifr_name));
if (ioctl(ifsock, SIOCGIFFLAGS, &ifr) == -1) {
log_warn("ioctl(SIOCGIFFLAGS) on %s", ifname);
goto inactive;
}
if ((ifr.ifr_flags & (IFF_UP|IFF_RUNNING)) != (IFF_UP|IFF_RUNNING))
goto inactive;
if (ifinfo->noifmedia)
goto active;
memset(&ifmr, 0, sizeof(ifmr));
strlcpy(ifmr.ifm_name, ifname, sizeof(ifmr.ifm_name));
if (ioctl(ifsock, SIOCGIFMEDIA, (caddr_t)&ifmr) == -1) {
if (errno != EINVAL) {
log_debug("ioctl(SIOCGIFMEDIA) on %s", ifname);
ifinfo->noifmedia = 1;
goto active;
}
ifinfo->noifmedia = 1;
goto active;
}
if (ifmr.ifm_status & IFM_AVALID) {
switch (ifmr.ifm_active & IFM_NMASK) {
case IFM_ETHER:
if (ifmr.ifm_status & IFM_ACTIVE)
goto active;
else
goto inactive;
break;
default:
goto inactive;
}
}
inactive:
return (0);
active:
return (1);
}
int
locate_network(struct packet *packet)
{
struct iaddr ia;
if (packet->raw->giaddr.s_addr) {
struct subnet *subnet;
ia.len = 4;
memcpy(ia.iabuf, &packet->raw->giaddr, 4);
subnet = find_subnet(ia);
if (subnet)
packet->shared_network = subnet->shared_network;
else
packet->shared_network = NULL;
} else {
packet->shared_network = packet->interface->shared_network;
}
if (packet->shared_network)
return 1;
return 0;
}
void
add_timeout(time_t when, void (*where)(void *), void *what)
{
struct dhcpd_timeout *t, *q;
t = NULL;
for (q = timeouts; q; q = q->next) {
if (q->func == where && q->what == what) {
if (t)
t->next = q->next;
else
timeouts = q->next;
break;
}
t = q;
}
if (!q) {
q = malloc(sizeof (struct dhcpd_timeout));
if (!q)
fatalx("Can't allocate timeout structure!");
q->func = where;
q->what = what;
}
q->when = when;
if (!timeouts || timeouts->when > q->when) {
q->next = timeouts;
timeouts = q;
return;
}
for (t = timeouts; t->next; t = t->next) {
if (t->next->when > q->when) {
q->next = t->next;
t->next = q;
return;
}
}
t->next = q;
q->next = NULL;
}
void
cancel_timeout(void (*where)(void *), void *what)
{
struct dhcpd_timeout *t, *q;
t = NULL;
for (q = timeouts; q; q = q->next) {
if (q->func == where && q->what == what) {
if (t)
t->next = q->next;
else
timeouts = q->next;
free(q);
return;
}
t = q;
}
}
void
add_protocol(char *name, int fd, void (*handler)(struct protocol *),
void *local)
{
struct protocol *p;
p = malloc(sizeof *p);
if (!p)
fatalx("can't allocate protocol struct for %s", name);
p->fd = fd;
p->handler = handler;
p->local = local;
p->next = protocols;
protocols = p;
}
void
remove_protocol(struct protocol *proto)
{
struct protocol *p, *next, *prev = NULL;
for (p = protocols; p; p = next) {
next = p->next;
if (p == proto) {
if (prev)
prev->next = p->next;
else
protocols = p->next;
free(p);
}
}
}