X-Git-Url: http://git.archive.openwrt.org/?p=project%2Fnetifd.git;a=blobdiff_plain;f=interface-ip.c;h=bbf2ccffd85dd10a9c035ce43bbc2eec011d140b;hp=2088937b3cfe6479aa84f5a45feb2feffae9ffd2;hb=b7d1c27b752df3e6c5dfe6a5b789d6cb0c87d0f6;hpb=fb8c31126f5837d8d014f31083e97f32aa5063b2 diff --git a/interface-ip.c b/interface-ip.c index 2088937..bbf2ccf 100644 --- a/interface-ip.c +++ b/interface-ip.c @@ -18,7 +18,6 @@ enum { ROUTE_TARGET, ROUTE_MASK, ROUTE_GATEWAY, - ROUTE_DEVICE, ROUTE_METRIC, ROUTE_MTU, __ROUTE_MAX @@ -29,7 +28,6 @@ static const struct blobmsg_policy route_attr[__ROUTE_MAX] = { [ROUTE_TARGET] = { .name = "target", .type = BLOBMSG_TYPE_STRING }, [ROUTE_MASK] = { .name = "netmask", .type = BLOBMSG_TYPE_STRING }, [ROUTE_GATEWAY] = { .name = "gateway", .type = BLOBMSG_TYPE_STRING }, - [ROUTE_DEVICE] = { .name = "device", .type = BLOBMSG_TYPE_STRING }, [ROUTE_METRIC] = { .name = "metric", .type = BLOBMSG_TYPE_INT32 }, [ROUTE_MTU] = { .name = "mtu", .type = BLOBMSG_TYPE_INT32 }, }; @@ -46,13 +44,9 @@ interface_ip_add_route(struct interface *iface, struct blob_attr *attr, bool v6) struct blob_attr *tb[__ROUTE_MAX], *cur; struct device_route *route; int af = v6 ? AF_INET6 : AF_INET; - bool config = false; blobmsg_parse(route_attr, __ROUTE_MAX, tb, blobmsg_data(attr), blobmsg_data_len(attr)); - if (!tb[ROUTE_GATEWAY] && !tb[ROUTE_DEVICE]) - return; - if (!iface) { if ((cur = tb[ROUTE_INTERFACE]) == NULL) return; @@ -62,7 +56,6 @@ interface_ip_add_route(struct interface *iface, struct blob_attr *attr, bool v6) return; ip = &iface->config_ip; - config = true; } else { ip = &iface->proto_ip; } @@ -71,6 +64,7 @@ interface_ip_add_route(struct interface *iface, struct blob_attr *attr, bool v6) if (!route) return; + route->flags = v6 ? DEVADDR_INET6 : DEVADDR_INET4; route->mask = v6 ? 128 : 32; if ((cur = tb[ROUTE_MASK]) != NULL) { route->mask = parse_netmask_string(blobmsg_data(cur), v6); @@ -92,18 +86,15 @@ interface_ip_add_route(struct interface *iface, struct blob_attr *attr, bool v6) } } - if ((cur = tb[ROUTE_METRIC]) != NULL) + if ((cur = tb[ROUTE_METRIC]) != NULL) { route->metric = blobmsg_get_u32(cur); - else - route->metric = -1; + route->flags |= DEVROUTE_METRIC; + } if ((cur = tb[ROUTE_MTU]) != NULL) route->mtu = blobmsg_get_u32(cur); - if (!config && (cur = tb[ROUTE_DEVICE]) != NULL) - route->device = device_get(blobmsg_data(cur), true); - - vlist_add(&ip->route, &route->node); + vlist_add(&ip->route, &route->node, &route->mask); return; error: @@ -132,28 +123,51 @@ interface_update_proto_addr(struct vlist_tree *tree, struct interface_ip_settings *ip; struct interface *iface; struct device *dev; - struct device_addr *addr; + struct device_addr *a_new = NULL, *a_old = NULL; bool keep = false; ip = container_of(tree, struct interface_ip_settings, addr); iface = ip->iface; - dev = iface->l3_dev->dev; + dev = iface->l3_dev.dev; - if (node_old && node_new) + if (node_new) { + a_new = container_of(node_new, struct device_addr, node); + + if ((a_new->flags & DEVADDR_FAMILY) == DEVADDR_INET4 && + !a_new->broadcast) { + + uint32_t mask = ~0; + uint32_t *a = (uint32_t *) &a_new->addr; + + mask >>= a_new->mask; + a_new->broadcast = *a | mask; + } + } + + if (node_old) + a_old = container_of(node_old, struct device_addr, node); + + if (a_new && a_old) { keep = true; + if (a_old->flags != a_new->flags) + keep = false; + + if ((a_new->flags & DEVADDR_FAMILY) == DEVADDR_INET4 && + a_new->broadcast != a_old->broadcast) + keep = false; + } + if (node_old) { - addr = container_of(node_old, struct device_addr, node); - if (!(addr->flags & DEVADDR_EXTERNAL) && addr->enabled && !keep) - system_del_address(dev, addr); - free(addr); + if (!(a_old->flags & DEVADDR_EXTERNAL) && a_old->enabled && !keep) + system_del_address(dev, a_old); + free(a_old); } if (node_new) { - addr = container_of(node_new, struct device_addr, node); - if (!(addr->flags & DEVADDR_EXTERNAL) && !keep) - system_add_address(dev, addr); - addr->enabled = true; + if (!(a_new->flags & DEVADDR_EXTERNAL) && !keep) + system_add_address(dev, a_new); + a_new->enabled = true; } } @@ -179,7 +193,7 @@ interface_update_proto_route(struct vlist_tree *tree, ip = container_of(tree, struct interface_ip_settings, route); iface = ip->iface; - dev = iface->l3_dev->dev; + dev = iface->l3_dev.dev; route_old = container_of(node_old, struct device_route, node); route_new = container_of(node_new, struct device_route, node); @@ -339,7 +353,7 @@ void interface_ip_set_enabled(struct interface_ip_settings *ip, bool enabled) struct device *dev; ip->enabled = enabled; - dev = ip->iface->l3_dev->dev; + dev = ip->iface->l3_dev.dev; if (!dev) return; @@ -363,9 +377,12 @@ void interface_ip_set_enabled(struct interface_ip_settings *ip, bool enabled) if (route->enabled == _enabled) continue; - if (_enabled) + if (_enabled) { + if (!(route->flags & DEVROUTE_METRIC)) + route->metric = ip->iface->metric; + system_add_route(dev, route); - else + } else system_del_route(dev, route); route->enabled = _enabled; } @@ -405,8 +422,6 @@ interface_ip_init(struct interface_ip_settings *ip, struct interface *iface) ip->enabled = true; vlist_simple_init(&ip->dns_search, struct dns_search_domain, node); vlist_simple_init(&ip->dns_servers, struct dns_server, node); - vlist_init(&ip->route, route_cmp, interface_update_proto_route, - struct device_route, node, mask); - vlist_init(&ip->addr, addr_cmp, interface_update_proto_addr, - struct device_addr, node, mask); + vlist_init(&ip->route, route_cmp, interface_update_proto_route); + vlist_init(&ip->addr, addr_cmp, interface_update_proto_addr); }