dhcpcd/dhcpcd.c

1498 lines
36 KiB
C
Raw Normal View History

2013-04-05 04:31:04 +08:00
/*
* dhcpcd - DHCP client daemon
2014-01-04 05:12:19 +08:00
* Copyright (c) 2006-2014 Roy Marples <roy@marples.name>
* All rights reserved
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
2006-11-28 04:23:22 +08:00
*/
const char dhcpcd_copyright[] = "Copyright (c) 2006-2014 Roy Marples";
#include <sys/file.h>
2009-04-17 21:42:44 +08:00
#include <sys/socket.h>
2008-08-13 23:09:13 +08:00
#include <sys/stat.h>
#include <sys/time.h>
#include <sys/types.h>
2009-01-26 23:15:28 +08:00
#include <sys/uio.h>
#include <ctype.h>
2006-11-28 04:23:22 +08:00
#include <errno.h>
#include <getopt.h>
#include <limits.h>
2006-11-28 04:23:22 +08:00
#include <paths.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <syslog.h>
2006-11-28 04:23:22 +08:00
#include <unistd.h>
2008-08-12 20:20:25 +08:00
#include <time.h>
2006-11-28 04:23:22 +08:00
2007-05-11 00:08:49 +08:00
#include "config.h"
#include "arp.h"
#include "common.h"
#include "control.h"
#include "dev.h"
2006-11-28 04:23:22 +08:00
#include "dhcpcd.h"
#include "dhcp6.h"
#include "duid.h"
#include "eloop.h"
#include "if-options.h"
#include "if-pref.h"
#include "ipv4.h"
#include "ipv6.h"
#include "ipv6nd.h"
#include "net.h"
#include "platform.h"
#include "script.h"
2006-11-28 04:23:22 +08:00
2013-02-19 23:43:29 +08:00
struct if_head *ifaces = NULL;
struct if_options *if_options = NULL;
int ifac = 0;
char **ifav = NULL;
int ifdc = 0;
char **ifdv = NULL;
sigset_t dhcpcd_sigset;
const int handle_sigs[] = {
SIGALRM,
SIGHUP,
SIGINT,
SIGPIPE,
SIGTERM,
SIGUSR1,
0
};
2014-02-05 21:18:58 +08:00
static int pidfd = -1;
static char *cffile;
static int linkfd = -1;
static char **ifv;
static int ifc;
static char **margv;
static int margc;
static pid_t
read_pid(const char *pidfile)
2006-11-28 04:23:22 +08:00
{
2007-04-11 21:18:33 +08:00
FILE *fp;
pid_t pid;
2006-11-28 04:23:22 +08:00
if ((fp = fopen(pidfile, "r")) == NULL) {
2007-04-11 21:18:33 +08:00
errno = ENOENT;
return 0;
}
if (fscanf(fp, "%d", &pid) != 1)
pid = 0;
fclose(fp);
return pid;
2006-11-28 04:23:22 +08:00
}
static void
usage(void)
2006-11-28 04:23:22 +08:00
{
printf("usage: "PACKAGE"\t[-46ABbDdEGgHJKkLnpqTVw]\n"
"\t\t[-C, --nohook hook] [-c, --script script]\n"
"\t\t[-e, --env value] [-F, --fqdn FQDN] [-f, --config file]\n"
"\t\t[-h, --hostname hostname] [-I, --clientid clientid]\n"
"\t\t[-i, --vendorclassid vendorclassid] [-l, --leasetime seconds]\n"
"\t\t[-m, --metric metric] [-O, --nooption option]\n"
"\t\t[-o, --option option] [-Q, --require option]\n"
"\t\t[-r, --request address] [-S, --static value]\n"
"\t\t[-s, --inform address[/cidr]] [-t, --timeout seconds]\n"
"\t\t[-u, --userclass class] [-v, --vendor code, value]\n"
"\t\t[-W, --whitelist address[/cidr]] [-y, --reboot seconds]\n"
"\t\t[-X, --blacklist address[/cidr]] [-Z, --denyinterfaces pattern]\n"
"\t\t[-z, --allowinterfaces pattern] [interface] [...]\n"
" "PACKAGE"\t-k, --release [interface]\n"
" "PACKAGE"\t-U, --dumplease interface\n"
2012-08-18 21:58:53 +08:00
" "PACKAGE"\t--version\n"
" "PACKAGE"\t-x, --exit [interface]\n");
}
static void
free_globals(void)
{
struct dhcp_opt *opt;
if (ifac) {
for (ifac--; ifac >= 0; ifac--)
free(ifav[ifac]);
free(ifav);
ifav = NULL;
}
if (ifdc) {
for (ifdc--; ifdc >= 0; ifdc--)
free(ifdv[ifac]);
free(ifdv);
ifdv = NULL;
}
#ifdef INET
if (dhcp_opts) {
for (opt = dhcp_opts; dhcp_opts_len > 0;
opt++, dhcp_opts_len--)
free_dhcp_opt_embenc(opt);
free(dhcp_opts);
dhcp_opts = NULL;
}
#endif
#ifdef INET6
if (dhcp6_opts) {
for (opt = dhcp6_opts; dhcp6_opts_len > 0;
opt++, dhcp6_opts_len--)
free_dhcp_opt_embenc(opt);
free(dhcp6_opts);
dhcp6_opts = NULL;
}
#endif
if (vivso) {
for (opt = vivso; vivso_len > 0; opt++, vivso_len--)
free_dhcp_opt_embenc(opt);
free(vivso);
vivso = NULL;
}
}
2008-11-10 19:15:27 +08:00
/* ARGSUSED */
static void
handle_exit_timeout(__unused void *arg)
{
int timeout;
syslog(LOG_ERR, "timed out");
if (!(options & DHCPCD_IPV4) || !(options & DHCPCD_TIMEOUT_IPV4LL)) {
if (options & DHCPCD_MASTER) {
/* We've timed out, so remove the waitip requirements.
* If the user doesn't like this they can always set
* an infinite timeout. */
options &=
~(DHCPCD_WAITIP | DHCPCD_WAITIP4 | DHCPCD_WAITIP6);
daemonise();
} else
eloop_exit(EXIT_FAILURE);
return;
}
options &= ~DHCPCD_TIMEOUT_IPV4LL;
timeout = (PROBE_NUM * PROBE_MAX) + (PROBE_WAIT * 2);
syslog(LOG_WARNING, "allowing %d seconds for IPv4LL timeout", timeout);
2012-11-13 19:25:51 +08:00
eloop_timeout_add_sec(timeout, handle_exit_timeout, NULL);
}
static inline int
write_pid(int fd, pid_t pid)
{
if (ftruncate(fd, (off_t)0) == -1)
return -1;
lseek(fd, (off_t)0, SEEK_SET);
return dprintf(fd, "%d\n", (int)pid);
}
/* Returns the pid of the child, otherwise 0. */
pid_t
daemonise(void)
{
#ifdef THERE_IS_NO_FORK
errno = ENOSYS;
return 0;
#else
pid_t pid;
char buf = '\0';
int sidpipe[2], fd;
if (options & DHCPCD_DAEMONISE && !(options & DHCPCD_DAEMONISED)) {
if (options & DHCPCD_WAITIP4 &&
!ipv4_addrexists(NULL))
return 0;
if (options & DHCPCD_WAITIP6 &&
!ipv6nd_addrexists(NULL) &&
!dhcp6_addrexists(NULL))
return 0;
if ((options &
(DHCPCD_WAITIP | DHCPCD_WAITIP4 | DHCPCD_WAITIP6)) ==
DHCPCD_WAITIP &&
!ipv4_addrexists(NULL) &&
!ipv6nd_addrexists(NULL) &&
!dhcp6_addrexists(NULL))
return 0;
}
eloop_timeout_delete(handle_exit_timeout, NULL);
if (options & DHCPCD_DAEMONISED || !(options & DHCPCD_DAEMONISE))
return 0;
/* Setup a signal pipe so parent knows when to exit. */
if (pipe(sidpipe) == -1) {
syslog(LOG_ERR, "pipe: %m");
return 0;
}
syslog(LOG_DEBUG, "forking to background");
switch (pid = fork()) {
case -1:
syslog(LOG_ERR, "fork: %m");
return 0;
case 0:
setsid();
/* Notify parent it's safe to exit as we've detached. */
close(sidpipe[0]);
if (write(sidpipe[1], &buf, 1) == -1)
syslog(LOG_ERR, "failed to notify parent: %m");
close(sidpipe[1]);
if ((fd = open(_PATH_DEVNULL, O_RDWR, 0)) != -1) {
dup2(fd, STDIN_FILENO);
dup2(fd, STDOUT_FILENO);
dup2(fd, STDERR_FILENO);
close(fd);
}
break;
default:
/* Wait for child to detach */
close(sidpipe[1]);
if (read(sidpipe[0], &buf, 1) == -1)
syslog(LOG_ERR, "failed to read child: %m");
close(sidpipe[0]);
break;
}
/* Done with the fd now */
if (pid != 0) {
2014-02-04 23:46:38 +08:00
syslog(LOG_INFO, "forked to background, child pid %d", pid);
write_pid(pidfd, pid);
close(pidfd);
pidfd = -1;
options |= DHCPCD_FORKED;
eloop_exit(EXIT_SUCCESS);
return pid;
}
options |= DHCPCD_DAEMONISED;
return pid;
#endif
}
struct interface *
find_interface(const char *ifname)
{
struct interface *ifp;
TAILQ_FOREACH(ifp, ifaces, next) {
if (strcmp(ifp->name, ifname) == 0)
return ifp;
}
return NULL;
}
static void
stop_interface(struct interface *ifp)
{
syslog(LOG_INFO, "%s: removing interface", ifp->name);
ifp->options->options |= DHCPCD_STOPPING;
// Remove the interface from our list
TAILQ_REMOVE(ifaces, ifp, next);
dhcp6_drop(ifp, NULL);
ipv6nd_drop(ifp);
2013-04-05 07:57:12 +08:00
dhcp_drop(ifp, "STOP");
eloop_timeout_delete(NULL, ifp);
if (ifp->options->options & DHCPCD_DEPARTED)
script_runreason(ifp, "DEPARTED");
free_interface(ifp);
if (!(options & (DHCPCD_MASTER | DHCPCD_TEST)))
eloop_exit(EXIT_FAILURE);
}
static void
configure_interface1(struct interface *ifp)
{
struct if_options *ifo = ifp->options;
int ra_global, ra_iface;
/* Do any platform specific configuration */
if_conf(ifp);
/* If we want to release a lease, we can't really persist the
* address either. */
if (ifo->options & DHCPCD_RELEASE)
ifo->options &= ~DHCPCD_PERSISTENT;
if (ifp->flags & IFF_POINTOPOINT && !(ifo->options & DHCPCD_INFORM))
ifo->options |= DHCPCD_STATIC;
if (ifp->flags & IFF_NOARP ||
2009-03-24 06:02:37 +08:00
ifo->options & (DHCPCD_INFORM | DHCPCD_STATIC))
ifo->options &= ~(DHCPCD_ARP | DHCPCD_IPV4LL);
if (!(ifp->flags & (IFF_POINTOPOINT | IFF_LOOPBACK | IFF_MULTICAST)))
ifo->options &= ~DHCPCD_IPV6RS;
if (ifo->options & DHCPCD_LINK && carrier_status(ifp) == LINK_UNKNOWN)
ifo->options &= ~DHCPCD_LINK;
if (ifo->metric != -1)
ifp->metric = ifo->metric;
if (!(ifo->options & DHCPCD_IPV6))
ifo->options &= ~DHCPCD_IPV6RS;
/* We want to disable kernel interface RA as early as possible. */
if (ifo->options & DHCPCD_IPV6RS) {
2013-07-02 03:34:54 +08:00
ra_global = check_ipv6(NULL, options & DHCPCD_IPV6RA_OWN ? 1:0);
ra_iface = check_ipv6(ifp->name,
2013-07-02 03:34:54 +08:00
ifp->options->options & DHCPCD_IPV6RA_OWN ? 1 : 0);
if (ra_global == -1 || ra_iface == -1)
ifo->options &= ~DHCPCD_IPV6RS;
else if (ra_iface == 0)
ifo->options |= DHCPCD_IPV6RA_OWN;
}
/* If we haven't specified a ClientID and our hardware address
* length is greater than DHCP_CHADDR_LEN then we enforce a ClientID
* of the hardware address family and the hardware address. */
if (ifp->hwlen > DHCP_CHADDR_LEN)
ifo->options |= DHCPCD_CLIENTID;
/* Firewire and InfiniBand interfaces require ClientID and
* the broadcast option being set. */
switch (ifp->family) {
case ARPHRD_IEEE1394: /* FALLTHROUGH */
case ARPHRD_INFINIBAND:
ifo->options |= DHCPCD_CLIENTID | DHCPCD_BROADCAST;
break;
}
if (!(ifo->options & DHCPCD_IAID)) {
/*
* An IAID is for identifying a unqiue interface within
* the client. It is 4 bytes long. Working out a default
* value is problematic.
*
* Interface name and number are not stable
* between different OS's. Some OS's also cannot make
* up their mind what the interface should be called
* (yes, udev, I'm looking at you).
* Also, the name could be longer than 4 bytes.
* Also, with pluggable interfaces the name and index
* could easily get swapped per actual interface.
*
* The MAC address is 6 bytes long, the final 3
* being unique to the manufacturer and the initial 3
* being unique to the organisation which makes it.
* We could use the last 4 bytes of the MAC address
* as the IAID as it's the most stable part given the
* above, but equally it's not guaranteed to be
* unique.
*
* Given the above, and our need to reliably work
* between reboots without persitent storage,
* generating the IAID from the MAC address is the only
* logical default.
*
* dhclient uses the last 4 bytes of the MAC address.
* dibbler uses an increamenting counter.
* wide-dhcpv6 uses 0 or a configured value.
* odhcp6c uses 1.
* Windows 7 uses the first 3 bytes of the MAC address
* and an unknown byte.
* dhcpcd-6.1.0 and earlier used the interface name,
* falling back to interface index if name > 4.
*/
memcpy(ifo->iaid, ifp->hwaddr + ifp->hwlen - sizeof(ifo->iaid),
sizeof(ifo->iaid));
#if 0
len = strlen(ifp->name);
if (len <= sizeof(ifo->iaid)) {
memcpy(ifo->iaid, ifp->name, len);
memset(ifo->iaid + len, 0, sizeof(ifo->iaid) - len);
} else {
/* IAID is the same size as a uint32_t */
len = htonl(ifp->index);
memcpy(ifo->iaid, &len, sizeof(len));
}
#endif
ifo->options |= DHCPCD_IAID;
}
#ifdef INET6
if (ifo->ia == NULL && ifo->options & DHCPCD_IPV6) {
ifo->ia = malloc(sizeof(*ifo->ia));
if (ifo->ia == NULL)
syslog(LOG_ERR, "%s: %m", __func__);
else {
if (ifo->ia_type == 0)
ifo->ia_type = D6_OPTION_IA_NA;
memcpy(ifo->ia->iaid, ifo->iaid, sizeof(ifo->iaid));
ifo->ia_len = 1;
ifo->ia->sla = NULL;
ifo->ia->sla_len = 0;
}
}
#endif
/* If we are not sending an authentication option, don't require it */
if (!(ifo->auth.options & DHCPCD_AUTH_SEND))
ifo->auth.options &= ~DHCPCD_AUTH_REQUIRE;
}
int
select_profile(struct interface *ifp, const char *profile)
{
struct if_options *ifo;
2010-11-09 08:34:38 +08:00
int ret;
2010-11-09 08:34:38 +08:00
ret = 0;
ifo = read_config(cffile, ifp->name, ifp->ssid, profile);
if (ifo == NULL) {
syslog(LOG_DEBUG, "%s: no profile %s", ifp->name, profile);
2010-11-09 08:34:38 +08:00
ret = -1;
goto exit;
}
if (profile != NULL) {
strlcpy(ifp->profile, profile, sizeof(ifp->profile));
syslog(LOG_INFO, "%s: selected profile %s",
ifp->name, profile);
} else
*ifp->profile = '\0';
free_options(ifp->options);
ifp->options = ifo;
2010-11-09 08:34:38 +08:00
exit:
if (profile)
configure_interface1(ifp);
2010-11-09 08:34:38 +08:00
return ret;
}
static void
configure_interface(struct interface *ifp, int argc, char **argv)
{
select_profile(ifp, NULL);
add_options(ifp->name, ifp->options, argc, argv);
configure_interface1(ifp);
}
void
handle_carrier(int carrier, int flags, const char *ifname)
{
struct interface *ifp;
ifp = find_interface(ifname);
if (ifp == NULL || !(ifp->options->options & DHCPCD_LINK))
return;
if (carrier == LINK_UNKNOWN)
carrier = carrier_status(ifp); /* will set ifp->flags */
else
ifp->flags = flags;
if (carrier == LINK_UNKNOWN)
syslog(LOG_ERR, "%s: carrier_status: %m", ifname);
/* IFF_RUNNING is checked, if needed, earlier and is OS dependant */
else if (carrier == LINK_DOWN || (ifp->flags & IFF_UP) == 0) {
if (ifp->carrier != LINK_DOWN) {
if (ifp->carrier == LINK_UP)
syslog(LOG_INFO, "%s: carrier lost", ifp->name);
ifp->carrier = LINK_DOWN;
dhcp6_drop(ifp, "EXPIRE6");
ipv6nd_drop(ifp);
/* Don't blindly delete our knowledge of LL addresses.
* We need to listen to what the kernel does with
* them as some OS's will remove, mark tentative or
* do nothing. */
ipv6_free_ll_callbacks(ifp);
dhcp_drop(ifp, "NOCARRIER");
}
} else if (carrier == LINK_UP && ifp->flags & IFF_UP) {
if (ifp->carrier != LINK_UP) {
syslog(LOG_INFO, "%s: carrier acquired", ifp->name);
ifp->carrier = LINK_UP;
#if !defined(__linux__) && !defined(__NetBSD__)
/* BSD does not emit RTM_NEWADDR or RTM_CHGADDR when the
* hardware address changes so we have to go
* through the disovery process to work it out. */
handle_interface(0, ifp->name);
#endif
if (ifp->wireless)
getifssid(ifp->name, ifp->ssid);
configure_interface(ifp, margc, margv);
script_runreason(ifp, "CARRIER");
start_interface(ifp);
}
}
}
static void
warn_iaid_conflict(struct interface *ifp, uint8_t *iaid)
{
struct interface *ifn;
size_t i;
TAILQ_FOREACH(ifn, ifaces, next) {
if (ifn == ifp)
continue;
if (memcmp(ifn->options->iaid, iaid,
sizeof(ifn->options->iaid)) == 0)
break;
for (i = 0; i < ifn->options->ia_len; i++) {
if (memcmp(&ifn->options->ia[i].iaid, iaid,
sizeof(ifn->options->ia[i].iaid)) == 0)
break;
}
}
/* This is only a problem if the interfaces are on the same network. */
if (ifn)
syslog(LOG_ERR,
"%s: IAID conflicts with one assigned to %s",
ifp->name, ifn->name);
}
void
start_interface(void *arg)
2006-11-28 04:23:22 +08:00
{
struct interface *ifp = arg;
struct if_options *ifo = ifp->options;
int nolease;
size_t i;
handle_carrier(LINK_UNKNOWN, 0, ifp->name);
if (ifp->carrier == LINK_DOWN) {
syslog(LOG_INFO, "%s: waiting for carrier", ifp->name);
return;
}
if (ifo->options & (DHCPCD_DUID | DHCPCD_IPV6)) {
/* Report client DUID */
if (duid == NULL) {
if (duid_init(ifp) == 0)
return;
syslog(LOG_INFO, "DUID %s",
hwaddr_ntoa(duid, duid_len));
}
/* Report IAIDs */
syslog(LOG_INFO, "%s: IAID %s", ifp->name,
hwaddr_ntoa(ifo->iaid, sizeof(ifo->iaid)));
warn_iaid_conflict(ifp, ifo->iaid);
for (i = 0; i < ifo->ia_len; i++) {
if (memcmp(ifo->iaid, ifo->ia[i].iaid,
sizeof(ifo->iaid)))
{
syslog(LOG_INFO, "%s: IAID %s", ifp->name,
hwaddr_ntoa(ifo->ia[i].iaid,
sizeof(ifo->ia[i].iaid)));
warn_iaid_conflict(ifp, ifo->ia[i].iaid);
}
}
}
if (ifo->options & DHCPCD_IPV6) {
2013-04-05 07:25:31 +08:00
if (ifo->options & DHCPCD_IPV6RS &&
!(ifo->options & DHCPCD_INFORM))
ipv6nd_startrs(ifp);
2013-04-05 07:25:31 +08:00
2013-04-02 15:01:11 +08:00
if (!(ifo->options & DHCPCD_IPV6RS)) {
if (ifo->options & DHCPCD_IA_FORCED)
nolease = dhcp6_start(ifp, DH6S_INIT);
else {
nolease = dhcp6_find_delegates(ifp);
/* Enabling the below doesn't really make
* sense as there is currently no standard
* to push routes via DHCPv6.
* (There is an expired working draft,
* maybe abandoned?)
* You can also get it to work by forcing
* an IA as shown above. */
#if 0
/* With no RS or delegates we might
* as well try and solicit a DHCPv6 address */
if (nolease == 0)
nolease = dhcp6_start(ifp, DH6S_INIT);
#endif
}
2013-04-02 15:01:11 +08:00
if (nolease == -1)
syslog(LOG_ERR,
"%s: dhcp6_start: %m", ifp->name);
}
}
if (ifo->options & DHCPCD_IPV4)
dhcp_start(ifp);
}
2007-04-11 21:18:33 +08:00
/* ARGSUSED */
static void
handle_link(__unused void *arg)
{
if (manage_link(linkfd) == -1 && errno != ENXIO && errno != ENODEV)
syslog(LOG_ERR, "manage_link: %m");
}
static void
init_state(struct interface *ifp, int argc, char **argv)
{
struct if_options *ifo;
const char *reason = NULL;
configure_interface(ifp, argc, argv);
ifo = ifp->options;
2013-02-19 23:23:53 +08:00
if (ifo->options & DHCPCD_IPV4 && ipv4_init() == -1) {
syslog(LOG_ERR, "ipv4_init: %m");
ifo->options &= ~DHCPCD_IPV4;
}
if (ifo->options & DHCPCD_IPV6 && ipv6_init() == -1) {
syslog(LOG_ERR, "ipv6_init: %m");
2013-02-19 23:23:53 +08:00
ifo->options &= ~DHCPCD_IPV6RS;
}
if (!(options & DHCPCD_TEST))
script_runreason(ifp, "PREINIT");
if (ifo->options & DHCPCD_LINK) {
switch (carrier_status(ifp)) {
2013-09-05 21:27:52 +08:00
case LINK_DOWN:
ifp->carrier = LINK_DOWN;
reason = "NOCARRIER";
break;
2013-09-05 21:27:52 +08:00
case LINK_UP:
ifp->carrier = LINK_UP;
reason = "CARRIER";
break;
default:
ifp->carrier = LINK_UNKNOWN;
return;
}
if (reason && !(options & DHCPCD_TEST))
script_runreason(ifp, reason);
} else
ifp->carrier = LINK_UNKNOWN;
}
void
handle_interface(int action, const char *ifname)
{
struct if_head *ifs;
struct interface *ifp, *ifn, *ifl = NULL;
const char * const argv[] = { ifname };
int i;
if (action == -1) {
ifp = find_interface(ifname);
if (ifp != NULL) {
ifp->options->options |= DHCPCD_DEPARTED;
stop_interface(ifp);
}
return;
}
/* If running off an interface list, check it's in it. */
if (ifc) {
for (i = 0; i < ifc; i++)
if (strcmp(ifv[i], ifname) == 0)
break;
if (i >= ifc)
return;
}
ifs = discover_interfaces(-1, UNCONST(argv));
TAILQ_FOREACH_SAFE(ifp, ifs, next, ifn) {
if (strcmp(ifp->name, ifname) != 0)
continue;
/* Check if we already have the interface */
ifl = find_interface(ifp->name);
if (ifl) {
/* The flags and hwaddr could have changed */
ifl->flags = ifp->flags;
ifl->hwlen = ifp->hwlen;
if (ifp->hwlen != 0)
memcpy(ifl->hwaddr, ifp->hwaddr, ifl->hwlen);
} else {
TAILQ_REMOVE(ifs, ifp, next);
TAILQ_INSERT_TAIL(ifaces, ifp, next);
}
if (action == 1) {
init_state(ifp, margc, margv);
start_interface(ifp);
}
}
/* Free our discovered list */
while ((ifp = TAILQ_FIRST(ifs))) {
TAILQ_REMOVE(ifs, ifp, next);
free_interface(ifp);
}
free(ifs);
}
void
handle_hwaddr(const char *ifname, const uint8_t *hwaddr, size_t hwlen)
{
struct interface *ifp;
ifp = find_interface(ifname);
if (ifp == NULL)
return;
if (hwlen > sizeof(ifp->hwaddr)) {
errno = ENOBUFS;
syslog(LOG_ERR, "%s: %s: %m", ifp->name, __func__);
return;
}
if (ifp->hwlen == hwlen && memcmp(ifp->hwaddr, hwaddr, hwlen) == 0)
return;
syslog(LOG_INFO, "%s: new hardware address: %s", ifp->name,
hwaddr_ntoa(hwaddr, hwlen));
ifp->hwlen = hwlen;
memcpy(ifp->hwaddr, hwaddr, hwlen);
}
static void
if_reboot(struct interface *ifp, int argc, char **argv)
{
int oldopts;
oldopts = ifp->options->options;
script_runreason(ifp, "RECONFIGURE");
configure_interface(ifp, argc, argv);
2013-03-26 18:42:30 +08:00
dhcp_reboot_newopts(ifp, oldopts);
dhcp6_reboot(ifp);
start_interface(ifp);
}
static void
reconf_reboot(int action, int argc, char **argv, int oi)
{
struct if_head *ifs;
struct interface *ifn, *ifp;
ifs = discover_interfaces(argc - oi, argv + oi);
if (ifs == NULL)
return;
while ((ifp = TAILQ_FIRST(ifs))) {
TAILQ_REMOVE(ifs, ifp, next);
ifn = find_interface(ifp->name);
if (ifn) {
if (action)
2010-06-10 18:38:37 +08:00
if_reboot(ifn, argc, argv);
else
ipv4_applyaddr(ifn);
free_interface(ifp);
} else {
init_state(ifp, argc, argv);
TAILQ_INSERT_TAIL(ifaces, ifp, next);
start_interface(ifp);
}
}
free(ifs);
sort_interfaces();
}
struct dhcpcd_siginfo {
int signo;
pid_t pid;
} dhcpcd_siginfo;
static void
handle_signal1(void *arg)
{
struct dhcpcd_siginfo *si;
2012-04-01 15:38:52 +08:00
struct interface *ifp;
struct if_options *ifo;
int do_release;
si = arg;
2012-04-01 15:38:52 +08:00
do_release = 0;
switch (si->signo) {
case SIGINT:
syslog(LOG_INFO, "received SIGINT from PID %d, stopping",
(int)si->pid);
break;
case SIGTERM:
syslog(LOG_INFO, "received SIGTERM from PID %d, stopping",
(int)si->pid);
break;
case SIGALRM:
syslog(LOG_INFO, "received SIGALRM from PID %d, rebinding",
(int)si->pid);
free_globals();
ifav = NULL;
ifac = 0;
ifdc = 0;
ifdv = NULL;
ifo = read_config(cffile, NULL, NULL, NULL);
add_options(NULL, ifo, margc, margv);
/* We need to preserve these two options. */
if (options & DHCPCD_MASTER)
ifo->options |= DHCPCD_MASTER;
if (options & DHCPCD_DAEMONISED)
ifo->options |= DHCPCD_DAEMONISED;
options = ifo->options;
free_options(ifo);
reconf_reboot(1, ifc, ifv, 0);
return;
case SIGHUP:
syslog(LOG_INFO, "received SIGHUP from PID %d, releasing",
(int)si->pid);
do_release = 1;
break;
case SIGUSR1:
syslog(LOG_INFO, "received SIGUSR from PID %d, reconfiguring",
(int)si->pid);
TAILQ_FOREACH(ifp, ifaces, next) {
ipv4_applyaddr(ifp);
}
2009-03-20 18:33:09 +08:00
return;
case SIGPIPE:
syslog(LOG_WARNING, "received SIGPIPE");
return;
default:
syslog(LOG_ERR,
"received signal %d from PID %d, "
"but don't know what to do with it",
si->signo, (int)si->pid);
return;
}
if (!(options & DHCPCD_TEST)) {
/* drop_dhcp could change the order, so we do it like this. */
for (;;) {
/* Be sane and drop the last config first */
ifp = TAILQ_LAST(ifaces, if_head);
if (ifp == NULL)
break;
if (do_release) {
ifp->options->options |= DHCPCD_RELEASE;
ifp->options->options &= ~DHCPCD_PERSISTENT;
}
ifp->options->options |= DHCPCD_EXITING;
stop_interface(ifp);
}
}
eloop_exit(EXIT_FAILURE);
}
static void
handle_signal(__unused int sig, siginfo_t *siginfo, __unused void *context)
{
/* So that we can operate safely under a signal we instruct
* eloop to pass a copy of the siginfo structure to handle_signal1
* as the very first thing to do. */
dhcpcd_siginfo.signo = siginfo->si_signo;
dhcpcd_siginfo.pid = siginfo->si_pid;
eloop_timeout_add_now(handle_signal1, &dhcpcd_siginfo);
}
int
handle_args(struct fd_list *fd, int argc, char **argv)
{
struct interface *ifp;
2012-04-01 15:38:52 +08:00
int do_exit = 0, do_release = 0, do_reboot = 0;
int opt, oi = 0;
ssize_t len;
2009-02-06 05:06:57 +08:00
size_t l;
struct iovec iov[2];
2009-02-06 05:06:57 +08:00
char *tmp, *p;
if (fd != NULL) {
/* Special commands for our control socket */
if (strcmp(*argv, "--version") == 0) {
len = strlen(VERSION) + 1;
iov[0].iov_base = &len;
iov[0].iov_len = sizeof(ssize_t);
iov[1].iov_base = UNCONST(VERSION);
iov[1].iov_len = len;
if (writev(fd->fd, iov, 2) == -1) {
syslog(LOG_ERR, "writev: %m");
return -1;
}
return 0;
} else if (strcmp(*argv, "--getconfigfile") == 0) {
len = strlen(cffile ? cffile : CONFIG) + 1;
iov[0].iov_base = &len;
iov[0].iov_len = sizeof(ssize_t);
iov[1].iov_base = cffile ? cffile : UNCONST(CONFIG);
iov[1].iov_len = len;
if (writev(fd->fd, iov, 2) == -1) {
syslog(LOG_ERR, "writev: %m");
return -1;
}
return 0;
} else if (strcmp(*argv, "--getinterfaces") == 0) {
len = 0;
2009-01-17 08:43:08 +08:00
if (argc == 1) {
TAILQ_FOREACH(ifp, ifaces, next) {
len++;
if (D6_STATE_RUNNING(ifp))
len++;
if (ipv6nd_has_ra(ifp))
len++;
}
len = write(fd->fd, &len, sizeof(len));
if (len != sizeof(len))
return -1;
TAILQ_FOREACH(ifp, ifaces, next) {
send_interface(fd->fd, ifp);
}
return 0;
}
opt = 0;
while (argv[++opt] != NULL) {
TAILQ_FOREACH(ifp, ifaces, next) {
if (strcmp(argv[opt], ifp->name) == 0) {
len++;
if (D6_STATE_RUNNING(ifp))
len++;
if (ipv6nd_has_ra(ifp))
len++;
}
}
}
len = write(fd->fd, &len, sizeof(len));
if (len != sizeof(len))
return -1;
opt = 0;
while (argv[++opt] != NULL) {
TAILQ_FOREACH(ifp, ifaces, next) {
if (strcmp(argv[opt], ifp->name) == 0)
send_interface(fd->fd, ifp);
}
}
return 0;
} else if (strcmp(*argv, "--listen") == 0) {
fd->listener = 1;
return 0;
}
}
2009-02-06 05:06:57 +08:00
/* Log the command */
len = 0;
for (opt = 0; opt < argc; opt++)
len += strlen(argv[opt]) + 1;
tmp = p = malloc(len + 1);
if (tmp == NULL) {
syslog(LOG_ERR, "%s: %m", __func__);
return -1;
}
2009-02-06 05:06:57 +08:00
for (opt = 0; opt < argc; opt++) {
l = strlen(argv[opt]);
strlcpy(p, argv[opt], l + 1);
p += l;
*p++ = ' ';
}
*--p = '\0';
syslog(LOG_INFO, "control command: %s", tmp);
free(tmp);
optind = 0;
while ((opt = getopt_long(argc, argv, IF_OPTS, cf_options, &oi)) != -1)
{
switch (opt) {
case 'g':
2012-04-01 15:38:52 +08:00
/* Assumed if below not set */
break;
case 'k':
do_release = 1;
break;
case 'n':
do_reboot = 1;
break;
case 'x':
do_exit = 1;
break;
}
}
2009-01-05 23:50:14 +08:00
/* We need at least one interface */
if (optind == argc) {
syslog(LOG_ERR, "%s: no interface", __func__);
return -1;
}
if (do_release || do_exit) {
2008-09-04 16:27:38 +08:00
for (oi = optind; oi < argc; oi++) {
if ((ifp = find_interface(argv[oi])) == NULL)
2008-09-04 16:27:38 +08:00
continue;
if (do_release) {
ifp->options->options |= DHCPCD_RELEASE;
ifp->options->options &= ~DHCPCD_PERSISTENT;
}
2013-08-25 18:22:48 +08:00
ifp->options->options |= DHCPCD_EXITING;
stop_interface(ifp);
}
return 0;
}
reconf_reboot(do_reboot, argc, argv, optind);
return 0;
}
static int
signal_init(void (*func)(int, siginfo_t *, void *), sigset_t *oldset)
{
unsigned int i;
struct sigaction sa;
sigset_t newset;
sigfillset(&newset);
if (sigprocmask(SIG_SETMASK, &newset, oldset) == -1)
return -1;
memset(&sa, 0, sizeof(sa));
sa.sa_sigaction = func;
sa.sa_flags = SA_SIGINFO;
sigemptyset(&sa.sa_mask);
for (i = 0; handle_sigs[i]; i++) {
if (sigaction(handle_sigs[i], &sa, NULL) == -1)
return -1;
}
return 0;
}
int
main(int argc, char **argv)
{
2014-02-08 08:29:02 +08:00
char *pidfile;
struct interface *ifp;
2012-10-12 19:26:20 +08:00
uint16_t family = 0;
int opt, oi = 0, sig = 0, i;
2008-09-05 22:16:53 +08:00
size_t len;
pid_t pid;
struct timespec ts;
2014-02-05 20:01:09 +08:00
struct control_ctx control_ctx;
pidfile = NULL;
closefrom(3);
openlog(PACKAGE, LOG_PERROR | LOG_PID, LOG_DAEMON);
setlogmask(LOG_UPTO(LOG_INFO));
/* Test for --help and --version */
if (argc > 1) {
if (strcmp(argv[1], "--help") == 0) {
usage();
return EXIT_SUCCESS;
} else if (strcmp(argv[1], "--version") == 0) {
printf(""PACKAGE" "VERSION"\n%s\n", dhcpcd_copyright);
return EXIT_SUCCESS;
}
}
control_ctx.fd = -1;
i = 0;
while ((opt = getopt_long(argc, argv, IF_OPTS, cf_options, &oi)) != -1)
{
2007-05-14 21:34:36 +08:00
switch (opt) {
case '4':
family = AF_INET;
break;
case '6':
family = AF_INET6;
break;
case 'f':
cffile = optarg;
break;
case 'g':
sig = SIGUSR1;
break;
case 'k':
sig = SIGHUP;
break;
case 'n':
sig = SIGALRM;
break;
case 'x':
sig = SIGTERM;
break;
case 'T':
i = 1;
break;
case 'U':
i = 2;
break;
case 'V':
i = 3;
break;
case '?':
usage();
goto exit_failure;
2007-04-11 21:18:33 +08:00
}
}
margv = argv;
margc = argc;
if_options = read_config(cffile, NULL, NULL, NULL);
opt = add_options(NULL, if_options, argc, argv);
if (opt != 1) {
if (opt == 0)
usage();
goto exit_failure;
}
if (i == 3) {
printf("Interface options:\n");
if_printoptions();
#ifdef INET
if (family == 0 || family == AF_INET) {
printf("\nDHCPv4 options:\n");
dhcp_printoptions();
}
#endif
#ifdef INET6
if (family == 0 || family == AF_INET6) {
printf("\nDHCPv6 options:\n");
dhcp6_printoptions();
}
#endif
goto exit_success;
}
options = if_options->options;
if (i != 0) {
if (i == 1)
options |= DHCPCD_TEST;
else
options |= DHCPCD_DUMPLEASE;
options |= DHCPCD_PERSISTENT;
options &= ~DHCPCD_DAEMONISE;
}
#ifdef THERE_IS_NO_FORK
options &= ~DHCPCD_DAEMONISE;
#endif
2009-02-24 07:52:21 +08:00
if (options & DHCPCD_DEBUG)
setlogmask(LOG_UPTO(LOG_DEBUG));
2013-05-24 02:58:28 +08:00
if (options & DHCPCD_QUIET) {
i = open(_PATH_DEVNULL, O_RDWR);
if (i == -1)
syslog(LOG_ERR, "%s: open: %m", __func__);
else {
dup2(i, STDERR_FILENO);
close(i);
}
}
if (!(options & (DHCPCD_TEST | DHCPCD_DUMPLEASE))) {
/* If we have any other args, we should run as a single dhcpcd
* instance for that interface. */
len = strlen(PIDFILE) + IF_NAMESIZE + 2;
pidfile = malloc(len);
if (pidfile == NULL) {
syslog(LOG_ERR, "%s: %m", __func__);
goto exit_failure;
}
if (optind == argc - 1)
snprintf(pidfile, len, PIDFILE, "-", argv[optind]);
else {
snprintf(pidfile, len, PIDFILE, "", "");
options |= DHCPCD_MASTER;
}
}
2007-04-11 21:18:33 +08:00
if (chdir("/") == -1)
2009-01-29 21:18:57 +08:00
syslog(LOG_ERR, "chdir `/': %m");
if (options & DHCPCD_DUMPLEASE) {
if (optind != argc - 1) {
syslog(LOG_ERR, "dumplease requires an interface");
goto exit_failure;
}
if (dhcp_dump(argv[optind]) == -1)
goto exit_failure;
goto exit_success;
}
2010-08-24 21:35:15 +08:00
if (!(options & (DHCPCD_MASTER | DHCPCD_TEST))) {
2014-02-05 20:01:09 +08:00
if ((i = control_open(&control_ctx)) != -1) {
2009-02-12 01:56:22 +08:00
syslog(LOG_INFO,
"sending commands to master dhcpcd process");
2014-02-05 20:01:09 +08:00
len = control_send(&control_ctx, argc, argv);
close(i);
if (len > 0) {
syslog(LOG_DEBUG, "send OK");
goto exit_success;
} else {
syslog(LOG_ERR, "failed to send commands");
goto exit_failure;
}
} else {
if (errno != ENOENT)
2012-11-13 18:17:19 +08:00
syslog(LOG_ERR, "control_open: %m");
}
}
if (geteuid())
2009-02-12 01:56:22 +08:00
syslog(LOG_WARNING,
PACKAGE " will not work correctly unless run as root");
if (sig != 0) {
pid = read_pid(pidfile);
if (pid != 0)
syslog(LOG_INFO, "sending signal %d to pid %d",
2009-02-12 01:56:22 +08:00
sig, pid);
if (pid == 0 || kill(pid, sig) != 0) {
if (sig != SIGALRM && errno != EPERM)
syslog(LOG_ERR, ""PACKAGE" not running");
if (pid != 0 && errno != ESRCH) {
syslog(LOG_ERR, "kill: %m");
goto exit_failure;
}
unlink(pidfile);
if (sig != SIGALRM)
goto exit_failure;
} else {
if (sig == SIGALRM || sig == SIGUSR1)
goto exit_success;
/* Spin until it exits */
syslog(LOG_INFO, "waiting for pid %d to exit", pid);
ts.tv_sec = 0;
ts.tv_nsec = 100000000; /* 10th of a second */
for(i = 0; i < 100; i++) {
nanosleep(&ts, NULL);
if (read_pid(pidfile) == 0)
goto exit_success;
}
syslog(LOG_ERR, "pid %d failed to exit", pid);
goto exit_failure;
}
}
2007-04-11 21:18:33 +08:00
if (!(options & DHCPCD_TEST)) {
if ((pid = read_pid(pidfile)) > 0 &&
kill(pid, 0) == 0)
2008-01-17 00:38:47 +08:00
{
syslog(LOG_ERR, ""PACKAGE
2009-02-12 01:56:22 +08:00
" already running on pid %d (%s)",
pid, pidfile);
goto exit_failure;
}
/* Ensure we have the needed directories */
if (mkdir(RUNDIR, 0755) == -1 && errno != EEXIST)
syslog(LOG_ERR, "mkdir `%s': %m", RUNDIR);
if (mkdir(DBDIR, 0755) == -1 && errno != EEXIST)
syslog(LOG_ERR, "mkdir `%s': %m", DBDIR);
pidfd = open(pidfile,
O_WRONLY | O_CREAT | O_CLOEXEC | O_NONBLOCK,
0664);
if (pidfd == -1)
syslog(LOG_ERR, "open `%s': %m", pidfile);
else {
/* Lock the file so that only one instance of dhcpcd
* runs on an interface */
if (flock(pidfd, LOCK_EX | LOCK_NB) == -1) {
syslog(LOG_ERR, "flock `%s': %m", pidfile);
close(pidfd);
pidfd = -1;
goto exit_failure;
}
write_pid(pidfd, getpid());
}
}
syslog(LOG_INFO, "version " VERSION " starting");
options |= DHCPCD_STARTED;
/* Save signal mask, block and redirect signals to our handler */
if (signal_init(handle_signal, &dhcpcd_sigset) == -1) {
syslog(LOG_ERR, "signal_setup: %m");
goto exit_failure;
}
if (options & DHCPCD_MASTER) {
2014-02-05 20:01:09 +08:00
if (control_start(&control_ctx) == -1)
2012-11-13 18:17:19 +08:00
syslog(LOG_ERR, "control_start: %m");
}
#if 0
if (options & DHCPCD_IPV6RS && disable_rtadv() == -1) {
syslog(LOG_ERR, "disable_rtadvd: %m");
options &= ~DHCPCD_IPV6RS;
}
#endif
2009-02-25 15:52:07 +08:00
ifc = argc - optind;
ifv = argv + optind;
/* When running dhcpcd against a single interface, we need to retain
* the old behaviour of waiting for an IP address */
if (ifc == 1 && !(options & DHCPCD_BACKGROUND))
options |= DHCPCD_WAITIP;
/* RTM_NEWADDR goes through the link socket as well which we
* need for IPv6 DAD, so we check for DHCPCD_LINK in handle_carrier
* instead.
* We also need to open this before checking for interfaces below
* so that we pickup any new addresses during the discover phase. */
if (linkfd == -1) {
linkfd = open_link_socket();
if (linkfd == -1)
syslog(LOG_ERR, "open_link_socket: %m");
else
eloop_event_add(linkfd, handle_link, NULL);
}
/* Start any dev listening plugin which may want to
* change the interface name provided by the kernel */
if ((options & (DHCPCD_MASTER | DHCPCD_DEV)) ==
(DHCPCD_MASTER | DHCPCD_DEV))
dev_start(dev_load);
ifaces = discover_interfaces(ifc, ifv);
for (i = 0; i < ifc; i++) {
if (find_interface(ifv[i]) == NULL)
syslog(LOG_ERR, "%s: interface not found or invalid",
2009-02-12 01:56:22 +08:00
ifv[i]);
}
if (ifaces == NULL || TAILQ_FIRST(ifaces) == NULL) {
if (ifc == 0)
syslog(LOG_ERR, "no valid interfaces found");
else
goto exit_failure;
2009-02-25 15:52:07 +08:00
if (!(options & DHCPCD_LINK)) {
syslog(LOG_ERR,
"aborting as link detection is disabled");
goto exit_failure;
}
}
if (options & DHCPCD_BACKGROUND && daemonise())
goto exit_success;
opt = 0;
TAILQ_FOREACH(ifp, ifaces, next) {
init_state(ifp, argc, argv);
if (ifp->carrier != LINK_DOWN)
opt = 1;
}
if (!(options & DHCPCD_BACKGROUND)) {
/* If we don't have a carrier, we may have to wait for a second
* before one becomes available if we brought an interface up */
if (opt == 0 &&
options & DHCPCD_LINK &&
options & DHCPCD_WAITUP &&
!(options & DHCPCD_WAITIP))
{
ts.tv_sec = 1;
ts.tv_nsec = 0;
nanosleep(&ts, NULL);
TAILQ_FOREACH(ifp, ifaces, next) {
handle_carrier(LINK_UNKNOWN, 0, ifp->name);
if (ifp->carrier != LINK_DOWN) {
opt = 1;
break;
}
}
}
if (options & DHCPCD_MASTER)
i = if_options->timeout;
else if ((ifp = TAILQ_FIRST(ifaces)))
i = ifp->options->timeout;
else
i = 0;
if (opt == 0 &&
options & DHCPCD_LINK &&
!(options & DHCPCD_WAITIP))
{
syslog(LOG_WARNING, "no interfaces have a carrier");
if (daemonise())
goto exit_success;
} else if (i > 0) {
if (options & DHCPCD_IPV4LL)
options |= DHCPCD_TIMEOUT_IPV4LL;
2012-11-13 19:25:51 +08:00
eloop_timeout_add_sec(i, handle_exit_timeout, NULL);
}
}
free_options(if_options);
if_options = NULL;
sort_interfaces();
TAILQ_FOREACH(ifp, ifaces, next) {
eloop_timeout_add_sec(0, start_interface, ifp);
}
i = eloop_start(&dhcpcd_sigset);
goto exit1;
exit_success:
i = EXIT_SUCCESS;
goto exit1;
exit_failure:
i = EXIT_FAILURE;
exit1:
/* Free memory and close fd's.
* We also set global variables back to their initial variables
* so we work nicely in a threads based environment as opposed to
* the normal process one. */
if (ifaces) {
while ((ifp = TAILQ_FIRST(ifaces))) {
TAILQ_REMOVE(ifaces, ifp, next);
free_interface(ifp);
}
free(ifaces);
ifaces = NULL;
}
if (duid) {
free(duid);
duid = NULL;
}
if (if_options) {
free_options(if_options);
if_options = NULL;
}
free_globals();
restore_kernel_ra();
ipv4_free(NULL);
ipv6_free(NULL);
dev_stop(options & DHCPCD_DAEMONISED);
if (linkfd != -1) {
close(linkfd);
linkfd = -1;
}
if (pidfd > -1) {
if (options & DHCPCD_MASTER) {
2014-02-05 20:01:09 +08:00
if (control_stop(&control_ctx) == -1)
syslog(LOG_ERR, "control_stop: %m:");
}
close(pidfd);
unlink(pidfile);
pidfd = -1;
}
if (pidfile) {
free(pidfile);
pidfile = NULL;
}
if (options & DHCPCD_STARTED && !(options & DHCPCD_FORKED))
syslog(LOG_INFO, "exited");
return i;
2006-11-28 04:23:22 +08:00
}