X-Git-Url: http://git.archive.openwrt.org/?p=project%2Fnetifd.git;a=blobdiff_plain;f=bridge.c;h=f8478ad698a9f0609c4a08ff7be14063113594d9;hp=d0cb0488b7eb93bab5e7150f48bd31b9a4aabc11;hb=cc8c69f2533e3c85e3631f1371a4207c2d5becce;hpb=f4ea01b966f911442c763743fd6da2401ab84bcb diff --git a/bridge.c b/bridge.c index d0cb048..f8478ad 100644 --- a/bridge.c +++ b/bridge.c @@ -148,6 +148,32 @@ bridge_disable_member(struct bridge_member *bm) } static int +bridge_enable_interface(struct bridge_state *bst) +{ + int ret; + + if (bst->active) + return 0; + + ret = system_bridge_addbr(&bst->dev, &bst->config); + if (ret < 0) + return ret; + + bst->active = true; + return 0; +} + +static void +bridge_disable_interface(struct bridge_state *bst) +{ + if (!bst->active) + return; + + system_bridge_delbr(&bst->dev); + bst->active = false; +} + +static int bridge_enable_member(struct bridge_member *bm) { struct bridge_state *bst = bm->bst; @@ -156,6 +182,10 @@ bridge_enable_member(struct bridge_member *bm) if (!bm->present) return 0; + ret = bridge_enable_interface(bst); + if (ret) + goto error; + /* Disable IPv6 for bridge members */ if (!(bm->dev.dev->settings.flags & DEV_OPT_IPV6)) { bm->dev.dev->settings.ipv6 = 0; @@ -172,6 +202,7 @@ bridge_enable_member(struct bridge_member *bm) goto error; } + device_set_present(&bst->dev, true); device_broadcast_event(&bst->dev, DEV_EVENT_TOPO_CHANGE); return 0; @@ -180,6 +211,8 @@ error: bst->n_failed++; bm->present = false; bst->n_present--; + device_release(&bm->dev); + return ret; } @@ -291,7 +324,7 @@ bridge_set_down(struct bridge_state *bst) vlist_for_each_element(&bst->members, bm, node) bridge_disable_member(bm); - system_bridge_delbr(&bst->dev); + bridge_disable_interface(bst); return 0; } @@ -302,12 +335,14 @@ bridge_set_up(struct bridge_state *bst) struct bridge_member *bm; int ret; - if (!bst->force_active && !bst->n_present) - return -ENOENT; + if (!bst->n_present) { + if (!bst->force_active) + return -ENOENT; - ret = system_bridge_addbr(&bst->dev, &bst->config); - if (ret < 0) - goto out; + ret = bridge_enable_interface(bst); + if (ret) + return ret; + } bst->n_failed = 0; vlist_for_each_element(&bst->members, bm, node) @@ -316,7 +351,7 @@ bridge_set_up(struct bridge_state *bst) if (!bst->force_active && !bst->n_present) { /* initialization of all member interfaces failed */ - system_bridge_delbr(&bst->dev); + bridge_disable_interface(bst); device_set_present(&bst->dev, false); return -ENOENT; } @@ -326,7 +361,6 @@ bridge_set_up(struct bridge_state *bst) if (ret < 0) bridge_set_down(bst); -out: return ret; } @@ -612,6 +646,7 @@ bridge_retry_members(struct uloop_timeout *timeout) continue; bm->present = true; + bst->n_present++; bridge_enable_member(bm); } }