diff options
author | Mike Manning <mmanning@vyatta.att-mail.com> | 2019-04-18 13:35:34 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2019-04-19 16:58:17 -0400 |
commit | 80900acd3a30ed32d65ec591ded5d527d6ba373f (patch) | |
tree | 16b1089ff3506fd0af5a91aa1e7597de74e80345 /net/bridge | |
parent | 9c0ec2e7182a508335364c752da0883a2a7f3999 (diff) |
bridge: update vlan dev state when port added to or deleted from vlan
If vlan bridge binding is enabled, then the link state of a vlan device
that is an upper device of the bridge should track the state of bridge
ports that are members of that vlan. So if a bridge port becomes or
stops being a member of a vlan, then update the link state of the
vlan device if necessary.
Signed-off-by: Mike Manning <mmanning@vyatta.att-mail.com>
Acked-by: Nikolay Aleksandrov <nikolay@cumulusnetworks.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/bridge')
-rw-r--r-- | net/bridge/br_vlan.c | 19 |
1 files changed, 19 insertions, 0 deletions
diff --git a/net/bridge/br_vlan.c b/net/bridge/br_vlan.c index b903689a8fc5..89146a5f0c23 100644 --- a/net/bridge/br_vlan.c +++ b/net/bridge/br_vlan.c | |||
@@ -7,6 +7,8 @@ | |||
7 | #include "br_private.h" | 7 | #include "br_private.h" |
8 | #include "br_private_tunnel.h" | 8 | #include "br_private_tunnel.h" |
9 | 9 | ||
10 | static void nbp_vlan_set_vlan_dev_state(struct net_bridge_port *p, u16 vid); | ||
11 | |||
10 | static inline int br_vlan_cmp(struct rhashtable_compare_arg *arg, | 12 | static inline int br_vlan_cmp(struct rhashtable_compare_arg *arg, |
11 | const void *ptr) | 13 | const void *ptr) |
12 | { | 14 | { |
@@ -293,6 +295,9 @@ static int __vlan_add(struct net_bridge_vlan *v, u16 flags, | |||
293 | 295 | ||
294 | __vlan_add_list(v); | 296 | __vlan_add_list(v); |
295 | __vlan_add_flags(v, flags); | 297 | __vlan_add_flags(v, flags); |
298 | |||
299 | if (p) | ||
300 | nbp_vlan_set_vlan_dev_state(p, v->vid); | ||
296 | out: | 301 | out: |
297 | return err; | 302 | return err; |
298 | 303 | ||
@@ -357,6 +362,7 @@ static int __vlan_del(struct net_bridge_vlan *v) | |||
357 | rhashtable_remove_fast(&vg->vlan_hash, &v->vnode, | 362 | rhashtable_remove_fast(&vg->vlan_hash, &v->vnode, |
358 | br_vlan_rht_params); | 363 | br_vlan_rht_params); |
359 | __vlan_del_list(v); | 364 | __vlan_del_list(v); |
365 | nbp_vlan_set_vlan_dev_state(p, v->vid); | ||
360 | call_rcu(&v->rcu, nbp_vlan_rcu_free); | 366 | call_rcu(&v->rcu, nbp_vlan_rcu_free); |
361 | } | 367 | } |
362 | 368 | ||
@@ -1388,6 +1394,19 @@ static void br_vlan_upper_change(struct net_device *dev, | |||
1388 | } | 1394 | } |
1389 | 1395 | ||
1390 | /* Must be protected by RTNL. */ | 1396 | /* Must be protected by RTNL. */ |
1397 | static void nbp_vlan_set_vlan_dev_state(struct net_bridge_port *p, u16 vid) | ||
1398 | { | ||
1399 | struct net_device *vlan_dev; | ||
1400 | |||
1401 | if (!br_opt_get(p->br, BROPT_VLAN_BRIDGE_BINDING)) | ||
1402 | return; | ||
1403 | |||
1404 | vlan_dev = br_vlan_get_upper_bind_vlan_dev(p->br->dev, vid); | ||
1405 | if (vlan_dev) | ||
1406 | br_vlan_set_vlan_dev_state(p->br, vlan_dev); | ||
1407 | } | ||
1408 | |||
1409 | /* Must be protected by RTNL. */ | ||
1391 | void br_vlan_bridge_event(struct net_device *dev, unsigned long event, | 1410 | void br_vlan_bridge_event(struct net_device *dev, unsigned long event, |
1392 | void *ptr) | 1411 | void *ptr) |
1393 | { | 1412 | { |