version 1.1.1.2, 2014/06/15 16:31:38
|
version 1.1.1.3, 2016/11/02 09:57:01
|
Line 1
|
Line 1
|
/* dnsmasq is Copyright (c) 2000-2014 Simon Kelley | /* dnsmasq is Copyright (c) 2000-2016 Simon Kelley |
|
|
This program is free software; you can redistribute it and/or modify |
This program is free software; you can redistribute it and/or modify |
it under the terms of the GNU General Public License as published by |
it under the terms of the GNU General Public License as published by |
Line 27 struct iface_param {
|
Line 27 struct iface_param {
|
int ind, addr_match; |
int ind, addr_match; |
}; |
}; |
|
|
struct mac_param { |
|
struct in6_addr *target; |
|
unsigned char *mac; |
|
unsigned int maclen; |
|
}; |
|
|
|
|
|
static int complete_context6(struct in6_addr *local, int prefix, |
static int complete_context6(struct in6_addr *local, int prefix, |
int scope, int if_index, int flags, |
int scope, int if_index, int flags, |
unsigned int preferred, unsigned int valid, void *vparam); |
unsigned int preferred, unsigned int valid, void *vparam); |
static int find_mac(int family, char *addrp, char *mac, size_t maclen, void *parmv); |
|
static int make_duid1(int index, unsigned int type, char *mac, size_t maclen, void *parm); |
static int make_duid1(int index, unsigned int type, char *mac, size_t maclen, void *parm); |
|
|
void dhcp6_init(void) |
void dhcp6_init(void) |
Line 144 void dhcp6_packet(time_t now)
|
Line 137 void dhcp6_packet(time_t now)
|
|
|
if ((port = relay_reply6(&from, sz, ifr.ifr_name)) == 0) |
if ((port = relay_reply6(&from, sz, ifr.ifr_name)) == 0) |
{ |
{ |
|
struct dhcp_bridge *bridge, *alias; |
|
|
for (tmp = daemon->if_except; tmp; tmp = tmp->next) |
for (tmp = daemon->if_except; tmp; tmp = tmp->next) |
if (tmp->name && wildcard_match(tmp->name, ifr.ifr_name)) |
if (tmp->name && wildcard_match(tmp->name, ifr.ifr_name)) |
return; |
return; |
Line 160 void dhcp6_packet(time_t now)
|
Line 155 void dhcp6_packet(time_t now)
|
memset(&parm.fallback, 0, IN6ADDRSZ); |
memset(&parm.fallback, 0, IN6ADDRSZ); |
memset(&parm.ll_addr, 0, IN6ADDRSZ); |
memset(&parm.ll_addr, 0, IN6ADDRSZ); |
memset(&parm.ula_addr, 0, IN6ADDRSZ); |
memset(&parm.ula_addr, 0, IN6ADDRSZ); |
|
|
|
/* If the interface on which the DHCPv6 request was received is |
|
an alias of some other interface (as specified by the |
|
--bridge-interface option), change parm.ind so that we look |
|
for DHCPv6 contexts associated with the aliased interface |
|
instead of with the aliasing one. */ |
|
for (bridge = daemon->bridges; bridge; bridge = bridge->next) |
|
{ |
|
for (alias = bridge->alias; alias; alias = alias->next) |
|
if (wildcard_matchn(alias->iface, ifr.ifr_name, IF_NAMESIZE)) |
|
{ |
|
parm.ind = if_nametoindex(bridge->iface); |
|
if (!parm.ind) |
|
{ |
|
my_syslog(MS_DHCP | LOG_WARNING, |
|
_("unknown interface %s in bridge-interface"), |
|
bridge->iface); |
|
return; |
|
} |
|
break; |
|
} |
|
if (alias) |
|
break; |
|
} |
|
|
for (context = daemon->dhcp6; context; context = context->next) |
for (context = daemon->dhcp6; context; context = context->next) |
if (IN6_IS_ADDR_UNSPECIFIED(&context->start6) && context->prefix == 0) |
if (IN6_IS_ADDR_UNSPECIFIED(&context->start6) && context->prefix == 0) |
Line 201 void dhcp6_packet(time_t now)
|
Line 220 void dhcp6_packet(time_t now)
|
inet_pton(AF_INET6, ALL_SERVERS, &all_servers); |
inet_pton(AF_INET6, ALL_SERVERS, &all_servers); |
|
|
if (!IN6_ARE_ADDR_EQUAL(&dst_addr, &all_servers)) |
if (!IN6_ARE_ADDR_EQUAL(&dst_addr, &all_servers)) |
relay_upstream6(parm.relay, sz, &from.sin6_addr, from.sin6_scope_id); | relay_upstream6(parm.relay, sz, &from.sin6_addr, from.sin6_scope_id, now); |
return; |
return; |
} |
} |
|
|
/* May have configured relay, but not DHCP server */ |
/* May have configured relay, but not DHCP server */ |
if (!daemon->doing_dhcp6) |
if (!daemon->doing_dhcp6) |
return; |
return; |
| |
lease_prune(NULL, now); /* lose any expired leases */ |
lease_prune(NULL, now); /* lose any expired leases */ |
|
|
port = dhcp6_reply(parm.current, if_index, ifr.ifr_name, &parm.fallback, |
port = dhcp6_reply(parm.current, if_index, ifr.ifr_name, &parm.fallback, |
Line 225 void dhcp6_packet(time_t now)
|
Line 244 void dhcp6_packet(time_t now)
|
if (port != 0) |
if (port != 0) |
{ |
{ |
from.sin6_port = htons(port); |
from.sin6_port = htons(port); |
while (sendto(daemon->dhcp6fd, daemon->outpacket.iov_base, save_counter(0), | while (retry_send(sendto(daemon->dhcp6fd, daemon->outpacket.iov_base, |
0, (struct sockaddr *)&from, sizeof(from)) == -1 && | save_counter(0), 0, (struct sockaddr *)&from, |
retry_send()); | sizeof(from)))); |
} |
} |
} |
} |
|
|
void get_client_mac(struct in6_addr *client, int iface, unsigned char *mac, unsigned int *maclenp, unsigned int *mactypep) | void get_client_mac(struct in6_addr *client, int iface, unsigned char *mac, unsigned int *maclenp, unsigned int *mactypep, time_t now) |
{ |
{ |
/* Recieving a packet from a host does not populate the neighbour |
/* Recieving a packet from a host does not populate the neighbour |
cache, so we send a neighbour discovery request if we can't |
cache, so we send a neighbour discovery request if we can't |
find the sender. Repeat a few times in case of packet loss. */ |
find the sender. Repeat a few times in case of packet loss. */ |
|
|
struct neigh_packet neigh; |
struct neigh_packet neigh; |
struct sockaddr_in6 addr; | union mysockaddr addr; |
struct mac_param mac_param; | int i, maclen; |
int i; | |
|
|
neigh.type = ND_NEIGHBOR_SOLICIT; |
neigh.type = ND_NEIGHBOR_SOLICIT; |
neigh.code = 0; |
neigh.code = 0; |
neigh.reserved = 0; |
neigh.reserved = 0; |
neigh.target = *client; |
neigh.target = *client; |
| /* RFC4443 section-2.3: checksum has to be zero to be calculated */ |
| neigh.checksum = 0; |
| |
memset(&addr, 0, sizeof(addr)); |
memset(&addr, 0, sizeof(addr)); |
#ifdef HAVE_SOCKADDR_SA_LEN |
#ifdef HAVE_SOCKADDR_SA_LEN |
addr.sin6_len = sizeof(struct sockaddr_in6); | addr.in6.sin6_len = sizeof(struct sockaddr_in6); |
#endif |
#endif |
addr.sin6_family = AF_INET6; | addr.in6.sin6_family = AF_INET6; |
addr.sin6_port = htons(IPPROTO_ICMPV6); | addr.in6.sin6_port = htons(IPPROTO_ICMPV6); |
addr.sin6_addr = *client; | addr.in6.sin6_addr = *client; |
addr.sin6_scope_id = iface; | addr.in6.sin6_scope_id = iface; |
|
|
mac_param.target = client; |
|
mac_param.maclen = 0; |
|
mac_param.mac = mac; |
|
|
|
for (i = 0; i < 5; i++) |
for (i = 0; i < 5; i++) |
{ |
{ |
struct timespec ts; |
struct timespec ts; |
|
|
iface_enumerate(AF_UNSPEC, &mac_param, find_mac); | if ((maclen = find_mac(&addr, mac, 0, now)) != 0) |
| |
if (mac_param.maclen != 0) | |
break; |
break; |
|
|
|
sendto(daemon->icmp6fd, &neigh, sizeof(neigh), 0, &addr.sa, sizeof(addr)); |
|
|
sendto(daemon->icmp6fd, &neigh, sizeof(neigh), 0, (struct sockaddr *)&addr, sizeof(addr)); |
|
|
|
ts.tv_sec = 0; |
ts.tv_sec = 0; |
ts.tv_nsec = 100000000; /* 100ms */ |
ts.tv_nsec = 100000000; /* 100ms */ |
nanosleep(&ts, NULL); |
nanosleep(&ts, NULL); |
} |
} |
|
|
*maclenp = mac_param.maclen; | *maclenp = maclen; |
*mactypep = ARPHRD_ETHER; |
*mactypep = ARPHRD_ETHER; |
} |
} |
|
|
static int find_mac(int family, char *addrp, char *mac, size_t maclen, void *parmv) |
|
{ |
|
struct mac_param *parm = parmv; |
|
|
|
if (family == AF_INET6 && IN6_ARE_ADDR_EQUAL(parm->target, (struct in6_addr *)addrp)) |
|
{ |
|
if (maclen <= DHCP_CHADDR_MAX) |
|
{ |
|
parm->maclen = maclen; |
|
memcpy(parm->mac, mac, maclen); |
|
} |
|
|
|
return 0; /* found, abort */ |
|
} |
|
|
|
return 1; |
|
} |
|
|
|
static int complete_context6(struct in6_addr *local, int prefix, |
static int complete_context6(struct in6_addr *local, int prefix, |
int scope, int if_index, int flags, unsigned int preferred, |
int scope, int if_index, int flags, unsigned int preferred, |
unsigned int valid, void *vparam) |
unsigned int valid, void *vparam) |
Line 424 struct dhcp_context *address6_allocate(struct dhcp_con
|
Line 420 struct dhcp_context *address6_allocate(struct dhcp_con
|
j = rand64(); |
j = rand64(); |
else |
else |
for (j = iaid, i = 0; i < clid_len; i++) |
for (j = iaid, i = 0; i < clid_len; i++) |
j += clid[i] + (j << 6) + (j << 16) - j; | j = clid[i] + (j << 6) + (j << 16) - j; |
|
|
for (pass = 0; pass <= plain_range ? 1 : 0; pass++) |
for (pass = 0; pass <= plain_range ? 1 : 0; pass++) |
for (c = context; c; c = c->current) |
for (c = context; c; c = c->current) |
Line 438 struct dhcp_context *address6_allocate(struct dhcp_con
|
Line 434 struct dhcp_context *address6_allocate(struct dhcp_con
|
/* seed is largest extant lease addr in this context */ |
/* seed is largest extant lease addr in this context */ |
start = lease_find_max_addr6(c) + serial; |
start = lease_find_max_addr6(c) + serial; |
else |
else |
start = addr6part(&c->start6) + ((j + c->addr_epoch) % (1 + addr6part(&c->end6) - addr6part(&c->start6))); | { |
| u64 range = 1 + addr6part(&c->end6) - addr6part(&c->start6); |
| u64 offset = j + c->addr_epoch; |
|
|
|
/* don't divide by zero if range is whole 2^64 */ |
|
if (range != 0) |
|
offset = offset % range; |
|
|
|
start = addr6part(&c->start6) + offset; |
|
} |
|
|
/* iterate until we find a free address. */ |
/* iterate until we find a free address. */ |
addr = start; |
addr = start; |
|
|
Line 727 void dhcp_construct_contexts(time_t now)
|
Line 732 void dhcp_construct_contexts(time_t now)
|
|
|
if (context->flags & CONTEXT_GC && !(context->flags & CONTEXT_OLD)) |
if (context->flags & CONTEXT_GC && !(context->flags & CONTEXT_OLD)) |
{ |
{ |
if ((context->flags & (CONTEXT_RA_ONLY | CONTEXT_RA_NAME | CONTEXT_RA_STATELESS)) || | if ((context->flags & CONTEXT_RA) || option_bool(OPT_RA)) |
option_bool(OPT_RA)) | |
{ |
{ |
/* previously constructed context has gone. advertise it's demise */ |
/* previously constructed context has gone. advertise it's demise */ |
context->flags |= CONTEXT_OLD; |
context->flags |= CONTEXT_OLD; |