reset iface->config_state when handling it
[project/netifd.git] / interface-ip.c
index 254e0bb..0558b6c 100644 (file)
@@ -50,23 +50,33 @@ const struct config_param_list route_attr_list = {
        .params = route_attr,
 };
 
-static bool
-match_if_addr(union if_addr *a1, union if_addr *a2, int mask)
+static void
+clear_if_addr(union if_addr *a, int mask)
 {
-       uint8_t *p1, *p2;
        int m_bytes = (mask + 7) / 8;
        uint8_t m_clear = (1 << (m_bytes * 8 - mask)) - 1;
+       uint8_t *p = (uint8_t *) a;
 
-       p1 = alloca(m_bytes);
-       p2 = alloca(m_bytes);
+       if (m_bytes < sizeof(a))
+               memset(p + m_bytes, 0, sizeof(a) - m_bytes);
 
-       memcpy(p1, a1, m_bytes);
-       memcpy(p2, a2, m_bytes);
+       p[m_bytes - 1] &= ~m_clear;
+}
 
-       p1[m_bytes - 1] &= ~m_clear;
-       p2[m_bytes - 1] &= ~m_clear;
+static bool
+match_if_addr(union if_addr *a1, union if_addr *a2, int mask)
+{
+       union if_addr *p1, *p2;
+
+       p1 = alloca(sizeof(*a1));
+       p2 = alloca(sizeof(*a2));
+
+       memcpy(p1, a1, sizeof(*a1));
+       clear_if_addr(p1, mask);
+       memcpy(p2, a2, sizeof(*a2));
+       clear_if_addr(p2, mask);
 
-       return !memcmp(p1, p2, m_bytes);
+       return !memcmp(p1, p2, sizeof(*p1));
 }
 
 static bool
@@ -252,6 +262,7 @@ interface_update_proto_addr(struct vlist_tree *tree,
        struct device *dev;
        struct device_addr *a_new = NULL, *a_old = NULL;
        bool keep = false;
+       struct device_route *route;
 
        ip = container_of(tree, struct interface_ip_settings, addr);
        iface = ip->iface;
@@ -292,10 +303,28 @@ interface_update_proto_addr(struct vlist_tree *tree,
        }
 
        if (node_new) {
-               if (!(a_new->flags & DEVADDR_EXTERNAL) && !keep)
-                       system_add_address(dev, a_new);
                a_new->enabled = true;
+               if (!(a_new->flags & DEVADDR_EXTERNAL) && !keep) {
+                       system_add_address(dev, a_new);
+                       if (iface->metric)
+                               goto replace_route;
+               }
        }
+       return;
+
+replace_route:
+       route = calloc(1, sizeof(*route));
+       route->iface = iface;
+       route->flags = a_new->flags | DEVADDR_KERNEL;
+       route->mask = a_new->mask;
+       memcpy(&route->addr, &a_new->addr, sizeof(route->addr));
+       clear_if_addr(&route->addr, route->mask);
+
+       system_del_route(dev, route);
+
+       route->flags &= ~DEVADDR_KERNEL;
+       route->metric = iface->metric;
+       vlist_add(&ip->route, &route->node, &route->flags);
 }
 
 static bool
@@ -337,6 +366,9 @@ interface_update_proto_route(struct vlist_tree *tree,
        if (node_new) {
                bool _enabled = enable_route(ip, route_new);
 
+               if (!(route_new->flags & DEVROUTE_METRIC))
+                       route_new->metric = iface->metric;
+
                if (!(route_new->flags & DEVADDR_EXTERNAL) && !keep && _enabled)
                        system_add_route(dev, route_new);
 
@@ -489,7 +521,8 @@ interface_write_resolv_conf(void)
 
                fprintf(f, "# Interface %s\n", iface->name);
                write_resolv_conf_entries(f, &iface->config_ip);
-               write_resolv_conf_entries(f, &iface->proto_ip);
+               if (!iface->proto_ip.no_dns)
+                       write_resolv_conf_entries(f, &iface->proto_ip);
        }
        fclose(f);
        if (rename(path, resolv_conf) < 0) {
@@ -543,8 +576,10 @@ void interface_ip_set_enabled(struct interface_ip_settings *ip, bool enabled)
 void
 interface_ip_update_start(struct interface_ip_settings *ip)
 {
-       vlist_simple_update(&ip->dns_servers);
-       vlist_simple_update(&ip->dns_search);
+       if (ip != &ip->iface->config_ip) {
+               vlist_simple_update(&ip->dns_servers);
+               vlist_simple_update(&ip->dns_search);
+       }
        vlist_update(&ip->route);
        vlist_update(&ip->addr);
 }