summaryrefslogtreecommitdiffstats
path: root/net/bridge
diff options
context:
space:
mode:
authorMike Manning <mmanning@vyatta.att-mail.com>2019-04-18 13:35:34 -0400
committerDavid S. Miller <davem@davemloft.net>2019-04-19 16:58:17 -0400
commit80900acd3a30ed32d65ec591ded5d527d6ba373f (patch)
tree16b1089ff3506fd0af5a91aa1e7597de74e80345 /net/bridge
parent9c0ec2e7182a508335364c752da0883a2a7f3999 (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.c19
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
10static void nbp_vlan_set_vlan_dev_state(struct net_bridge_port *p, u16 vid);
11
10static inline int br_vlan_cmp(struct rhashtable_compare_arg *arg, 12static 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);
296out: 301out:
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. */
1397static 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. */
1391void br_vlan_bridge_event(struct net_device *dev, unsigned long event, 1410void br_vlan_bridge_event(struct net_device *dev, unsigned long event,
1392 void *ptr) 1411 void *ptr)
1393{ 1412{