summaryrefslogtreecommitdiff
path: root/interface-ip.c
diff options
context:
space:
mode:
authorSteven Barth <steven@midlink.org>2013-04-09 15:41:00 +0200
committerSteven Barth <steven@midlink.org>2013-04-09 15:41:00 +0200
commite0f63be5d2dd6d52d9886231d64040f5ef8142bf (patch)
tree3063cac657f08b57000ac230ffca0d57946d1460 /interface-ip.c
parentcdd7dc012c8f84828cd85722d48180fab8f1f7a5 (diff)
downloadnetifd-e0f63be5d2dd6d52d9886231d64040f5ef8142bf.tar.gz
IPv6: Remove local ULA if there is an external one
Diffstat (limited to 'interface-ip.c')
-rw-r--r--interface-ip.c39
1 files changed, 39 insertions, 0 deletions
diff --git a/interface-ip.c b/interface-ip.c
index bf7cf40..d18fa74 100644
--- a/interface-ip.c
+++ b/interface-ip.c
@@ -476,6 +476,41 @@ interface_update_host_route(struct vlist_tree *tree,
static void
interface_set_prefix_address(struct device_prefix_assignment *assignment,
+ const struct device_prefix *prefix, struct interface *iface, bool add);
+
+static void interface_trigger_ula_prefix(struct interface *iface,
+ const struct device_prefix *prefix, bool enable)
+{
+ if (prefix == ula_prefix || (prefix->addr.s6_addr[0] & 0xfe) != 0xfc)
+ return;
+
+ bool external_ula = false;
+ struct device_prefix_assignment *ula_assign = NULL;
+ struct device_prefix *c;
+ list_for_each_entry(c, &prefixes, head) {
+ if (c != ula_prefix && (c->addr.s6_addr[0] & 0xfe) != 0xfc)
+ continue;
+
+ struct device_prefix_assignment *a;
+ list_for_each_entry(a, &c->assignments, head) {
+ if (!strcmp(a->name, iface->name)) {
+ if (c == ula_prefix)
+ ula_assign = a;
+ else if (a->enabled)
+ external_ula = true;
+ }
+ }
+
+ }
+
+ // Remove ULA assignment if there is an externally managed ULA and vice versa
+ if (ula_assign && ((enable && !external_ula) || (!enable && external_ula)))
+ interface_set_prefix_address(ula_assign, ula_prefix, iface, enable);
+}
+
+
+static void
+interface_set_prefix_address(struct device_prefix_assignment *assignment,
const struct device_prefix *prefix, struct interface *iface, bool add)
{
const struct interface *uplink = prefix->iface;
@@ -501,6 +536,8 @@ interface_set_prefix_address(struct device_prefix_assignment *assignment,
addr.valid_until = now + 7200;
system_add_address(l3_downlink, &addr);
assignment->enabled = false;
+
+ interface_trigger_ula_prefix(iface, prefix, true);
} else if (add && (iface->state == IFS_UP || iface->state == IFS_SETUP)) {
system_add_address(l3_downlink, &addr);
if (uplink && uplink->l3_dev.dev) {
@@ -510,6 +547,8 @@ interface_set_prefix_address(struct device_prefix_assignment *assignment,
system_update_ipv6_mtu(l3_downlink, mtu);
}
assignment->enabled = true;
+
+ interface_trigger_ula_prefix(iface, prefix, false);
}
}