summaryrefslogtreecommitdiffhomepage
path: root/src/dhcpv6-ia.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/dhcpv6-ia.c')
-rw-r--r--src/dhcpv6-ia.c51
1 files changed, 34 insertions, 17 deletions
diff --git a/src/dhcpv6-ia.c b/src/dhcpv6-ia.c
index fa652ab..4ee6dd2 100644
--- a/src/dhcpv6-ia.c
+++ b/src/dhcpv6-ia.c
@@ -41,6 +41,7 @@
static void dhcpv6_netevent_cb(unsigned long event, struct netevent_handler_info *info);
static void free_dhcpv6_assignment(struct dhcpv6_assignment *c);
+static void set_border_assignment_size(struct interface *iface, struct dhcpv6_assignment *b);
static void handle_addrlist_change(struct netevent_handler_info *info);
static void start_reconf(struct dhcpv6_assignment *a);
static void stop_reconf(struct dhcpv6_assignment *a);
@@ -72,11 +73,15 @@ int dhcpv6_setup_ia_interface(struct interface *iface, bool enable)
}
if (enable && iface->dhcpv6 == MODE_SERVER) {
+ struct dhcpv6_assignment *border;
+ struct lease *lease;
+
if (!iface->ia_assignments.next)
INIT_LIST_HEAD(&iface->ia_assignments);
if (list_empty(&iface->ia_assignments)) {
- struct dhcpv6_assignment *border = calloc(1, sizeof(*border));
+ border = calloc(1, sizeof(*border));
+
if (!border) {
syslog(LOG_ERR, "Calloc failed for border on interface %s", iface->ifname);
return -1;
@@ -84,10 +89,12 @@ int dhcpv6_setup_ia_interface(struct interface *iface, bool enable)
border->length = 64;
list_add(&border->head, &iface->ia_assignments);
- }
+ } else
+ border = list_last_entry(&iface->ia_assignments, struct dhcpv6_assignment, head);
+
+ set_border_assignment_size(iface, border);
/* Parse static entries */
- struct lease *lease;
list_for_each_entry(lease, &leases, head) {
/* Construct entry */
size_t duid_len = lease->duid_len ? lease->duid_len : 14;
@@ -489,6 +496,25 @@ static void apply_lease(struct interface *iface, struct dhcpv6_assignment *a, bo
__apply_lease(iface, a, addrs, addrlen, add);
}
+/* Set border assignment size based on the IPv6 address prefixes */
+static void set_border_assignment_size(struct interface *iface, struct dhcpv6_assignment *b)
+{
+ time_t now = odhcpd_time();
+ int minprefix = -1;
+
+ for (size_t i = 0; i < iface->addr6_len; ++i) {
+ if (iface->addr6[i].preferred > (uint32_t)now &&
+ iface->addr6[i].prefix < 64 &&
+ iface->addr6[i].prefix > minprefix)
+ minprefix = iface->addr6[i].prefix;
+ }
+
+ if (minprefix > 32 && minprefix <= 64)
+ b->assigned = 1U << (64 - minprefix);
+ else
+ b->assigned = 0;
+}
+
/* More data was received from TCP connection */
static void managed_handle_pd_data(struct ustream *s, _unused int bytes_new)
{
@@ -695,28 +721,18 @@ static void handle_addrlist_change(struct netevent_handler_info *info)
struct interface *iface = info->iface;
struct dhcpv6_assignment *c, *d, *border = list_last_entry(
&iface->ia_assignments, struct dhcpv6_assignment, head);
+ struct list_head reassign = LIST_HEAD_INIT(reassign);
time_t now = odhcpd_time();
- int minprefix = -1;
- list_for_each_entry(c, &iface->ia_assignments, head)
+ list_for_each_entry(c, &iface->ia_assignments, head) {
if (c != border && iface->ra_managed == RA_MANAGED_NO_MFLAG
&& (c->flags & OAF_BOUND))
__apply_lease(iface, c, info->addrs_old.addrs,
info->addrs_old.len, false);
-
- for (size_t i = 0; i < iface->addr6_len; ++i) {
- if (iface->addr6[i].preferred > (uint32_t)now &&
- iface->addr6[i].prefix < 64 &&
- iface->addr6[i].prefix > minprefix)
- minprefix = iface->addr6[i].prefix;
}
- if (minprefix > 32 && minprefix <= 64)
- border->assigned = 1U << (64 - minprefix);
- else
- border->assigned = 0;
+ set_border_assignment_size(iface, border);
- struct list_head reassign = LIST_HEAD_INIT(reassign);
list_for_each_entry_safe(c, d, &iface->ia_assignments, head) {
if (c->clid_len == 0 || (!INFINITE_VALID(c->valid_until) && c->valid_until < now) ||
c->managed_size)
@@ -728,10 +744,11 @@ static void handle_addrlist_change(struct netevent_handler_info *info)
apply_lease(iface, c, true);
if (c->accept_reconf && c->reconf_cnt == 0) {
+ struct dhcpv6_assignment *a;
+
start_reconf(c);
/* Leave all other assignments of that client alone */
- struct dhcpv6_assignment *a;
list_for_each_entry(a, &iface->ia_assignments, head)
if (a != c && a->clid_len == c->clid_len &&
!memcmp(a->clid_data, c->clid_data, a->clid_len))