diff options
author | Petr Machata <petrm@mellanox.com> | 2018-05-29 21:00:45 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2018-05-31 14:13:44 -0400 |
commit | 9c86ce2c1ae337fc10568a12aea812ed03de8319 (patch) | |
tree | 894ecca9ea6eb719e2d569086938ef8b7fa6439f /net/bridge | |
parent | a73bceb86df8f1e007b7f7691d7e38bb29601943 (diff) |
net: bridge: Notify about bridge VLANs
A driver might need to react to changes in settings of brentry VLANs.
Therefore send switchdev port notifications for these as well. Reuse
SWITCHDEV_OBJ_ID_PORT_VLAN for this purpose. Listeners should use
netif_is_bridge_master() on orig_dev to determine whether the
notification is about a bridge port or a bridge.
Signed-off-by: Petr Machata <petrm@mellanox.com>
Reviewed-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 | 28 |
1 files changed, 25 insertions, 3 deletions
diff --git a/net/bridge/br_vlan.c b/net/bridge/br_vlan.c index 602c869c9821..7df269092103 100644 --- a/net/bridge/br_vlan.c +++ b/net/bridge/br_vlan.c | |||
@@ -246,6 +246,10 @@ static int __vlan_add(struct net_bridge_vlan *v, u16 flags) | |||
246 | goto out_filt; | 246 | goto out_filt; |
247 | v->brvlan = masterv; | 247 | v->brvlan = masterv; |
248 | v->stats = masterv->stats; | 248 | v->stats = masterv->stats; |
249 | } else { | ||
250 | err = br_switchdev_port_vlan_add(dev, v->vid, flags); | ||
251 | if (err && err != -EOPNOTSUPP) | ||
252 | goto out; | ||
249 | } | 253 | } |
250 | 254 | ||
251 | /* Add the dev mac and count the vlan only if it's usable */ | 255 | /* Add the dev mac and count the vlan only if it's usable */ |
@@ -281,6 +285,8 @@ out_filt: | |||
281 | br_vlan_put_master(masterv); | 285 | br_vlan_put_master(masterv); |
282 | v->brvlan = NULL; | 286 | v->brvlan = NULL; |
283 | } | 287 | } |
288 | } else { | ||
289 | br_switchdev_port_vlan_del(dev, v->vid); | ||
284 | } | 290 | } |
285 | 291 | ||
286 | goto out; | 292 | goto out; |
@@ -306,6 +312,11 @@ static int __vlan_del(struct net_bridge_vlan *v) | |||
306 | err = __vlan_vid_del(p->dev, p->br, v->vid); | 312 | err = __vlan_vid_del(p->dev, p->br, v->vid); |
307 | if (err) | 313 | if (err) |
308 | goto out; | 314 | goto out; |
315 | } else { | ||
316 | err = br_switchdev_port_vlan_del(v->br->dev, v->vid); | ||
317 | if (err && err != -EOPNOTSUPP) | ||
318 | goto out; | ||
319 | err = 0; | ||
309 | } | 320 | } |
310 | 321 | ||
311 | if (br_vlan_should_use(v)) { | 322 | if (br_vlan_should_use(v)) { |
@@ -558,16 +569,22 @@ static int br_vlan_add_existing(struct net_bridge *br, | |||
558 | { | 569 | { |
559 | int err; | 570 | int err; |
560 | 571 | ||
572 | err = br_switchdev_port_vlan_add(br->dev, vlan->vid, flags); | ||
573 | if (err && err != -EOPNOTSUPP) | ||
574 | return err; | ||
575 | |||
561 | if (!br_vlan_is_brentry(vlan)) { | 576 | if (!br_vlan_is_brentry(vlan)) { |
562 | /* Trying to change flags of non-existent bridge vlan */ | 577 | /* Trying to change flags of non-existent bridge vlan */ |
563 | if (!(flags & BRIDGE_VLAN_INFO_BRENTRY)) | 578 | if (!(flags & BRIDGE_VLAN_INFO_BRENTRY)) { |
564 | return -EINVAL; | 579 | err = -EINVAL; |
580 | goto err_flags; | ||
581 | } | ||
565 | /* It was only kept for port vlans, now make it real */ | 582 | /* It was only kept for port vlans, now make it real */ |
566 | err = br_fdb_insert(br, NULL, br->dev->dev_addr, | 583 | err = br_fdb_insert(br, NULL, br->dev->dev_addr, |
567 | vlan->vid); | 584 | vlan->vid); |
568 | if (err) { | 585 | if (err) { |
569 | br_err(br, "failed to insert local address into bridge forwarding table\n"); | 586 | br_err(br, "failed to insert local address into bridge forwarding table\n"); |
570 | return err; | 587 | goto err_fdb_insert; |
571 | } | 588 | } |
572 | 589 | ||
573 | refcount_inc(&vlan->refcnt); | 590 | refcount_inc(&vlan->refcnt); |
@@ -580,6 +597,11 @@ static int br_vlan_add_existing(struct net_bridge *br, | |||
580 | *changed = true; | 597 | *changed = true; |
581 | 598 | ||
582 | return 0; | 599 | return 0; |
600 | |||
601 | err_fdb_insert: | ||
602 | err_flags: | ||
603 | br_switchdev_port_vlan_del(br->dev, vlan->vid); | ||
604 | return err; | ||
583 | } | 605 | } |
584 | 606 | ||
585 | /* Must be protected by RTNL. | 607 | /* Must be protected by RTNL. |