X-Git-Url: http://git.archive.openwrt.org/?p=project%2Fnetifd.git;a=blobdiff_plain;f=interface.c;h=5bbc0c3f77639d9d275a245ae538fc6248be9266;hp=17cb75418550c228123f999125d83320c84c95be;hb=8527f455f6d35cf9e142cf9981b6553f0e239562;hpb=34a1a4bad31ba6f138cd6a98249c774ff66285d1 diff --git a/interface.c b/interface.c index 17cb754..5bbc0c3 100644 --- a/interface.c +++ b/interface.c @@ -79,12 +79,18 @@ void interface_add_error(struct interface *iface, const char *subsystem, static void interface_event(struct interface *iface, enum interface_event ev) { + struct interface_user *dep, *tmp; + + list_for_each_entry_safe(dep, tmp, &iface->users, list) + dep->cb(dep, IFEV_UP); + interface_queue_event(iface, ev); } static void mark_interface_down(struct interface *iface) { + interface_clear_dns(iface); vlist_flush_all(&iface->proto_addr); vlist_flush_all(&iface->proto_route); if (iface->main_dev.dev) @@ -114,7 +120,6 @@ __interface_set_up(struct interface *iface) } return 0; - } static void @@ -168,6 +173,22 @@ interface_set_available(struct interface *iface, bool new_state) __interface_set_down(iface, true); } +void +interface_add_user(struct interface_user *dep, struct interface *iface) +{ + dep->iface = iface; + list_add(&dep->list, &iface->users); + if (iface->state == IFS_UP) + dep->cb(dep, IFEV_UP); +} + +void +interface_remove_user(struct interface_user *dep) +{ + list_del_init(&dep->list); + dep->iface = NULL; +} + static void interface_claim_device(struct interface *iface) { @@ -185,6 +206,12 @@ interface_claim_device(struct interface *iface) static void interface_cleanup(struct interface *iface) { + struct interface_user *dep, *tmp; + + list_for_each_entry_safe(dep, tmp, &iface->users, list) + interface_remove_user(dep); + + interface_clear_dns(iface); interface_clear_errors(iface); if (iface->main_dev.dev) device_remove_user(&iface->main_dev); @@ -238,16 +265,23 @@ interface_proto_cb(struct interface_proto_state *state, enum interface_proto_eve if (iface->state != IFS_SETUP) return; + system_flush_routes(); iface->state = IFS_UP; iface->start_time = system_get_rtime(); interface_event(iface, IFEV_UP); + interface_write_resolv_conf(); break; case IFPEV_DOWN: if (iface->state == IFS_DOWN) return; + system_flush_routes(); mark_interface_down(iface); interface_handle_config_change(iface); + if (iface->autostart) + __interface_set_up(iface); + if (iface->l3_dev->dev) + device_release(iface->l3_dev); break; case IFPEV_LINK_LOST: if (iface->state != IFS_UP) @@ -284,7 +318,10 @@ interface_init(struct interface *iface, const char *name, strncpy(iface->name, name, sizeof(iface->name) - 1); INIT_LIST_HEAD(&iface->errors); + INIT_LIST_HEAD(&iface->users); INIT_LIST_HEAD(&iface->hotplug_list); + INIT_LIST_HEAD(&iface->proto_dns_search); + INIT_LIST_HEAD(&iface->proto_dns_servers); iface->main_dev.cb = interface_cb; iface->l3_dev = &iface->main_dev; @@ -354,14 +391,15 @@ interface_set_up(struct interface *iface) { iface->autostart = true; + if (iface->state != IFS_DOWN) + return 0; + + interface_clear_errors(iface); if (!iface->available) { interface_add_error(iface, "interface", "NO_DEVICE", NULL, 0); return -1; } - if (iface->state != IFS_DOWN) - return 0; - return __interface_set_up(iface); }