2013-04-05 04:31:04 +08:00
|
|
|
/*
|
2012-07-06 00:37:41 +08:00
|
|
|
* dhcpcd - DHCP client daemon
|
2013-02-02 22:05:55 +08:00
|
|
|
* Copyright (c) 2006-2013 Roy Marples <roy@marples.name>
|
2012-07-06 00:37:41 +08:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2013-05-22 00:05:36 +08:00
|
|
|
#include <sys/param.h>
|
2012-07-06 00:37:41 +08:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/socket.h>
|
|
|
|
|
2013-05-15 18:27:36 +08:00
|
|
|
#include <net/route.h>
|
2012-07-06 00:37:41 +08:00
|
|
|
#include <netinet/in.h>
|
|
|
|
|
2013-05-15 18:27:36 +08:00
|
|
|
#ifdef __linux__
|
|
|
|
# include <asm/types.h> /* for systems with broken headers */
|
|
|
|
# include <linux/rtnetlink.h>
|
2013-05-16 18:31:21 +08:00
|
|
|
/* Match Linux defines to BSD */
|
2013-05-16 18:54:20 +08:00
|
|
|
# define IN6_IFF_TENTATIVE (IFA_F_TENTATIVE | IFA_F_OPTIMISTIC)
|
2013-05-16 18:31:21 +08:00
|
|
|
# define IN6_IFF_DUPLICATED IFA_F_DADFAILED
|
|
|
|
#else
|
2013-05-16 18:48:03 +08:00
|
|
|
#ifdef __FreeBSD__ /* Needed so that including netinet6/in6_var.h works */
|
|
|
|
# include <net/if.h>
|
|
|
|
# include <net/if_var.h>
|
|
|
|
#endif
|
2013-05-16 18:31:21 +08:00
|
|
|
# include <netinet6/in6_var.h>
|
2013-05-15 18:27:36 +08:00
|
|
|
#endif
|
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
#include <errno.h>
|
|
|
|
#include <ifaddrs.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <syslog.h>
|
2013-02-16 04:33:13 +08:00
|
|
|
#include <unistd.h>
|
2012-07-06 00:37:41 +08:00
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "dhcpcd.h"
|
2012-11-07 07:40:15 +08:00
|
|
|
#include "dhcp6.h"
|
2013-05-18 07:09:36 +08:00
|
|
|
#include "eloop.h"
|
2012-07-06 00:37:41 +08:00
|
|
|
#include "ipv6.h"
|
|
|
|
#include "ipv6rs.h"
|
|
|
|
|
|
|
|
/* Hackery at it's finest. */
|
|
|
|
#ifndef s6_addr32
|
|
|
|
# define s6_addr32 __u6_addr.__u6_addr32
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static struct rt6head *routes;
|
2013-05-30 13:38:21 +08:00
|
|
|
static uint8_t do_pfx_flush;
|
2012-07-06 00:37:41 +08:00
|
|
|
|
|
|
|
#ifdef DEBUG_MEMORY
|
|
|
|
static void
|
|
|
|
ipv6_cleanup()
|
|
|
|
{
|
2012-07-12 17:19:14 +08:00
|
|
|
struct rt6 *rt;
|
2012-07-06 00:37:41 +08:00
|
|
|
|
2012-07-12 17:19:14 +08:00
|
|
|
while ((rt = TAILQ_FIRST(routes))) {
|
|
|
|
TAILQ_REMOVE(routes, rt, next);
|
|
|
|
free(rt);
|
|
|
|
}
|
2012-07-06 00:37:41 +08:00
|
|
|
free(routes);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-02-19 23:23:53 +08:00
|
|
|
int
|
|
|
|
ipv6_init(void)
|
2012-07-06 00:37:41 +08:00
|
|
|
{
|
2012-11-07 07:40:15 +08:00
|
|
|
|
2013-02-16 04:33:13 +08:00
|
|
|
if (routes == NULL) {
|
|
|
|
routes = malloc(sizeof(*routes));
|
2013-04-02 15:01:11 +08:00
|
|
|
if (routes == NULL)
|
2013-02-16 04:33:13 +08:00
|
|
|
return -1;
|
|
|
|
TAILQ_INIT(routes);
|
2012-07-06 00:37:41 +08:00
|
|
|
#ifdef DEBUG_MEMORY
|
2013-02-16 04:33:13 +08:00
|
|
|
atexit(ipv6_cleanup);
|
2012-07-06 00:37:41 +08:00
|
|
|
#endif
|
2013-02-16 04:33:13 +08:00
|
|
|
}
|
2012-11-07 07:40:15 +08:00
|
|
|
return 0;
|
2012-07-06 00:37:41 +08:00
|
|
|
}
|
|
|
|
|
2012-11-25 04:11:49 +08:00
|
|
|
ssize_t
|
|
|
|
ipv6_printaddr(char *s, ssize_t sl, const uint8_t *d, const char *ifname)
|
|
|
|
{
|
|
|
|
char buf[INET6_ADDRSTRLEN];
|
|
|
|
const char *p;
|
|
|
|
ssize_t l;
|
|
|
|
|
|
|
|
p = inet_ntop(AF_INET6, d, buf, sizeof(buf));
|
|
|
|
if (p == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
l = strlen(p);
|
|
|
|
if (d[0] == 0xfe && (d[1] & 0xc0) == 0x80)
|
|
|
|
l += 1 + strlen(ifname);
|
|
|
|
|
|
|
|
if (s == NULL)
|
|
|
|
return l;
|
|
|
|
|
|
|
|
if (sl < l) {
|
|
|
|
errno = ENOMEM;
|
|
|
|
return -1;
|
|
|
|
}
|
2013-04-02 15:01:11 +08:00
|
|
|
|
2012-11-25 04:11:49 +08:00
|
|
|
s += strlcpy(s, p, sl);
|
|
|
|
if (d[0] == 0xfe && (d[1] & 0xc0) == 0x80) {
|
|
|
|
*s++ = '%';
|
|
|
|
s += strlcpy(s, ifname, sl);
|
|
|
|
}
|
|
|
|
*s = '\0';
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
int
|
2013-05-18 07:09:36 +08:00
|
|
|
ipv6_makeaddr(struct in6_addr *addr, const struct interface *ifp,
|
2012-07-06 00:37:41 +08:00
|
|
|
const struct in6_addr *prefix, int prefix_len)
|
|
|
|
{
|
2013-05-18 07:09:36 +08:00
|
|
|
const struct ipv6_state *state;
|
|
|
|
const struct ll_addr *ap;
|
2012-07-06 00:37:41 +08:00
|
|
|
|
2012-11-07 07:40:15 +08:00
|
|
|
if (prefix_len < 0 || prefix_len > 64) {
|
2012-07-06 00:37:41 +08:00
|
|
|
errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-05-18 07:09:36 +08:00
|
|
|
state = IPV6_CSTATE(ifp);
|
|
|
|
ap = TAILQ_FIRST(&state->ll_addrs);
|
|
|
|
if (ap == NULL) {
|
2012-07-06 00:37:41 +08:00
|
|
|
errno = ENOENT;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(addr, prefix, sizeof(*prefix));
|
2013-05-18 07:09:36 +08:00
|
|
|
addr->s6_addr32[2] = ap->addr.s6_addr32[2];
|
|
|
|
addr->s6_addr32[3] = ap->addr.s6_addr32[3];
|
2012-07-06 00:37:41 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-07 07:40:15 +08:00
|
|
|
int
|
|
|
|
ipv6_makeprefix(struct in6_addr *prefix, const struct in6_addr *addr, int len)
|
|
|
|
{
|
|
|
|
int bytelen, bitlen;
|
|
|
|
|
|
|
|
if (len < 0 || len > 128) {
|
|
|
|
errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
bytelen = len / NBBY;
|
|
|
|
bitlen = len % NBBY;
|
|
|
|
memcpy(&prefix->s6_addr, &addr->s6_addr, bytelen);
|
|
|
|
if (bitlen != 0)
|
|
|
|
prefix->s6_addr[bytelen] >>= NBBY - bitlen;
|
|
|
|
memset((char *)prefix->s6_addr + bytelen, 0,
|
|
|
|
sizeof(prefix->s6_addr) - bytelen);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
int
|
|
|
|
ipv6_mask(struct in6_addr *mask, int len)
|
|
|
|
{
|
|
|
|
static const unsigned char masks[NBBY] =
|
|
|
|
{ 0x80, 0xc0, 0xe0, 0xf0, 0xf8, 0xfc, 0xfe, 0xff };
|
|
|
|
int bytes, bits, i;
|
|
|
|
|
|
|
|
if (len < 0 || len > 128) {
|
|
|
|
errno = EINVAL;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(mask, 0, sizeof(*mask));
|
|
|
|
bytes = len / NBBY;
|
|
|
|
bits = len % NBBY;
|
|
|
|
for (i = 0; i < bytes; i++)
|
|
|
|
mask->s6_addr[i] = 0xff;
|
|
|
|
if (bits)
|
2013-04-02 15:01:11 +08:00
|
|
|
mask->s6_addr[bytes] = masks[bits - 1];
|
2012-07-06 00:37:41 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
ipv6_prefixlen(const struct in6_addr *mask)
|
|
|
|
{
|
|
|
|
int x = 0, y;
|
|
|
|
const unsigned char *lim, *p;
|
|
|
|
|
|
|
|
lim = (const unsigned char *)mask + sizeof(*mask);
|
|
|
|
for (p = (const unsigned char *)mask; p < lim; x++, p++) {
|
|
|
|
if (*p != 0xff)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
y = 0;
|
|
|
|
if (p < lim) {
|
|
|
|
for (y = 0; y < NBBY; y++) {
|
|
|
|
if ((*p & (0x80 >> y)) == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* when the limit pointer is given, do a stricter check on the
|
|
|
|
* remaining bits.
|
|
|
|
*/
|
|
|
|
if (p < lim) {
|
|
|
|
if (y != 0 && (*p & (0x00ff >> y)) != 0)
|
|
|
|
return -1;
|
|
|
|
for (p = p + 1; p < lim; p++)
|
|
|
|
if (*p != 0)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return x * NBBY + y;
|
|
|
|
}
|
|
|
|
|
2013-05-03 20:11:49 +08:00
|
|
|
int
|
|
|
|
ipv6_addaddr(struct ipv6_addr *ap)
|
|
|
|
{
|
|
|
|
|
2013-05-31 21:01:54 +08:00
|
|
|
syslog(ap->flags & IPV6_AF_NEW ? LOG_INFO : LOG_DEBUG,
|
2013-05-03 20:11:49 +08:00
|
|
|
"%s: adding address %s", ap->iface->name, ap->saddr);
|
|
|
|
if (add_address6(ap->iface, ap) == -1) {
|
|
|
|
syslog(LOG_ERR, "add_address6 %m");
|
|
|
|
return -1;
|
|
|
|
}
|
2013-05-31 21:01:54 +08:00
|
|
|
ap->flags &= ~IPV6_AF_NEW;
|
|
|
|
ap->flags |= IPV6_AF_ADDED;
|
|
|
|
if (ap->delegating_iface)
|
|
|
|
ap->flags |= IPV6_AF_DELEGATED;
|
2013-05-03 20:11:49 +08:00
|
|
|
if (ipv6_removesubnet(ap->iface, ap) == -1)
|
|
|
|
syslog(LOG_ERR,"ipv6_removesubnet %m");
|
2013-05-31 21:40:46 +08:00
|
|
|
if (ap->prefix_pltime == ND6_INFINITE_LIFETIME &&
|
|
|
|
ap->prefix_vltime == ND6_INFINITE_LIFETIME)
|
|
|
|
syslog(LOG_DEBUG,
|
|
|
|
"%s: pltime infinity, vltime infinity",
|
|
|
|
ap->iface->name);
|
|
|
|
else if (ap->prefix_pltime == ND6_INFINITE_LIFETIME)
|
|
|
|
syslog(LOG_DEBUG,
|
|
|
|
"%s: pltime infinity, vltime %d seconds",
|
|
|
|
ap->iface->name, ap->prefix_vltime);
|
|
|
|
else if (ap->prefix_vltime == ND6_INFINITE_LIFETIME)
|
|
|
|
syslog(LOG_DEBUG,
|
|
|
|
"%s: pltime %d seconds, vltime infinity",
|
|
|
|
ap->iface->name, ap->prefix_pltime);
|
|
|
|
else
|
|
|
|
syslog(LOG_DEBUG,
|
|
|
|
"%s: pltime %d seconds, vltime %d seconds",
|
|
|
|
ap->iface->name, ap->prefix_pltime, ap->prefix_vltime);
|
2013-05-03 20:11:49 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-11-07 07:40:15 +08:00
|
|
|
ssize_t
|
2013-05-03 20:11:49 +08:00
|
|
|
ipv6_addaddrs(struct ipv6_addrhead *addrs)
|
2012-11-07 07:40:15 +08:00
|
|
|
{
|
|
|
|
struct ipv6_addr *ap;
|
|
|
|
ssize_t i;
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
TAILQ_FOREACH(ap, addrs, next) {
|
|
|
|
if (ap->prefix_vltime == 0 ||
|
2013-05-31 21:01:54 +08:00
|
|
|
IN6_IS_ADDR_UNSPECIFIED(&ap->addr) ||
|
|
|
|
ap->flags & IPV6_AF_DELEGATED)
|
2012-11-07 07:40:15 +08:00
|
|
|
continue;
|
2013-05-03 20:11:49 +08:00
|
|
|
if (ipv6_addaddr(ap) == 0)
|
2012-11-07 07:40:15 +08:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2013-05-18 07:09:36 +08:00
|
|
|
static struct ipv6_state *
|
|
|
|
ipv6_getstate(struct interface *ifp)
|
|
|
|
{
|
|
|
|
struct ipv6_state *state;
|
|
|
|
|
|
|
|
state = IPV6_STATE(ifp);
|
|
|
|
if (state == NULL) {
|
|
|
|
ifp->if_data[IF_DATA_IPV6] = malloc(sizeof(*state));
|
|
|
|
state = IPV6_STATE(ifp);
|
|
|
|
if (state == NULL) {
|
|
|
|
syslog(LOG_ERR, "%s: %m", __func__);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
TAILQ_INIT(&state->ll_addrs);
|
|
|
|
TAILQ_INIT(&state->ll_callbacks);
|
|
|
|
}
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
2013-05-15 18:27:36 +08:00
|
|
|
void
|
2013-05-18 07:09:36 +08:00
|
|
|
ipv6_handleifa(int cmd, struct if_head *ifs, const char *ifname,
|
2013-05-16 18:31:21 +08:00
|
|
|
const struct in6_addr *addr, int flags)
|
2013-05-15 18:27:36 +08:00
|
|
|
{
|
2013-05-18 07:09:36 +08:00
|
|
|
struct interface *ifp;
|
|
|
|
struct ipv6_state *state;
|
|
|
|
struct ll_addr *ap;
|
|
|
|
struct ll_callback *cb;
|
|
|
|
|
2013-05-29 20:59:47 +08:00
|
|
|
#if 0
|
|
|
|
char buf[INET6_ADDRSTRLEN];
|
|
|
|
inet_ntop(AF_INET6, &addr->s6_addr,
|
|
|
|
buf, INET6_ADDRSTRLEN);
|
|
|
|
syslog(LOG_DEBUG, "%s: cmd %d addr %s flags %d",
|
|
|
|
ifname, cmd, buf, flags);
|
|
|
|
#endif
|
|
|
|
|
2013-05-18 07:09:36 +08:00
|
|
|
/* Safety - ignore tentative announcements */
|
2013-05-30 13:38:21 +08:00
|
|
|
if (cmd == RTM_NEWADDR) {
|
|
|
|
if (flags & (IN6_IFF_TENTATIVE || IN6_IFF_DUPLICATED))
|
|
|
|
cmd = RTM_DELADDR;
|
|
|
|
#ifdef IN6_IFF_DETACHED
|
|
|
|
if (flags & IN6_IFF_DETACHED)
|
|
|
|
cmd = RTM_DELADDR;
|
|
|
|
#endif
|
|
|
|
}
|
2013-05-18 07:09:36 +08:00
|
|
|
|
|
|
|
if (ifs == NULL)
|
|
|
|
ifs = ifaces;
|
|
|
|
if (ifs == NULL) {
|
|
|
|
errno = ESRCH;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
TAILQ_FOREACH(ifp, ifs, next) {
|
|
|
|
if (strcmp(ifp->name, ifname) == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (ifp == NULL) {
|
|
|
|
errno = ESRCH;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IN6_IS_ADDR_LINKLOCAL(addr)) {
|
|
|
|
state = ipv6_getstate(ifp);
|
|
|
|
if (state == NULL)
|
|
|
|
return;
|
|
|
|
TAILQ_FOREACH(ap, &state->ll_addrs, next) {
|
2013-05-18 23:37:27 +08:00
|
|
|
if (IN6_ARE_ADDR_EQUAL(&ap->addr, addr))
|
|
|
|
break;
|
2013-05-18 07:09:36 +08:00
|
|
|
}
|
|
|
|
switch (cmd) {
|
|
|
|
case RTM_DELADDR:
|
|
|
|
if (ap) {
|
|
|
|
TAILQ_REMOVE(&state->ll_addrs, ap, next);
|
|
|
|
free(ap);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
case RTM_NEWADDR:
|
|
|
|
if (ap == NULL) {
|
|
|
|
ap = calloc(1, sizeof(*ap));
|
|
|
|
memcpy(ap->addr.s6_addr, addr->s6_addr,
|
|
|
|
sizeof(ap->addr.s6_addr));
|
|
|
|
TAILQ_INSERT_TAIL(&state->ll_addrs,
|
|
|
|
ap, next);
|
|
|
|
|
|
|
|
/* Now run any callbacks.
|
|
|
|
* Typically IPv6RS or DHCPv6 */
|
|
|
|
while ((cb = TAILQ_FIRST(&state->ll_callbacks)))
|
|
|
|
{
|
|
|
|
TAILQ_REMOVE(&state->ll_callbacks,
|
|
|
|
cb, next);
|
|
|
|
cb->callback(cb->arg);
|
|
|
|
free(cb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2013-05-15 18:27:36 +08:00
|
|
|
|
2013-05-16 18:31:21 +08:00
|
|
|
ipv6rs_handleifa(cmd, ifname, addr, flags);
|
|
|
|
dhcp6_handleifa(cmd, ifname, addr, flags);
|
2013-05-15 18:27:36 +08:00
|
|
|
}
|
|
|
|
|
2013-05-30 20:07:25 +08:00
|
|
|
const struct ll_addr *
|
|
|
|
ipv6_linklocal(const struct interface *ifp)
|
2013-05-18 07:09:36 +08:00
|
|
|
{
|
|
|
|
const struct ipv6_state *state;
|
|
|
|
|
|
|
|
state = IPV6_CSTATE(ifp);
|
2013-05-30 20:07:25 +08:00
|
|
|
if (state)
|
|
|
|
return TAILQ_FIRST(&state->ll_addrs);
|
|
|
|
return NULL;
|
2013-05-18 07:09:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int ipv6_addlinklocalcallback(struct interface *ifp,
|
|
|
|
void (*callback)(void *), void *arg)
|
|
|
|
{
|
|
|
|
struct ipv6_state *state;
|
|
|
|
struct ll_callback *cb;
|
|
|
|
|
|
|
|
state = ipv6_getstate(ifp);
|
|
|
|
TAILQ_FOREACH(cb, &state->ll_callbacks, next) {
|
|
|
|
if (cb->callback == callback && cb->arg == arg)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (cb == NULL) {
|
|
|
|
cb = malloc(sizeof(*cb));
|
|
|
|
if (cb == NULL) {
|
|
|
|
syslog(LOG_ERR, "%s: %m", __func__);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
cb->callback = callback;
|
|
|
|
cb->arg = arg;
|
|
|
|
TAILQ_INSERT_TAIL(&state->ll_callbacks, cb, next);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ipv6_free(struct interface *ifp)
|
|
|
|
{
|
|
|
|
struct ipv6_state *state;
|
|
|
|
struct ll_addr *ap;
|
|
|
|
struct ll_callback *cb;
|
|
|
|
|
|
|
|
state = IPV6_STATE(ifp);
|
|
|
|
if (state) {
|
|
|
|
while ((ap = TAILQ_FIRST(&state->ll_addrs))) {
|
|
|
|
TAILQ_REMOVE(&state->ll_addrs, ap, next);
|
|
|
|
free(ap);
|
|
|
|
}
|
|
|
|
while ((cb = TAILQ_FIRST(&state->ll_callbacks))) {
|
|
|
|
TAILQ_REMOVE(&state->ll_callbacks, cb, next);
|
|
|
|
free(cb);
|
|
|
|
}
|
|
|
|
free(state);
|
2013-05-30 13:38:21 +08:00
|
|
|
ifp->if_data[IF_DATA_IPV6] = NULL;
|
2013-05-18 07:09:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-05-15 18:27:36 +08:00
|
|
|
int
|
|
|
|
ipv6_handleifa_addrs(int cmd,
|
2013-05-16 18:31:21 +08:00
|
|
|
struct ipv6_addrhead *addrs, const struct in6_addr *addr, int flags)
|
2013-05-15 18:27:36 +08:00
|
|
|
{
|
|
|
|
struct ipv6_addr *ap, *apn;
|
|
|
|
uint8_t found, alldadcompleted;
|
|
|
|
|
|
|
|
alldadcompleted = 1;
|
|
|
|
found = 0;
|
|
|
|
TAILQ_FOREACH_SAFE(ap, addrs, next, apn) {
|
2013-05-18 23:37:27 +08:00
|
|
|
if (!IN6_ARE_ADDR_EQUAL(addr, &ap->addr)) {
|
2013-05-31 21:01:54 +08:00
|
|
|
if ((ap->flags & IPV6_AF_DADCOMPLETED) == 0)
|
2013-05-15 18:27:36 +08:00
|
|
|
alldadcompleted = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
switch (cmd) {
|
|
|
|
case RTM_DELADDR:
|
|
|
|
syslog(LOG_INFO, "%s: deleted address %s",
|
|
|
|
ap->iface->name, ap->saddr);
|
|
|
|
TAILQ_REMOVE(addrs, ap, next);
|
|
|
|
free(ap);
|
|
|
|
break;
|
|
|
|
case RTM_NEWADDR:
|
2013-05-16 18:31:21 +08:00
|
|
|
/* Safety - ignore tentative announcements */
|
|
|
|
if (flags & IN6_IFF_TENTATIVE)
|
|
|
|
break;
|
2013-05-31 21:01:54 +08:00
|
|
|
if ((ap->flags & IPV6_AF_DADCOMPLETED) == 0) {
|
2013-05-15 18:27:36 +08:00
|
|
|
found++;
|
2013-05-31 21:01:54 +08:00
|
|
|
if (flags & IN6_IFF_DUPLICATED)
|
|
|
|
ap->flags |= IPV6_AF_DUPLICATED;
|
2013-05-16 18:31:21 +08:00
|
|
|
if (ap->dadcallback)
|
|
|
|
ap->dadcallback(ap);
|
|
|
|
/* We need to set this here in-case the
|
|
|
|
* dadcallback function checks it */
|
2013-05-31 21:01:54 +08:00
|
|
|
ap->flags |= IPV6_AF_DADCOMPLETED;
|
2013-05-15 18:27:36 +08:00
|
|
|
}
|
2013-05-29 20:59:47 +08:00
|
|
|
break;
|
2013-05-15 18:27:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return alldadcompleted ? found : 0;
|
|
|
|
}
|
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
static struct rt6 *
|
|
|
|
find_route6(struct rt6head *rts, const struct rt6 *r)
|
|
|
|
{
|
|
|
|
struct rt6 *rt;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(rt, rts, next) {
|
|
|
|
if (IN6_ARE_ADDR_EQUAL(&rt->dest, &r->dest) &&
|
|
|
|
#if HAVE_ROUTE_METRIC
|
|
|
|
rt->iface->metric == r->iface->metric &&
|
|
|
|
#endif
|
|
|
|
IN6_ARE_ADDR_EQUAL(&rt->net, &r->net))
|
|
|
|
return rt;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
desc_route(const char *cmd, const struct rt6 *rt)
|
|
|
|
{
|
|
|
|
char destbuf[INET6_ADDRSTRLEN];
|
|
|
|
char gatebuf[INET6_ADDRSTRLEN];
|
|
|
|
const char *ifname = rt->iface->name, *dest, *gate;
|
|
|
|
|
|
|
|
dest = inet_ntop(AF_INET6, &rt->dest.s6_addr,
|
|
|
|
destbuf, INET6_ADDRSTRLEN);
|
|
|
|
gate = inet_ntop(AF_INET6, &rt->gate.s6_addr,
|
|
|
|
gatebuf, INET6_ADDRSTRLEN);
|
|
|
|
if (IN6_ARE_ADDR_EQUAL(&rt->gate, &in6addr_any))
|
2012-11-01 01:10:22 +08:00
|
|
|
syslog(LOG_INFO, "%s: %s route to %s/%d", ifname, cmd,
|
2012-07-06 00:37:41 +08:00
|
|
|
dest, ipv6_prefixlen(&rt->net));
|
|
|
|
else if (IN6_ARE_ADDR_EQUAL(&rt->dest, &in6addr_any) &&
|
|
|
|
IN6_ARE_ADDR_EQUAL(&rt->net, &in6addr_any))
|
2012-11-01 01:10:22 +08:00
|
|
|
syslog(LOG_INFO, "%s: %s default route via %s", ifname, cmd,
|
2012-07-06 00:37:41 +08:00
|
|
|
gate);
|
|
|
|
else
|
2012-11-01 01:10:22 +08:00
|
|
|
syslog(LOG_INFO, "%s: %s route to %s/%d via %s", ifname, cmd,
|
2012-07-06 00:37:41 +08:00
|
|
|
dest, ipv6_prefixlen(&rt->net), gate);
|
|
|
|
}
|
|
|
|
|
2013-05-24 16:06:57 +08:00
|
|
|
#define n_route(a) nc_route(1, a, a)
|
|
|
|
#define c_route(a, b) nc_route(0, a, b)
|
2012-07-06 00:37:41 +08:00
|
|
|
static int
|
2013-05-24 16:06:57 +08:00
|
|
|
nc_route(int add, struct rt6 *ort, struct rt6 *nrt)
|
2012-07-06 00:37:41 +08:00
|
|
|
{
|
|
|
|
|
|
|
|
/* Don't set default routes if not asked to */
|
|
|
|
if (IN6_IS_ADDR_UNSPECIFIED(&nrt->dest) &&
|
|
|
|
IN6_IS_ADDR_UNSPECIFIED(&nrt->net) &&
|
2013-02-03 18:35:59 +08:00
|
|
|
!(nrt->iface->options->options & DHCPCD_GATEWAY))
|
2012-07-06 00:37:41 +08:00
|
|
|
return -1;
|
|
|
|
|
2013-05-24 16:06:57 +08:00
|
|
|
desc_route(add ? "adding" : "changing", nrt);
|
|
|
|
/* We delete and add the route so that we can change metric and
|
|
|
|
* prefer the interface. */
|
2013-05-30 13:38:21 +08:00
|
|
|
if (del_route6(ort) == 0 && add)
|
|
|
|
do_pfx_flush = 1;
|
|
|
|
if (add_route6(nrt) == 0)
|
2012-07-06 00:37:41 +08:00
|
|
|
return 0;
|
2013-05-24 16:06:57 +08:00
|
|
|
syslog(LOG_ERR, "%s: add_route6: %m", nrt->iface->name);
|
2012-07-06 00:37:41 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
d_route(struct rt6 *rt)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
desc_route("deleting", rt);
|
|
|
|
retval = del_route6(rt);
|
|
|
|
if (retval != 0 && errno != ENOENT && errno != ESRCH)
|
2013-05-24 16:06:57 +08:00
|
|
|
syslog(LOG_ERR,"%s: del_route6: %m", rt->iface->name);
|
2012-07-06 00:37:41 +08:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct rt6 *
|
2012-11-07 07:40:15 +08:00
|
|
|
make_route(const struct interface *ifp, struct ra *rap)
|
2012-07-06 00:37:41 +08:00
|
|
|
{
|
|
|
|
struct rt6 *r;
|
|
|
|
|
2013-02-16 05:45:08 +08:00
|
|
|
r = calloc(1, sizeof(*r));
|
|
|
|
if (r == NULL) {
|
|
|
|
syslog(LOG_ERR, "%s: %m", __func__);
|
|
|
|
return NULL;
|
|
|
|
}
|
2012-07-06 00:37:41 +08:00
|
|
|
r->ra = rap;
|
2012-11-07 07:40:15 +08:00
|
|
|
r->iface = ifp;
|
|
|
|
r->metric = ifp->metric;
|
|
|
|
if (rap)
|
|
|
|
r->mtu = rap->mtu;
|
|
|
|
else
|
|
|
|
r->mtu = 0;
|
2012-07-06 00:37:41 +08:00
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct rt6 *
|
2012-11-07 07:40:15 +08:00
|
|
|
make_prefix(const struct interface * ifp,struct ra *rap, struct ipv6_addr *addr)
|
2012-07-06 00:37:41 +08:00
|
|
|
{
|
|
|
|
struct rt6 *r;
|
|
|
|
|
|
|
|
if (addr == NULL || addr->prefix_len > 128)
|
|
|
|
return NULL;
|
|
|
|
|
2012-11-07 07:40:15 +08:00
|
|
|
r = make_route(ifp, rap);
|
2013-02-16 05:45:08 +08:00
|
|
|
if (r == NULL)
|
|
|
|
return r;
|
2012-07-06 00:37:41 +08:00
|
|
|
r->dest = addr->prefix;
|
|
|
|
ipv6_mask(&r->net, addr->prefix_len);
|
|
|
|
r->gate = in6addr_any;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2012-11-07 07:40:15 +08:00
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
static struct rt6 *
|
|
|
|
make_router(struct ra *rap)
|
|
|
|
{
|
|
|
|
struct rt6 *r;
|
|
|
|
|
2012-11-07 07:40:15 +08:00
|
|
|
r = make_route(rap->iface, rap);
|
2013-02-16 05:45:08 +08:00
|
|
|
if (r == NULL)
|
|
|
|
return NULL;
|
2012-07-06 00:37:41 +08:00
|
|
|
r->dest = in6addr_any;
|
|
|
|
r->net = in6addr_any;
|
|
|
|
r->gate = rap->from;
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2012-11-07 07:40:15 +08:00
|
|
|
ipv6_removesubnet(const struct interface *ifp, struct ipv6_addr *addr)
|
2012-07-06 00:37:41 +08:00
|
|
|
{
|
|
|
|
struct rt6 *rt;
|
2012-08-07 15:03:13 +08:00
|
|
|
#if HAVE_ROUTE_METRIC
|
|
|
|
struct rt6 *ort;
|
|
|
|
#endif
|
2012-07-06 00:37:41 +08:00
|
|
|
int r;
|
|
|
|
|
|
|
|
/* We need to delete the subnet route to have our metric or
|
|
|
|
* prefer the interface. */
|
|
|
|
r = 0;
|
2012-11-07 07:40:15 +08:00
|
|
|
rt = make_prefix(ifp, NULL, addr);
|
2012-07-06 00:37:41 +08:00
|
|
|
if (rt) {
|
2012-11-07 07:40:15 +08:00
|
|
|
rt->iface = ifp;
|
2012-07-06 00:37:41 +08:00
|
|
|
#ifdef __linux__
|
2013-04-02 15:30:13 +08:00
|
|
|
rt->metric = 256;
|
2012-07-06 00:37:41 +08:00
|
|
|
#else
|
|
|
|
rt->metric = 0;
|
|
|
|
#endif
|
2012-08-07 15:03:13 +08:00
|
|
|
#if HAVE_ROUTE_METRIC
|
|
|
|
/* For some reason, Linux likes to re-add the subnet
|
|
|
|
route under the original metric.
|
|
|
|
I would love to find a way of stopping this! */
|
|
|
|
if ((ort = find_route6(routes, rt)) == NULL ||
|
|
|
|
ort->metric != rt->metric)
|
|
|
|
#else
|
2012-07-06 00:37:41 +08:00
|
|
|
if (!find_route6(routes, rt))
|
2012-08-07 15:03:13 +08:00
|
|
|
#endif
|
2012-10-12 20:13:16 +08:00
|
|
|
{
|
2012-07-06 00:37:41 +08:00
|
|
|
r = del_route6(rt);
|
2012-10-12 20:13:16 +08:00
|
|
|
if (r == -1 && errno == ESRCH)
|
|
|
|
r = 0;
|
|
|
|
}
|
2012-07-06 00:37:41 +08:00
|
|
|
free(rt);
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define RT_IS_DEFAULT(rtp) \
|
|
|
|
(IN6_ARE_ADDR_EQUAL(&((rtp)->dest), &in6addr_any) && \
|
|
|
|
IN6_ARE_ADDR_EQUAL(&((rtp)->net), &in6addr_any))
|
|
|
|
|
2013-05-21 17:24:49 +08:00
|
|
|
static void
|
|
|
|
ipv6_buildroutes1(struct rt6head *dnr, int expired)
|
|
|
|
{
|
|
|
|
struct rt6 *rt;
|
|
|
|
struct ra *rap;
|
|
|
|
struct ipv6_addr *addr;
|
|
|
|
|
|
|
|
TAILQ_FOREACH(rap, &ipv6_routers, next) {
|
|
|
|
if (rap->expired != expired)
|
|
|
|
continue;
|
2013-05-23 06:55:36 +08:00
|
|
|
if (rap->iface->options->options & DHCPCD_IPV6RA_OWN) {
|
2013-05-21 17:24:49 +08:00
|
|
|
TAILQ_FOREACH(addr, &rap->addrs, next) {
|
2013-05-31 21:01:54 +08:00
|
|
|
if ((addr->flags & IPV6_AF_ONLINK) == 0)
|
2013-05-21 17:24:49 +08:00
|
|
|
continue;
|
|
|
|
rt = make_prefix(rap->iface, rap, addr);
|
|
|
|
if (rt)
|
|
|
|
TAILQ_INSERT_TAIL(dnr, rt, next);
|
|
|
|
}
|
|
|
|
}
|
2013-05-23 06:55:36 +08:00
|
|
|
if (rap->iface->options->options &
|
|
|
|
(DHCPCD_IPV6RA_OWN | DHCPCD_IPV6RA_OWN_DEFAULT))
|
|
|
|
{
|
|
|
|
rt = make_router(rap);
|
|
|
|
if (rt)
|
|
|
|
TAILQ_INSERT_TAIL(dnr, rt, next);
|
|
|
|
}
|
2013-05-21 17:24:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
void
|
2012-11-07 07:40:15 +08:00
|
|
|
ipv6_buildroutes(void)
|
2012-07-06 00:37:41 +08:00
|
|
|
{
|
|
|
|
struct rt6head dnr, *nrs;
|
|
|
|
struct rt6 *rt, *rtn, *or;
|
|
|
|
struct ipv6_addr *addr;
|
2012-11-07 07:40:15 +08:00
|
|
|
const struct interface *ifp;
|
|
|
|
const struct dhcp6_state *d6_state;
|
2013-05-30 13:38:21 +08:00
|
|
|
uint8_t have_default;
|
2012-07-06 00:37:41 +08:00
|
|
|
|
|
|
|
if (!(options & (DHCPCD_IPV6RA_OWN | DHCPCD_IPV6RA_OWN_DEFAULT)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
TAILQ_INIT(&dnr);
|
2013-05-21 17:24:49 +08:00
|
|
|
|
|
|
|
/* First add reachable routers and their prefixes */
|
|
|
|
ipv6_buildroutes1(&dnr, 0);
|
2013-05-22 16:11:13 +08:00
|
|
|
#ifdef HAVE_ROUTE_METRIC
|
|
|
|
have_default = (TAILQ_FIRST(&dnr) != NULL);
|
|
|
|
#endif
|
2013-05-21 17:24:49 +08:00
|
|
|
|
|
|
|
/* We have no way of knowing if prefixes added by DHCP are reachable
|
|
|
|
* or not, so we have to assume they are */
|
2013-02-19 21:37:42 +08:00
|
|
|
TAILQ_FOREACH(ifp, ifaces, next) {
|
2012-11-07 07:40:15 +08:00
|
|
|
d6_state = D6_CSTATE(ifp);
|
2013-04-01 20:15:47 +08:00
|
|
|
if (d6_state &&
|
|
|
|
(d6_state->state == DH6S_BOUND ||
|
|
|
|
d6_state->state == DH6S_DELEGATED))
|
|
|
|
{
|
2012-11-07 07:40:15 +08:00
|
|
|
TAILQ_FOREACH(addr, &d6_state->addrs, next) {
|
2013-05-31 21:01:54 +08:00
|
|
|
if ((addr->flags & IPV6_AF_ONLINK) == 0 ||
|
2013-05-24 05:50:34 +08:00
|
|
|
IN6_IS_ADDR_UNSPECIFIED(&addr->addr))
|
2012-12-11 17:00:49 +08:00
|
|
|
continue;
|
2012-11-07 07:40:15 +08:00
|
|
|
rt = make_prefix(ifp, NULL, addr);
|
|
|
|
if (rt)
|
|
|
|
TAILQ_INSERT_TAIL(&dnr, rt, next);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-05-21 17:24:49 +08:00
|
|
|
|
2013-05-22 16:11:13 +08:00
|
|
|
#ifdef HAVE_ROUTE_METRIC
|
|
|
|
/* If we have an unreachable router, we really do need to remove the
|
|
|
|
* route to it beause it could be a lower metric than a reachable
|
|
|
|
* router. Of course, we should at least have some routers if all
|
|
|
|
* are unreachable. */
|
|
|
|
if (!have_default)
|
|
|
|
#endif
|
2013-05-21 17:24:49 +08:00
|
|
|
/* Add our non-reachable routers and prefixes
|
|
|
|
* Unsure if this is needed, but it's a close match to kernel
|
|
|
|
* behaviour */
|
|
|
|
ipv6_buildroutes1(&dnr, 1);
|
2012-07-06 00:37:41 +08:00
|
|
|
|
2013-02-16 21:21:35 +08:00
|
|
|
nrs = malloc(sizeof(*nrs));
|
|
|
|
if (nrs == NULL) {
|
|
|
|
syslog(LOG_ERR, "%s: %m", __func__);
|
|
|
|
return;
|
|
|
|
}
|
2012-07-06 00:37:41 +08:00
|
|
|
TAILQ_INIT(nrs);
|
2013-05-30 13:38:21 +08:00
|
|
|
have_default = do_pfx_flush = 0;
|
2012-07-06 00:37:41 +08:00
|
|
|
TAILQ_FOREACH_SAFE(rt, &dnr, next, rtn) {
|
|
|
|
/* Is this route already in our table? */
|
|
|
|
if (find_route6(nrs, rt) != NULL)
|
|
|
|
continue;
|
|
|
|
//rt->src.s_addr = ifp->addr.s_addr;
|
|
|
|
/* Do we already manage it? */
|
|
|
|
if ((or = find_route6(routes, rt))) {
|
|
|
|
if (or->iface != rt->iface ||
|
|
|
|
// or->src.s_addr != ifp->addr.s_addr ||
|
|
|
|
!IN6_ARE_ADDR_EQUAL(&rt->gate, &or->gate) ||
|
|
|
|
rt->metric != or->metric)
|
|
|
|
{
|
|
|
|
if (c_route(or, rt) != 0)
|
|
|
|
continue;
|
2013-05-30 13:38:21 +08:00
|
|
|
do_pfx_flush = 1;
|
2012-07-06 00:37:41 +08:00
|
|
|
}
|
|
|
|
TAILQ_REMOVE(routes, or, next);
|
|
|
|
free(or);
|
|
|
|
} else {
|
|
|
|
if (n_route(rt) != 0)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (RT_IS_DEFAULT(rt))
|
|
|
|
have_default = 1;
|
|
|
|
TAILQ_REMOVE(&dnr, rt, next);
|
|
|
|
TAILQ_INSERT_TAIL(nrs, rt, next);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Free any routes we failed to add/change */
|
|
|
|
while ((rt = TAILQ_FIRST(&dnr))) {
|
|
|
|
TAILQ_REMOVE(&dnr, rt, next);
|
|
|
|
free(rt);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Remove old routes we used to manage
|
|
|
|
* If we own the default route, but not RA management itself
|
|
|
|
* then we need to preserve the last best default route we had */
|
2012-11-01 07:07:06 +08:00
|
|
|
while ((rt = TAILQ_LAST(routes, rt6head))) {
|
2012-11-01 04:14:08 +08:00
|
|
|
TAILQ_REMOVE(routes, rt, next);
|
2012-07-06 00:37:41 +08:00
|
|
|
if (find_route6(nrs, rt) == NULL) {
|
|
|
|
if (!have_default &&
|
|
|
|
(options & DHCPCD_IPV6RA_OWN_DEFAULT) &&
|
|
|
|
!(options & DHCPCD_IPV6RA_OWN) &&
|
|
|
|
RT_IS_DEFAULT(rt))
|
|
|
|
have_default = 1;
|
|
|
|
/* no need to add it back to our routing table
|
|
|
|
* as we delete an exiting route when we add
|
|
|
|
* a new one */
|
|
|
|
else
|
|
|
|
d_route(rt);
|
|
|
|
}
|
|
|
|
free(rt);
|
|
|
|
}
|
2012-11-01 04:14:08 +08:00
|
|
|
|
2013-05-30 13:38:21 +08:00
|
|
|
if (do_pfx_flush) {
|
|
|
|
/* We need to flush all entries in our prefix list
|
|
|
|
* if we changed a route so that we leave via the
|
|
|
|
* correct interface */
|
|
|
|
if (pfx_flush() == -1)
|
|
|
|
syslog(LOG_ERR, "pfx_flush: %m");
|
|
|
|
|
|
|
|
/* XXX FIXME
|
|
|
|
* On multi-homed systems on the same network
|
|
|
|
* (ie wired and wireless) swapping the active routes
|
|
|
|
* between interfaces (ie, bringing down the wired and
|
|
|
|
* back up again while wireless is active) causes the
|
|
|
|
* ping6 command to emit bogus Network is down errors.
|
|
|
|
* This could be a kernel bug though */
|
|
|
|
}
|
|
|
|
|
2012-07-06 00:37:41 +08:00
|
|
|
free(routes);
|
|
|
|
routes = nrs;
|
|
|
|
}
|