diff options
author | Nikolay Aleksandrov <nikolay@cumulusnetworks.com> | 2015-06-23 08:28:16 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2015-06-24 08:40:55 -0400 |
commit | 1ea2d020ba477cb7011a7174e8501a9e04a325d4 (patch) | |
tree | 2e32130a1c69c146ee95171dd601f2fcf891f3eb | |
parent | 9aa66382163e784acac0ce3580ed202d9a56d1ac (diff) |
bridge: vlan: flush the dynamically learned entries on port vlan delete
Add a new argument to br_fdb_delete_by_port which allows to specify a
vid to match when flushing entries and use it in nbp_vlan_delete() to
flush the dynamically learned entries of the vlan/port pair when removing
a vlan from a port. Before this patch only the local mac was being
removed and the dynamically learned ones were left to expire.
Note that the do_all argument is still respected and if specified, the
vid will be ignored.
Signed-off-by: Nikolay Aleksandrov <nikolay@cumulusnetworks.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | net/bridge/br_fdb.c | 7 | ||||
-rw-r--r-- | net/bridge/br_if.c | 4 | ||||
-rw-r--r-- | net/bridge/br_private.h | 2 | ||||
-rw-r--r-- | net/bridge/br_stp_if.c | 2 | ||||
-rw-r--r-- | net/bridge/br_sysfs_if.c | 2 | ||||
-rw-r--r-- | net/bridge/br_vlan.c | 1 |
6 files changed, 11 insertions, 7 deletions
diff --git a/net/bridge/br_fdb.c b/net/bridge/br_fdb.c index be84b7e5a3da..9e9875da0a4f 100644 --- a/net/bridge/br_fdb.c +++ b/net/bridge/br_fdb.c | |||
@@ -330,9 +330,11 @@ void br_fdb_flush(struct net_bridge *br) | |||
330 | 330 | ||
331 | /* Flush all entries referring to a specific port. | 331 | /* Flush all entries referring to a specific port. |
332 | * if do_all is set also flush static entries | 332 | * if do_all is set also flush static entries |
333 | * if vid is set delete all entries that match the vlan_id | ||
333 | */ | 334 | */ |
334 | void br_fdb_delete_by_port(struct net_bridge *br, | 335 | void br_fdb_delete_by_port(struct net_bridge *br, |
335 | const struct net_bridge_port *p, | 336 | const struct net_bridge_port *p, |
337 | u16 vid, | ||
336 | int do_all) | 338 | int do_all) |
337 | { | 339 | { |
338 | int i; | 340 | int i; |
@@ -347,8 +349,9 @@ void br_fdb_delete_by_port(struct net_bridge *br, | |||
347 | if (f->dst != p) | 349 | if (f->dst != p) |
348 | continue; | 350 | continue; |
349 | 351 | ||
350 | if (f->is_static && !do_all) | 352 | if (!do_all) |
351 | continue; | 353 | if (f->is_static || (vid && f->vlan_id != vid)) |
354 | continue; | ||
352 | 355 | ||
353 | if (f->is_local) | 356 | if (f->is_local) |
354 | fdb_delete_local(br, p, f); | 357 | fdb_delete_local(br, p, f); |
diff --git a/net/bridge/br_if.c b/net/bridge/br_if.c index 1849d96b3c91..a538cb1199a3 100644 --- a/net/bridge/br_if.c +++ b/net/bridge/br_if.c | |||
@@ -249,7 +249,7 @@ static void del_nbp(struct net_bridge_port *p) | |||
249 | list_del_rcu(&p->list); | 249 | list_del_rcu(&p->list); |
250 | 250 | ||
251 | nbp_vlan_flush(p); | 251 | nbp_vlan_flush(p); |
252 | br_fdb_delete_by_port(br, p, 1); | 252 | br_fdb_delete_by_port(br, p, 0, 1); |
253 | nbp_update_port_count(br); | 253 | nbp_update_port_count(br); |
254 | 254 | ||
255 | netdev_upper_dev_unlink(dev, br->dev); | 255 | netdev_upper_dev_unlink(dev, br->dev); |
@@ -278,7 +278,7 @@ void br_dev_delete(struct net_device *dev, struct list_head *head) | |||
278 | del_nbp(p); | 278 | del_nbp(p); |
279 | } | 279 | } |
280 | 280 | ||
281 | br_fdb_delete_by_port(br, NULL, 1); | 281 | br_fdb_delete_by_port(br, NULL, 0, 1); |
282 | 282 | ||
283 | br_vlan_flush(br); | 283 | br_vlan_flush(br); |
284 | del_timer_sync(&br->gc_timer); | 284 | del_timer_sync(&br->gc_timer); |
diff --git a/net/bridge/br_private.h b/net/bridge/br_private.h index 5dccced71269..8b21146b24a0 100644 --- a/net/bridge/br_private.h +++ b/net/bridge/br_private.h | |||
@@ -387,7 +387,7 @@ void br_fdb_changeaddr(struct net_bridge_port *p, const unsigned char *newaddr); | |||
387 | void br_fdb_change_mac_address(struct net_bridge *br, const u8 *newaddr); | 387 | void br_fdb_change_mac_address(struct net_bridge *br, const u8 *newaddr); |
388 | void br_fdb_cleanup(unsigned long arg); | 388 | void br_fdb_cleanup(unsigned long arg); |
389 | void br_fdb_delete_by_port(struct net_bridge *br, | 389 | void br_fdb_delete_by_port(struct net_bridge *br, |
390 | const struct net_bridge_port *p, int do_all); | 390 | const struct net_bridge_port *p, u16 vid, int do_all); |
391 | struct net_bridge_fdb_entry *__br_fdb_get(struct net_bridge *br, | 391 | struct net_bridge_fdb_entry *__br_fdb_get(struct net_bridge *br, |
392 | const unsigned char *addr, __u16 vid); | 392 | const unsigned char *addr, __u16 vid); |
393 | int br_fdb_test_addr(struct net_device *dev, unsigned char *addr); | 393 | int br_fdb_test_addr(struct net_device *dev, unsigned char *addr); |
diff --git a/net/bridge/br_stp_if.c b/net/bridge/br_stp_if.c index 7832d07f48f6..a2730e7196cd 100644 --- a/net/bridge/br_stp_if.c +++ b/net/bridge/br_stp_if.c | |||
@@ -111,7 +111,7 @@ void br_stp_disable_port(struct net_bridge_port *p) | |||
111 | del_timer(&p->forward_delay_timer); | 111 | del_timer(&p->forward_delay_timer); |
112 | del_timer(&p->hold_timer); | 112 | del_timer(&p->hold_timer); |
113 | 113 | ||
114 | br_fdb_delete_by_port(br, p, 0); | 114 | br_fdb_delete_by_port(br, p, 0, 0); |
115 | br_multicast_disable_port(p); | 115 | br_multicast_disable_port(p); |
116 | 116 | ||
117 | br_configuration_update(br); | 117 | br_configuration_update(br); |
diff --git a/net/bridge/br_sysfs_if.c b/net/bridge/br_sysfs_if.c index 4905845a94e9..efe415ad842a 100644 --- a/net/bridge/br_sysfs_if.c +++ b/net/bridge/br_sysfs_if.c | |||
@@ -160,7 +160,7 @@ static BRPORT_ATTR(hold_timer, S_IRUGO, show_hold_timer, NULL); | |||
160 | 160 | ||
161 | static int store_flush(struct net_bridge_port *p, unsigned long v) | 161 | static int store_flush(struct net_bridge_port *p, unsigned long v) |
162 | { | 162 | { |
163 | br_fdb_delete_by_port(p->br, p, 0); // Don't delete local entry | 163 | br_fdb_delete_by_port(p->br, p, 0, 0); // Don't delete local entry |
164 | return 0; | 164 | return 0; |
165 | } | 165 | } |
166 | static BRPORT_ATTR(flush, S_IWUSR, NULL, store_flush); | 166 | static BRPORT_ATTR(flush, S_IWUSR, NULL, store_flush); |
diff --git a/net/bridge/br_vlan.c b/net/bridge/br_vlan.c index 574feea6a8cc..0d41f81838ff 100644 --- a/net/bridge/br_vlan.c +++ b/net/bridge/br_vlan.c | |||
@@ -741,6 +741,7 @@ int nbp_vlan_delete(struct net_bridge_port *port, u16 vid) | |||
741 | return -EINVAL; | 741 | return -EINVAL; |
742 | 742 | ||
743 | br_fdb_find_delete_local(port->br, port, port->dev->dev_addr, vid); | 743 | br_fdb_find_delete_local(port->br, port, port->dev->dev_addr, vid); |
744 | br_fdb_delete_by_port(port->br, port, vid, 0); | ||
744 | 745 | ||
745 | return __vlan_del(pv, vid); | 746 | return __vlan_del(pv, vid); |
746 | } | 747 | } |