Commit f545923b authored by Nikolay Aleksandrov's avatar Nikolay Aleksandrov Committed by David S. Miller

net: bridge: vlan: notify on vlan add/delete/change flags

Now that we can notify, send a notification on add/del or change of flags.
Notifications are also compressed when possible to reduce their number
and relieve user-space of extra processing, due to that we have to
manually notify after each add/del in order to avoid double
notifications. We try hard to notify only about the vlans which actually
changed, thus a single command can result in multiple notifications
about disjoint ranges if there were vlans which didn't change inside.
Signed-off-by: default avatarNikolay Aleksandrov <nikolay@cumulusnetworks.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent cf5bddb9
...@@ -568,9 +568,14 @@ int br_process_vlan_info(struct net_bridge *br, ...@@ -568,9 +568,14 @@ int br_process_vlan_info(struct net_bridge *br,
bool *changed, bool *changed,
struct netlink_ext_ack *extack) struct netlink_ext_ack *extack)
{ {
int err, rtm_cmd;
if (!br_vlan_valid_id(vinfo_curr->vid, extack)) if (!br_vlan_valid_id(vinfo_curr->vid, extack))
return -EINVAL; return -EINVAL;
/* needed for vlan-only NEWVLAN/DELVLAN notifications */
rtm_cmd = br_afspec_cmd_to_rtm(cmd);
if (vinfo_curr->flags & BRIDGE_VLAN_INFO_RANGE_BEGIN) { if (vinfo_curr->flags & BRIDGE_VLAN_INFO_RANGE_BEGIN) {
if (!br_vlan_valid_range(vinfo_curr, *vinfo_last, extack)) if (!br_vlan_valid_range(vinfo_curr, *vinfo_last, extack))
return -EINVAL; return -EINVAL;
...@@ -580,7 +585,7 @@ int br_process_vlan_info(struct net_bridge *br, ...@@ -580,7 +585,7 @@ int br_process_vlan_info(struct net_bridge *br,
if (*vinfo_last) { if (*vinfo_last) {
struct bridge_vlan_info tmp_vinfo; struct bridge_vlan_info tmp_vinfo;
int v, err; int v, v_change_start = 0;
if (!br_vlan_valid_range(vinfo_curr, *vinfo_last, extack)) if (!br_vlan_valid_range(vinfo_curr, *vinfo_last, extack))
return -EINVAL; return -EINVAL;
...@@ -588,18 +593,41 @@ int br_process_vlan_info(struct net_bridge *br, ...@@ -588,18 +593,41 @@ int br_process_vlan_info(struct net_bridge *br,
memcpy(&tmp_vinfo, *vinfo_last, memcpy(&tmp_vinfo, *vinfo_last,
sizeof(struct bridge_vlan_info)); sizeof(struct bridge_vlan_info));
for (v = (*vinfo_last)->vid; v <= vinfo_curr->vid; v++) { for (v = (*vinfo_last)->vid; v <= vinfo_curr->vid; v++) {
bool curr_change = false;
tmp_vinfo.vid = v; tmp_vinfo.vid = v;
err = br_vlan_info(br, p, cmd, &tmp_vinfo, changed, err = br_vlan_info(br, p, cmd, &tmp_vinfo, &curr_change,
extack); extack);
if (err) if (err)
break; break;
if (curr_change) {
*changed = curr_change;
if (!v_change_start)
v_change_start = v;
} else {
/* nothing to notify yet */
if (!v_change_start)
continue;
br_vlan_notify(br, p, v_change_start,
v - 1, rtm_cmd);
v_change_start = 0;
} }
}
/* v_change_start is set only if the last/whole range changed */
if (v_change_start)
br_vlan_notify(br, p, v_change_start,
v - 1, rtm_cmd);
*vinfo_last = NULL; *vinfo_last = NULL;
return err; return err;
} }
return br_vlan_info(br, p, cmd, vinfo_curr, changed, extack); err = br_vlan_info(br, p, cmd, vinfo_curr, changed, extack);
if (*changed)
br_vlan_notify(br, p, vinfo_curr->vid, 0, rtm_cmd);
return err;
} }
static int br_afspec(struct net_bridge *br, static int br_afspec(struct net_bridge *br,
......
...@@ -554,6 +554,18 @@ static inline bool br_vlan_valid_range(const struct bridge_vlan_info *cur, ...@@ -554,6 +554,18 @@ static inline bool br_vlan_valid_range(const struct bridge_vlan_info *cur,
return true; return true;
} }
static inline int br_afspec_cmd_to_rtm(int cmd)
{
switch (cmd) {
case RTM_SETLINK:
return RTM_NEWVLAN;
case RTM_DELLINK:
return RTM_DELVLAN;
}
return 0;
}
static inline int br_opt_get(const struct net_bridge *br, static inline int br_opt_get(const struct net_bridge *br,
enum net_bridge_opts opt) enum net_bridge_opts opt)
{ {
......
...@@ -257,6 +257,10 @@ static int __vlan_add(struct net_bridge_vlan *v, u16 flags, ...@@ -257,6 +257,10 @@ static int __vlan_add(struct net_bridge_vlan *v, u16 flags,
&changed, extack); &changed, extack);
if (err) if (err)
goto out_filt; goto out_filt;
if (changed)
br_vlan_notify(br, NULL, v->vid, 0,
RTM_NEWVLAN);
} }
masterv = br_vlan_get_master(br, v->vid, extack); masterv = br_vlan_get_master(br, v->vid, extack);
...@@ -380,13 +384,31 @@ static void __vlan_group_free(struct net_bridge_vlan_group *vg) ...@@ -380,13 +384,31 @@ static void __vlan_group_free(struct net_bridge_vlan_group *vg)
kfree(vg); kfree(vg);
} }
static void __vlan_flush(struct net_bridge_vlan_group *vg) static void __vlan_flush(const struct net_bridge *br,
const struct net_bridge_port *p,
struct net_bridge_vlan_group *vg)
{ {
struct net_bridge_vlan *vlan, *tmp; struct net_bridge_vlan *vlan, *tmp;
u16 v_start = 0, v_end = 0;
__vlan_delete_pvid(vg, vg->pvid); __vlan_delete_pvid(vg, vg->pvid);
list_for_each_entry_safe(vlan, tmp, &vg->vlan_list, vlist) list_for_each_entry_safe(vlan, tmp, &vg->vlan_list, vlist) {
/* take care of disjoint ranges */
if (!v_start) {
v_start = vlan->vid;
} else if (vlan->vid - v_end != 1) {
/* found range end, notify and start next one */
br_vlan_notify(br, p, v_start, v_end, RTM_DELVLAN);
v_start = vlan->vid;
}
v_end = vlan->vid;
__vlan_del(vlan); __vlan_del(vlan);
}
/* notify about the last/whole vlan range */
if (v_start)
br_vlan_notify(br, p, v_start, v_end, RTM_DELVLAN);
} }
struct sk_buff *br_handle_vlan(struct net_bridge *br, struct sk_buff *br_handle_vlan(struct net_bridge *br,
...@@ -716,7 +738,7 @@ void br_vlan_flush(struct net_bridge *br) ...@@ -716,7 +738,7 @@ void br_vlan_flush(struct net_bridge *br)
ASSERT_RTNL(); ASSERT_RTNL();
vg = br_vlan_group(br); vg = br_vlan_group(br);
__vlan_flush(vg); __vlan_flush(br, NULL, vg);
RCU_INIT_POINTER(br->vlgrp, NULL); RCU_INIT_POINTER(br->vlgrp, NULL);
synchronize_rcu(); synchronize_rcu();
__vlan_group_free(vg); __vlan_group_free(vg);
...@@ -925,12 +947,15 @@ static void br_vlan_disable_default_pvid(struct net_bridge *br) ...@@ -925,12 +947,15 @@ static void br_vlan_disable_default_pvid(struct net_bridge *br)
/* Disable default_pvid on all ports where it is still /* Disable default_pvid on all ports where it is still
* configured. * configured.
*/ */
if (vlan_default_pvid(br_vlan_group(br), pvid)) if (vlan_default_pvid(br_vlan_group(br), pvid)) {
br_vlan_delete(br, pvid); if (!br_vlan_delete(br, pvid))
br_vlan_notify(br, NULL, pvid, 0, RTM_DELVLAN);
}
list_for_each_entry(p, &br->port_list, list) { list_for_each_entry(p, &br->port_list, list) {
if (vlan_default_pvid(nbp_vlan_group(p), pvid)) if (vlan_default_pvid(nbp_vlan_group(p), pvid) &&
nbp_vlan_delete(p, pvid); !nbp_vlan_delete(p, pvid))
br_vlan_notify(br, p, pvid, 0, RTM_DELVLAN);
} }
br->default_pvid = 0; br->default_pvid = 0;
...@@ -972,7 +997,10 @@ int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid, ...@@ -972,7 +997,10 @@ int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid,
&vlchange, extack); &vlchange, extack);
if (err) if (err)
goto out; goto out;
br_vlan_delete(br, old_pvid);
if (br_vlan_delete(br, old_pvid))
br_vlan_notify(br, NULL, old_pvid, 0, RTM_DELVLAN);
br_vlan_notify(br, NULL, pvid, 0, RTM_NEWVLAN);
set_bit(0, changed); set_bit(0, changed);
} }
...@@ -992,7 +1020,9 @@ int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid, ...@@ -992,7 +1020,9 @@ int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid,
&vlchange, extack); &vlchange, extack);
if (err) if (err)
goto err_port; goto err_port;
nbp_vlan_delete(p, old_pvid); if (nbp_vlan_delete(p, old_pvid))
br_vlan_notify(br, p, old_pvid, 0, RTM_DELVLAN);
br_vlan_notify(p->br, p, pvid, 0, RTM_NEWVLAN);
set_bit(p->port_no, changed); set_bit(p->port_no, changed);
} }
...@@ -1007,22 +1037,28 @@ int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid, ...@@ -1007,22 +1037,28 @@ int __br_vlan_set_default_pvid(struct net_bridge *br, u16 pvid,
if (!test_bit(p->port_no, changed)) if (!test_bit(p->port_no, changed))
continue; continue;
if (old_pvid) if (old_pvid) {
nbp_vlan_add(p, old_pvid, nbp_vlan_add(p, old_pvid,
BRIDGE_VLAN_INFO_PVID | BRIDGE_VLAN_INFO_PVID |
BRIDGE_VLAN_INFO_UNTAGGED, BRIDGE_VLAN_INFO_UNTAGGED,
&vlchange, NULL); &vlchange, NULL);
br_vlan_notify(p->br, p, old_pvid, 0, RTM_NEWVLAN);
}
nbp_vlan_delete(p, pvid); nbp_vlan_delete(p, pvid);
br_vlan_notify(br, p, pvid, 0, RTM_DELVLAN);
} }
if (test_bit(0, changed)) { if (test_bit(0, changed)) {
if (old_pvid) if (old_pvid) {
br_vlan_add(br, old_pvid, br_vlan_add(br, old_pvid,
BRIDGE_VLAN_INFO_PVID | BRIDGE_VLAN_INFO_PVID |
BRIDGE_VLAN_INFO_UNTAGGED | BRIDGE_VLAN_INFO_UNTAGGED |
BRIDGE_VLAN_INFO_BRENTRY, BRIDGE_VLAN_INFO_BRENTRY,
&vlchange, NULL); &vlchange, NULL);
br_vlan_notify(br, NULL, old_pvid, 0, RTM_NEWVLAN);
}
br_vlan_delete(br, pvid); br_vlan_delete(br, pvid);
br_vlan_notify(br, NULL, pvid, 0, RTM_DELVLAN);
} }
goto out; goto out;
} }
...@@ -1115,6 +1151,7 @@ int nbp_vlan_init(struct net_bridge_port *p, struct netlink_ext_ack *extack) ...@@ -1115,6 +1151,7 @@ int nbp_vlan_init(struct net_bridge_port *p, struct netlink_ext_ack *extack)
&changed, extack); &changed, extack);
if (ret) if (ret)
goto err_vlan_add; goto err_vlan_add;
br_vlan_notify(p->br, p, p->br->default_pvid, 0, RTM_NEWVLAN);
} }
out: out:
return ret; return ret;
...@@ -1196,7 +1233,7 @@ void nbp_vlan_flush(struct net_bridge_port *port) ...@@ -1196,7 +1233,7 @@ void nbp_vlan_flush(struct net_bridge_port *port)
ASSERT_RTNL(); ASSERT_RTNL();
vg = nbp_vlan_group(port); vg = nbp_vlan_group(port);
__vlan_flush(vg); __vlan_flush(port->br, port, vg);
RCU_INIT_POINTER(port->vlgrp, NULL); RCU_INIT_POINTER(port->vlgrp, NULL);
synchronize_rcu(); synchronize_rcu();
__vlan_group_free(vg); __vlan_group_free(vg);
...@@ -1462,8 +1499,8 @@ int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr) ...@@ -1462,8 +1499,8 @@ int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr)
{ {
struct netdev_notifier_changeupper_info *info; struct netdev_notifier_changeupper_info *info;
struct net_bridge *br = netdev_priv(dev); struct net_bridge *br = netdev_priv(dev);
bool changed; int vlcmd = 0, ret = 0;
int ret = 0; bool changed = false;
switch (event) { switch (event) {
case NETDEV_REGISTER: case NETDEV_REGISTER:
...@@ -1471,9 +1508,11 @@ int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr) ...@@ -1471,9 +1508,11 @@ int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr)
BRIDGE_VLAN_INFO_PVID | BRIDGE_VLAN_INFO_PVID |
BRIDGE_VLAN_INFO_UNTAGGED | BRIDGE_VLAN_INFO_UNTAGGED |
BRIDGE_VLAN_INFO_BRENTRY, &changed, NULL); BRIDGE_VLAN_INFO_BRENTRY, &changed, NULL);
vlcmd = RTM_NEWVLAN;
break; break;
case NETDEV_UNREGISTER: case NETDEV_UNREGISTER:
br_vlan_delete(br, br->default_pvid); changed = !br_vlan_delete(br, br->default_pvid);
vlcmd = RTM_DELVLAN;
break; break;
case NETDEV_CHANGEUPPER: case NETDEV_CHANGEUPPER:
info = ptr; info = ptr;
...@@ -1487,6 +1526,8 @@ int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr) ...@@ -1487,6 +1526,8 @@ int br_vlan_bridge_event(struct net_device *dev, unsigned long event, void *ptr)
br_vlan_link_state_change(dev, br); br_vlan_link_state_change(dev, br);
break; break;
} }
if (changed)
br_vlan_notify(br, NULL, br->default_pvid, 0, vlcmd);
return ret; return ret;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment