Commit 1958d581 authored by Vladimir Oltean's avatar Vladimir Oltean Committed by Jakub Kicinski

net: dsa: remove the transactional logic from VLAN objects

It should be the driver's business to logically separate its VLAN
offloading into a preparation and a commit phase, and some drivers don't
need / can't do this.

So remove the transactional shim from DSA and let drivers propagate
errors directly from the .port_vlan_add callback.

It would appear that the code has worse error handling now than it had
before. DSA is the only in-kernel user of switchdev that offloads one
switchdev object to more than one port: for every VLAN object offloaded
to a user port, that VLAN is also offloaded to the CPU port. So the
"prepare for user port -> check for errors -> prepare for CPU port ->
check for errors -> commit for user port -> commit for CPU port"
sequence appears to make more sense than the one we are using now:
"offload to user port -> check for errors -> offload to CPU port ->
check for errors", but it is really a compromise. In the new way, we can
catch errors from the commit phase that we previously had to ignore.
But we have our hands tied and cannot do any rollback now: if we add a
VLAN on the CPU port and it fails, we can't do the rollback by simply
deleting it from the user port, because the switchdev API is not so nice
with us: it could have simply been there already, even with the same
flags. So we don't even attempt to rollback anything on addition error,
just leave whatever VLANs managed to get offloaded right where they are.
This should not be a problem at all in practice.
Signed-off-by: default avatarVladimir Oltean <vladimir.oltean@nxp.com>
Reviewed-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Acked-by: default avatarLinus Walleij <linus.walleij@linaro.org>
Acked-by: default avatarJiri Pirko <jiri@nvidia.com>
Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent a52b2da7
...@@ -1384,7 +1384,7 @@ int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering) ...@@ -1384,7 +1384,7 @@ int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering)
} }
EXPORT_SYMBOL(b53_vlan_filtering); EXPORT_SYMBOL(b53_vlan_filtering);
int b53_vlan_prepare(struct dsa_switch *ds, int port, static int b53_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct b53_device *dev = ds->priv; struct b53_device *dev = ds->priv;
...@@ -1407,15 +1407,19 @@ int b53_vlan_prepare(struct dsa_switch *ds, int port, ...@@ -1407,15 +1407,19 @@ int b53_vlan_prepare(struct dsa_switch *ds, int port,
return 0; return 0;
} }
EXPORT_SYMBOL(b53_vlan_prepare);
void b53_vlan_add(struct dsa_switch *ds, int port, int b53_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct b53_device *dev = ds->priv; struct b53_device *dev = ds->priv;
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID; bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
struct b53_vlan *vl; struct b53_vlan *vl;
int err;
err = b53_vlan_prepare(ds, port, vlan);
if (err)
return err;
vl = &dev->vlans[vlan->vid]; vl = &dev->vlans[vlan->vid];
...@@ -1438,6 +1442,8 @@ void b53_vlan_add(struct dsa_switch *ds, int port, ...@@ -1438,6 +1442,8 @@ void b53_vlan_add(struct dsa_switch *ds, int port,
vlan->vid); vlan->vid);
b53_fast_age_vlan(dev, vlan->vid); b53_fast_age_vlan(dev, vlan->vid);
} }
return 0;
} }
EXPORT_SYMBOL(b53_vlan_add); EXPORT_SYMBOL(b53_vlan_add);
...@@ -2185,7 +2191,6 @@ static const struct dsa_switch_ops b53_switch_ops = { ...@@ -2185,7 +2191,6 @@ static const struct dsa_switch_ops b53_switch_ops = {
.port_fast_age = b53_br_fast_age, .port_fast_age = b53_br_fast_age,
.port_egress_floods = b53_br_egress_floods, .port_egress_floods = b53_br_egress_floods,
.port_vlan_filtering = b53_vlan_filtering, .port_vlan_filtering = b53_vlan_filtering,
.port_vlan_prepare = b53_vlan_prepare,
.port_vlan_add = b53_vlan_add, .port_vlan_add = b53_vlan_add,
.port_vlan_del = b53_vlan_del, .port_vlan_del = b53_vlan_del,
.port_fdb_dump = b53_fdb_dump, .port_fdb_dump = b53_fdb_dump,
......
...@@ -349,9 +349,7 @@ void b53_phylink_mac_link_up(struct dsa_switch *ds, int port, ...@@ -349,9 +349,7 @@ void b53_phylink_mac_link_up(struct dsa_switch *ds, int port,
int speed, int duplex, int speed, int duplex,
bool tx_pause, bool rx_pause); bool tx_pause, bool rx_pause);
int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering); int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering);
int b53_vlan_prepare(struct dsa_switch *ds, int port, int b53_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan);
void b53_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan); const struct switchdev_obj_port_vlan *vlan);
int b53_vlan_del(struct dsa_switch *ds, int port, int b53_vlan_del(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan); const struct switchdev_obj_port_vlan *vlan);
......
...@@ -1116,7 +1116,6 @@ static const struct dsa_switch_ops bcm_sf2_ops = { ...@@ -1116,7 +1116,6 @@ static const struct dsa_switch_ops bcm_sf2_ops = {
.port_stp_state_set = b53_br_set_stp_state, .port_stp_state_set = b53_br_set_stp_state,
.port_fast_age = b53_br_fast_age, .port_fast_age = b53_br_fast_age,
.port_vlan_filtering = b53_vlan_filtering, .port_vlan_filtering = b53_vlan_filtering,
.port_vlan_prepare = b53_vlan_prepare,
.port_vlan_add = b53_vlan_add, .port_vlan_add = b53_vlan_add,
.port_vlan_del = b53_vlan_del, .port_vlan_del = b53_vlan_del,
.port_fdb_dump = b53_fdb_dump, .port_fdb_dump = b53_fdb_dump,
......
...@@ -891,11 +891,9 @@ static int bcm_sf2_cfp_rule_insert(struct dsa_switch *ds, int port, ...@@ -891,11 +891,9 @@ static int bcm_sf2_cfp_rule_insert(struct dsa_switch *ds, int port,
else else
vlan.flags = 0; vlan.flags = 0;
ret = ds->ops->port_vlan_prepare(ds, port_num, &vlan); ret = ds->ops->port_vlan_add(ds, port_num, &vlan);
if (ret) if (ret)
return ret; return ret;
ds->ops->port_vlan_add(ds, port_num, &vlan);
} }
/* /*
...@@ -941,8 +939,7 @@ static int bcm_sf2_cfp_rule_set(struct dsa_switch *ds, int port, ...@@ -941,8 +939,7 @@ static int bcm_sf2_cfp_rule_set(struct dsa_switch *ds, int port,
return -EINVAL; return -EINVAL;
if ((fs->flow_type & FLOW_EXT) && if ((fs->flow_type & FLOW_EXT) &&
!(ds->ops->port_vlan_prepare || ds->ops->port_vlan_add || !(ds->ops->port_vlan_add || ds->ops->port_vlan_del))
ds->ops->port_vlan_del))
return -EOPNOTSUPP; return -EOPNOTSUPP;
if (fs->location != RX_CLS_LOC_ANY && if (fs->location != RX_CLS_LOC_ANY &&
......
...@@ -198,25 +198,7 @@ static int dsa_loop_port_vlan_filtering(struct dsa_switch *ds, int port, ...@@ -198,25 +198,7 @@ static int dsa_loop_port_vlan_filtering(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static int static int dsa_loop_port_vlan_add(struct dsa_switch *ds, int port,
dsa_loop_port_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan)
{
struct dsa_loop_priv *ps = ds->priv;
struct mii_bus *bus = ps->bus;
dev_dbg(ds->dev, "%s: port: %d, vlan: %d", __func__, port, vlan->vid);
/* Just do a sleeping operation to make lockdep checks effective */
mdiobus_read(bus, ps->port_base + port, MII_BMSR);
if (vlan->vid > ARRAY_SIZE(ps->vlans))
return -ERANGE;
return 0;
}
static void dsa_loop_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
...@@ -225,6 +207,9 @@ static void dsa_loop_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -225,6 +207,9 @@ static void dsa_loop_port_vlan_add(struct dsa_switch *ds, int port,
struct mii_bus *bus = ps->bus; struct mii_bus *bus = ps->bus;
struct dsa_loop_vlan *vl; struct dsa_loop_vlan *vl;
if (vlan->vid > ARRAY_SIZE(ps->vlans))
return -ERANGE;
/* Just do a sleeping operation to make lockdep checks effective */ /* Just do a sleeping operation to make lockdep checks effective */
mdiobus_read(bus, ps->port_base + port, MII_BMSR); mdiobus_read(bus, ps->port_base + port, MII_BMSR);
...@@ -241,6 +226,8 @@ static void dsa_loop_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -241,6 +226,8 @@ static void dsa_loop_port_vlan_add(struct dsa_switch *ds, int port,
if (pvid) if (pvid)
ps->ports[port].pvid = vlan->vid; ps->ports[port].pvid = vlan->vid;
return 0;
} }
static int dsa_loop_port_vlan_del(struct dsa_switch *ds, int port, static int dsa_loop_port_vlan_del(struct dsa_switch *ds, int port,
...@@ -300,7 +287,6 @@ static const struct dsa_switch_ops dsa_loop_driver = { ...@@ -300,7 +287,6 @@ static const struct dsa_switch_ops dsa_loop_driver = {
.port_bridge_leave = dsa_loop_port_bridge_leave, .port_bridge_leave = dsa_loop_port_bridge_leave,
.port_stp_state_set = dsa_loop_port_stp_state_set, .port_stp_state_set = dsa_loop_port_stp_state_set,
.port_vlan_filtering = dsa_loop_port_vlan_filtering, .port_vlan_filtering = dsa_loop_port_vlan_filtering,
.port_vlan_prepare = dsa_loop_port_vlan_prepare,
.port_vlan_add = dsa_loop_port_vlan_add, .port_vlan_add = dsa_loop_port_vlan_add,
.port_vlan_del = dsa_loop_port_vlan_del, .port_vlan_del = dsa_loop_port_vlan_del,
.port_change_mtu = dsa_loop_port_change_mtu, .port_change_mtu = dsa_loop_port_change_mtu,
......
...@@ -438,18 +438,25 @@ static void hellcreek_unapply_vlan(struct hellcreek *hellcreek, int port, ...@@ -438,18 +438,25 @@ static void hellcreek_unapply_vlan(struct hellcreek *hellcreek, int port,
mutex_unlock(&hellcreek->reg_lock); mutex_unlock(&hellcreek->reg_lock);
} }
static void hellcreek_vlan_add(struct dsa_switch *ds, int port, static int hellcreek_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID; bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
struct hellcreek *hellcreek = ds->priv; struct hellcreek *hellcreek = ds->priv;
int err;
err = hellcreek_vlan_prepare(ds, port, vlan);
if (err)
return err;
dev_dbg(hellcreek->dev, "Add VLAN %d on port %d, %s, %s\n", dev_dbg(hellcreek->dev, "Add VLAN %d on port %d, %s, %s\n",
vlan->vid, port, untagged ? "untagged" : "tagged", vlan->vid, port, untagged ? "untagged" : "tagged",
pvid ? "PVID" : "no PVID"); pvid ? "PVID" : "no PVID");
hellcreek_apply_vlan(hellcreek, port, vlan->vid, pvid, untagged); hellcreek_apply_vlan(hellcreek, port, vlan->vid, pvid, untagged);
return 0;
} }
static int hellcreek_vlan_del(struct dsa_switch *ds, int port, static int hellcreek_vlan_del(struct dsa_switch *ds, int port,
...@@ -1146,7 +1153,6 @@ static const struct dsa_switch_ops hellcreek_ds_ops = { ...@@ -1146,7 +1153,6 @@ static const struct dsa_switch_ops hellcreek_ds_ops = {
.port_vlan_add = hellcreek_vlan_add, .port_vlan_add = hellcreek_vlan_add,
.port_vlan_del = hellcreek_vlan_del, .port_vlan_del = hellcreek_vlan_del,
.port_vlan_filtering = hellcreek_vlan_filtering, .port_vlan_filtering = hellcreek_vlan_filtering,
.port_vlan_prepare = hellcreek_vlan_prepare,
.setup = hellcreek_setup, .setup = hellcreek_setup,
}; };
......
...@@ -1167,13 +1167,18 @@ static int gswip_port_vlan_prepare(struct dsa_switch *ds, int port, ...@@ -1167,13 +1167,18 @@ static int gswip_port_vlan_prepare(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static void gswip_port_vlan_add(struct dsa_switch *ds, int port, static int gswip_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct gswip_priv *priv = ds->priv; struct gswip_priv *priv = ds->priv;
struct net_device *bridge = dsa_to_port(ds, port)->bridge_dev; struct net_device *bridge = dsa_to_port(ds, port)->bridge_dev;
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID; bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
int err;
err = gswip_port_vlan_prepare(ds, port, vlan);
if (err)
return err;
/* We have to receive all packets on the CPU port and should not /* We have to receive all packets on the CPU port and should not
* do any VLAN filtering here. This is also called with bridge * do any VLAN filtering here. This is also called with bridge
...@@ -1181,9 +1186,10 @@ static void gswip_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -1181,9 +1186,10 @@ static void gswip_port_vlan_add(struct dsa_switch *ds, int port,
* this. * this.
*/ */
if (dsa_is_cpu_port(ds, port)) if (dsa_is_cpu_port(ds, port))
return; return 0;
gswip_vlan_add_aware(priv, bridge, port, vlan->vid, untagged, pvid); return gswip_vlan_add_aware(priv, bridge, port, vlan->vid,
untagged, pvid);
} }
static int gswip_port_vlan_del(struct dsa_switch *ds, int port, static int gswip_port_vlan_del(struct dsa_switch *ds, int port,
...@@ -1580,7 +1586,6 @@ static const struct dsa_switch_ops gswip_switch_ops = { ...@@ -1580,7 +1586,6 @@ static const struct dsa_switch_ops gswip_switch_ops = {
.port_bridge_leave = gswip_port_bridge_leave, .port_bridge_leave = gswip_port_bridge_leave,
.port_fast_age = gswip_port_fast_age, .port_fast_age = gswip_port_fast_age,
.port_vlan_filtering = gswip_port_vlan_filtering, .port_vlan_filtering = gswip_port_vlan_filtering,
.port_vlan_prepare = gswip_port_vlan_prepare,
.port_vlan_add = gswip_port_vlan_add, .port_vlan_add = gswip_port_vlan_add,
.port_vlan_del = gswip_port_vlan_del, .port_vlan_del = gswip_port_vlan_del,
.port_stp_state_set = gswip_port_stp_state_set, .port_stp_state_set = gswip_port_stp_state_set,
......
...@@ -792,7 +792,7 @@ static int ksz8795_port_vlan_filtering(struct dsa_switch *ds, int port, ...@@ -792,7 +792,7 @@ static int ksz8795_port_vlan_filtering(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static void ksz8795_port_vlan_add(struct dsa_switch *ds, int port, static int ksz8795_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
...@@ -828,6 +828,8 @@ static void ksz8795_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -828,6 +828,8 @@ static void ksz8795_port_vlan_add(struct dsa_switch *ds, int port,
vid |= new_pvid; vid |= new_pvid;
ksz_pwrite16(dev, port, REG_PORT_CTRL_VID, vid); ksz_pwrite16(dev, port, REG_PORT_CTRL_VID, vid);
} }
return 0;
} }
static int ksz8795_port_vlan_del(struct dsa_switch *ds, int port, static int ksz8795_port_vlan_del(struct dsa_switch *ds, int port,
...@@ -1110,7 +1112,6 @@ static const struct dsa_switch_ops ksz8795_switch_ops = { ...@@ -1110,7 +1112,6 @@ static const struct dsa_switch_ops ksz8795_switch_ops = {
.port_stp_state_set = ksz8795_port_stp_state_set, .port_stp_state_set = ksz8795_port_stp_state_set,
.port_fast_age = ksz_port_fast_age, .port_fast_age = ksz_port_fast_age,
.port_vlan_filtering = ksz8795_port_vlan_filtering, .port_vlan_filtering = ksz8795_port_vlan_filtering,
.port_vlan_prepare = ksz_port_vlan_prepare,
.port_vlan_add = ksz8795_port_vlan_add, .port_vlan_add = ksz8795_port_vlan_add,
.port_vlan_del = ksz8795_port_vlan_del, .port_vlan_del = ksz8795_port_vlan_del,
.port_fdb_dump = ksz_port_fdb_dump, .port_fdb_dump = ksz_port_fdb_dump,
......
...@@ -510,16 +510,18 @@ static int ksz9477_port_vlan_filtering(struct dsa_switch *ds, int port, ...@@ -510,16 +510,18 @@ static int ksz9477_port_vlan_filtering(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static void ksz9477_port_vlan_add(struct dsa_switch *ds, int port, static int ksz9477_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct ksz_device *dev = ds->priv; struct ksz_device *dev = ds->priv;
u32 vlan_table[3]; u32 vlan_table[3];
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
int err;
if (ksz9477_get_vlan_table(dev, vlan->vid, vlan_table)) { err = ksz9477_get_vlan_table(dev, vlan->vid, vlan_table);
if (err) {
dev_dbg(dev->dev, "Failed to get vlan table\n"); dev_dbg(dev->dev, "Failed to get vlan table\n");
return; return err;
} }
vlan_table[0] = VLAN_VALID | (vlan->vid & VLAN_FID_M); vlan_table[0] = VLAN_VALID | (vlan->vid & VLAN_FID_M);
...@@ -531,14 +533,17 @@ static void ksz9477_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -531,14 +533,17 @@ static void ksz9477_port_vlan_add(struct dsa_switch *ds, int port,
vlan_table[2] |= BIT(port) | BIT(dev->cpu_port); vlan_table[2] |= BIT(port) | BIT(dev->cpu_port);
if (ksz9477_set_vlan_table(dev, vlan->vid, vlan_table)) { err = ksz9477_set_vlan_table(dev, vlan->vid, vlan_table);
if (err) {
dev_dbg(dev->dev, "Failed to set vlan table\n"); dev_dbg(dev->dev, "Failed to set vlan table\n");
return; return err;
} }
/* change PVID */ /* change PVID */
if (vlan->flags & BRIDGE_VLAN_INFO_PVID) if (vlan->flags & BRIDGE_VLAN_INFO_PVID)
ksz_pwrite16(dev, port, REG_PORT_DEFAULT_VID, vlan->vid); ksz_pwrite16(dev, port, REG_PORT_DEFAULT_VID, vlan->vid);
return 0;
} }
static int ksz9477_port_vlan_del(struct dsa_switch *ds, int port, static int ksz9477_port_vlan_del(struct dsa_switch *ds, int port,
...@@ -1394,7 +1399,6 @@ static const struct dsa_switch_ops ksz9477_switch_ops = { ...@@ -1394,7 +1399,6 @@ static const struct dsa_switch_ops ksz9477_switch_ops = {
.port_stp_state_set = ksz9477_port_stp_state_set, .port_stp_state_set = ksz9477_port_stp_state_set,
.port_fast_age = ksz_port_fast_age, .port_fast_age = ksz_port_fast_age,
.port_vlan_filtering = ksz9477_port_vlan_filtering, .port_vlan_filtering = ksz9477_port_vlan_filtering,
.port_vlan_prepare = ksz_port_vlan_prepare,
.port_vlan_add = ksz9477_port_vlan_add, .port_vlan_add = ksz9477_port_vlan_add,
.port_vlan_del = ksz9477_port_vlan_del, .port_vlan_del = ksz9477_port_vlan_del,
.port_fdb_dump = ksz9477_port_fdb_dump, .port_fdb_dump = ksz9477_port_fdb_dump,
......
...@@ -213,15 +213,6 @@ void ksz_port_fast_age(struct dsa_switch *ds, int port) ...@@ -213,15 +213,6 @@ void ksz_port_fast_age(struct dsa_switch *ds, int port)
} }
EXPORT_SYMBOL_GPL(ksz_port_fast_age); EXPORT_SYMBOL_GPL(ksz_port_fast_age);
int ksz_port_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan)
{
/* nothing needed */
return 0;
}
EXPORT_SYMBOL_GPL(ksz_port_vlan_prepare);
int ksz_port_fdb_dump(struct dsa_switch *ds, int port, dsa_fdb_dump_cb_t *cb, int ksz_port_fdb_dump(struct dsa_switch *ds, int port, dsa_fdb_dump_cb_t *cb,
void *data) void *data)
{ {
......
...@@ -161,8 +161,6 @@ int ksz_port_bridge_join(struct dsa_switch *ds, int port, ...@@ -161,8 +161,6 @@ int ksz_port_bridge_join(struct dsa_switch *ds, int port,
void ksz_port_bridge_leave(struct dsa_switch *ds, int port, void ksz_port_bridge_leave(struct dsa_switch *ds, int port,
struct net_device *br); struct net_device *br);
void ksz_port_fast_age(struct dsa_switch *ds, int port); void ksz_port_fast_age(struct dsa_switch *ds, int port);
int ksz_port_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan);
int ksz_port_fdb_dump(struct dsa_switch *ds, int port, dsa_fdb_dump_cb_t *cb, int ksz_port_fdb_dump(struct dsa_switch *ds, int port, dsa_fdb_dump_cb_t *cb,
void *data); void *data);
int ksz_port_mdb_add(struct dsa_switch *ds, int port, int ksz_port_mdb_add(struct dsa_switch *ds, int port,
......
...@@ -1393,15 +1393,6 @@ mt7530_port_vlan_filtering(struct dsa_switch *ds, int port, ...@@ -1393,15 +1393,6 @@ mt7530_port_vlan_filtering(struct dsa_switch *ds, int port,
return 0; return 0;
} }
static int
mt7530_port_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan)
{
/* nothing needed */
return 0;
}
static void static void
mt7530_hw_vlan_add(struct mt7530_priv *priv, mt7530_hw_vlan_add(struct mt7530_priv *priv,
struct mt7530_hw_vlan_entry *entry) struct mt7530_hw_vlan_entry *entry)
...@@ -1489,7 +1480,7 @@ mt7530_hw_vlan_update(struct mt7530_priv *priv, u16 vid, ...@@ -1489,7 +1480,7 @@ mt7530_hw_vlan_update(struct mt7530_priv *priv, u16 vid,
mt7530_vlan_cmd(priv, MT7530_VTCR_WR_VID, vid); mt7530_vlan_cmd(priv, MT7530_VTCR_WR_VID, vid);
} }
static void static int
mt7530_port_vlan_add(struct dsa_switch *ds, int port, mt7530_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
...@@ -1510,6 +1501,8 @@ mt7530_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -1510,6 +1501,8 @@ mt7530_port_vlan_add(struct dsa_switch *ds, int port,
} }
mutex_unlock(&priv->reg_mutex); mutex_unlock(&priv->reg_mutex);
return 0;
} }
static int static int
...@@ -2608,7 +2601,6 @@ static const struct dsa_switch_ops mt7530_switch_ops = { ...@@ -2608,7 +2601,6 @@ static const struct dsa_switch_ops mt7530_switch_ops = {
.port_fdb_del = mt7530_port_fdb_del, .port_fdb_del = mt7530_port_fdb_del,
.port_fdb_dump = mt7530_port_fdb_dump, .port_fdb_dump = mt7530_port_fdb_dump,
.port_vlan_filtering = mt7530_port_vlan_filtering, .port_vlan_filtering = mt7530_port_vlan_filtering,
.port_vlan_prepare = mt7530_port_vlan_prepare,
.port_vlan_add = mt7530_port_vlan_add, .port_vlan_add = mt7530_port_vlan_add,
.port_vlan_del = mt7530_port_vlan_del, .port_vlan_del = mt7530_port_vlan_del,
.port_mirror_add = mt753x_port_mirror_add, .port_mirror_add = mt753x_port_mirror_add,
......
...@@ -1617,9 +1617,6 @@ mv88e6xxx_port_vlan_prepare(struct dsa_switch *ds, int port, ...@@ -1617,9 +1617,6 @@ mv88e6xxx_port_vlan_prepare(struct dsa_switch *ds, int port,
err = mv88e6xxx_port_check_hw_vlan(ds, port, vlan->vid); err = mv88e6xxx_port_check_hw_vlan(ds, port, vlan->vid);
mv88e6xxx_reg_unlock(chip); mv88e6xxx_reg_unlock(chip);
/* We don't need any dynamic resource from the kernel (yet),
* so skip the prepare phase.
*/
return err; return err;
} }
...@@ -1963,7 +1960,7 @@ static int mv88e6xxx_port_vlan_join(struct mv88e6xxx_chip *chip, int port, ...@@ -1963,7 +1960,7 @@ static int mv88e6xxx_port_vlan_join(struct mv88e6xxx_chip *chip, int port,
return 0; return 0;
} }
static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port, static int mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct mv88e6xxx_chip *chip = ds->priv; struct mv88e6xxx_chip *chip = ds->priv;
...@@ -1971,9 +1968,11 @@ static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -1971,9 +1968,11 @@ static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port,
bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID; bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
bool warn; bool warn;
u8 member; u8 member;
int err;
if (!mv88e6xxx_max_vid(chip)) err = mv88e6xxx_port_vlan_prepare(ds, port, vlan);
return; if (err)
return err;
if (dsa_is_dsa_port(ds, port) || dsa_is_cpu_port(ds, port)) if (dsa_is_dsa_port(ds, port) || dsa_is_cpu_port(ds, port))
member = MV88E6XXX_G1_VTU_DATA_MEMBER_TAG_UNMODIFIED; member = MV88E6XXX_G1_VTU_DATA_MEMBER_TAG_UNMODIFIED;
...@@ -1989,15 +1988,25 @@ static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -1989,15 +1988,25 @@ static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port,
mv88e6xxx_reg_lock(chip); mv88e6xxx_reg_lock(chip);
if (mv88e6xxx_port_vlan_join(chip, port, vlan->vid, member, warn)) err = mv88e6xxx_port_vlan_join(chip, port, vlan->vid, member, warn);
if (err) {
dev_err(ds->dev, "p%d: failed to add VLAN %d%c\n", port, dev_err(ds->dev, "p%d: failed to add VLAN %d%c\n", port,
vlan->vid, untagged ? 'u' : 't'); vlan->vid, untagged ? 'u' : 't');
goto out;
}
if (pvid && mv88e6xxx_port_set_pvid(chip, port, vlan->vid)) if (pvid) {
dev_err(ds->dev, "p%d: failed to set PVID %d\n", port, err = mv88e6xxx_port_set_pvid(chip, port, vlan->vid);
vlan->vid); if (err) {
dev_err(ds->dev, "p%d: failed to set PVID %d\n",
port, vlan->vid);
goto out;
}
}
out:
mv88e6xxx_reg_unlock(chip); mv88e6xxx_reg_unlock(chip);
return err;
} }
static int mv88e6xxx_port_vlan_leave(struct mv88e6xxx_chip *chip, static int mv88e6xxx_port_vlan_leave(struct mv88e6xxx_chip *chip,
...@@ -5388,7 +5397,6 @@ static const struct dsa_switch_ops mv88e6xxx_switch_ops = { ...@@ -5388,7 +5397,6 @@ static const struct dsa_switch_ops mv88e6xxx_switch_ops = {
.port_stp_state_set = mv88e6xxx_port_stp_state_set, .port_stp_state_set = mv88e6xxx_port_stp_state_set,
.port_fast_age = mv88e6xxx_port_fast_age, .port_fast_age = mv88e6xxx_port_fast_age,
.port_vlan_filtering = mv88e6xxx_port_vlan_filtering, .port_vlan_filtering = mv88e6xxx_port_vlan_filtering,
.port_vlan_prepare = mv88e6xxx_port_vlan_prepare,
.port_vlan_add = mv88e6xxx_port_vlan_add, .port_vlan_add = mv88e6xxx_port_vlan_add,
.port_vlan_del = mv88e6xxx_port_vlan_del, .port_vlan_del = mv88e6xxx_port_vlan_del,
.port_fdb_add = mv88e6xxx_port_fdb_add, .port_fdb_add = mv88e6xxx_port_fdb_add,
......
...@@ -134,13 +134,18 @@ static int felix_vlan_filtering(struct dsa_switch *ds, int port, bool enabled) ...@@ -134,13 +134,18 @@ static int felix_vlan_filtering(struct dsa_switch *ds, int port, bool enabled)
return ocelot_port_vlan_filtering(ocelot, port, enabled); return ocelot_port_vlan_filtering(ocelot, port, enabled);
} }
static void felix_vlan_add(struct dsa_switch *ds, int port, static int felix_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct ocelot *ocelot = ds->priv; struct ocelot *ocelot = ds->priv;
u16 flags = vlan->flags; u16 flags = vlan->flags;
int err;
err = felix_vlan_prepare(ds, port, vlan);
if (err)
return err;
ocelot_vlan_add(ocelot, port, vlan->vid, return ocelot_vlan_add(ocelot, port, vlan->vid,
flags & BRIDGE_VLAN_INFO_PVID, flags & BRIDGE_VLAN_INFO_PVID,
flags & BRIDGE_VLAN_INFO_UNTAGGED); flags & BRIDGE_VLAN_INFO_UNTAGGED);
} }
...@@ -770,7 +775,6 @@ const struct dsa_switch_ops felix_switch_ops = { ...@@ -770,7 +775,6 @@ const struct dsa_switch_ops felix_switch_ops = {
.port_bridge_join = felix_bridge_join, .port_bridge_join = felix_bridge_join,
.port_bridge_leave = felix_bridge_leave, .port_bridge_leave = felix_bridge_leave,
.port_stp_state_set = felix_bridge_stp_state_set, .port_stp_state_set = felix_bridge_stp_state_set,
.port_vlan_prepare = felix_vlan_prepare,
.port_vlan_filtering = felix_vlan_filtering, .port_vlan_filtering = felix_vlan_filtering,
.port_vlan_add = felix_vlan_add, .port_vlan_add = felix_vlan_add,
.port_vlan_del = felix_vlan_del, .port_vlan_del = felix_vlan_del,
......
...@@ -1312,13 +1312,6 @@ qca8k_port_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering) ...@@ -1312,13 +1312,6 @@ qca8k_port_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering)
} }
static int static int
qca8k_port_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan)
{
return 0;
}
static void
qca8k_port_vlan_add(struct dsa_switch *ds, int port, qca8k_port_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
...@@ -1328,8 +1321,10 @@ qca8k_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -1328,8 +1321,10 @@ qca8k_port_vlan_add(struct dsa_switch *ds, int port,
int ret = 0; int ret = 0;
ret = qca8k_vlan_add(priv, port, vlan->vid, untagged); ret = qca8k_vlan_add(priv, port, vlan->vid, untagged);
if (ret) if (ret) {
dev_err(priv->dev, "Failed to add VLAN to port %d (%d)", port, ret); dev_err(priv->dev, "Failed to add VLAN to port %d (%d)", port, ret);
return ret;
}
if (pvid) { if (pvid) {
int shift = 16 * (port % 2); int shift = 16 * (port % 2);
...@@ -1340,6 +1335,8 @@ qca8k_port_vlan_add(struct dsa_switch *ds, int port, ...@@ -1340,6 +1335,8 @@ qca8k_port_vlan_add(struct dsa_switch *ds, int port,
QCA8K_PORT_VLAN_CVID(vlan->vid) | QCA8K_PORT_VLAN_CVID(vlan->vid) |
QCA8K_PORT_VLAN_SVID(vlan->vid)); QCA8K_PORT_VLAN_SVID(vlan->vid));
} }
return 0;
} }
static int static int
...@@ -1382,7 +1379,6 @@ static const struct dsa_switch_ops qca8k_switch_ops = { ...@@ -1382,7 +1379,6 @@ static const struct dsa_switch_ops qca8k_switch_ops = {
.port_fdb_del = qca8k_port_fdb_del, .port_fdb_del = qca8k_port_fdb_del,
.port_fdb_dump = qca8k_port_fdb_dump, .port_fdb_dump = qca8k_port_fdb_dump,
.port_vlan_filtering = qca8k_port_vlan_filtering, .port_vlan_filtering = qca8k_port_vlan_filtering,
.port_vlan_prepare = qca8k_port_vlan_prepare,
.port_vlan_add = qca8k_port_vlan_add, .port_vlan_add = qca8k_port_vlan_add,
.port_vlan_del = qca8k_port_vlan_del, .port_vlan_del = qca8k_port_vlan_del,
.phylink_validate = qca8k_phylink_validate, .phylink_validate = qca8k_phylink_validate,
......
...@@ -132,9 +132,7 @@ int rtl8366_reset_vlan(struct realtek_smi *smi); ...@@ -132,9 +132,7 @@ int rtl8366_reset_vlan(struct realtek_smi *smi);
int rtl8366_init_vlan(struct realtek_smi *smi); int rtl8366_init_vlan(struct realtek_smi *smi);
int rtl8366_vlan_filtering(struct dsa_switch *ds, int port, int rtl8366_vlan_filtering(struct dsa_switch *ds, int port,
bool vlan_filtering); bool vlan_filtering);
int rtl8366_vlan_prepare(struct dsa_switch *ds, int port, int rtl8366_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan);
void rtl8366_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan); const struct switchdev_obj_port_vlan *vlan);
int rtl8366_vlan_del(struct dsa_switch *ds, int port, int rtl8366_vlan_del(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan); const struct switchdev_obj_port_vlan *vlan);
......
...@@ -374,36 +374,26 @@ int rtl8366_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering) ...@@ -374,36 +374,26 @@ int rtl8366_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering)
} }
EXPORT_SYMBOL_GPL(rtl8366_vlan_filtering); EXPORT_SYMBOL_GPL(rtl8366_vlan_filtering);
int rtl8366_vlan_prepare(struct dsa_switch *ds, int port, int rtl8366_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
bool untagged = !!(vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED);
bool pvid = !!(vlan->flags & BRIDGE_VLAN_INFO_PVID);
struct realtek_smi *smi = ds->priv; struct realtek_smi *smi = ds->priv;
u32 member = 0;
u32 untag = 0;
int ret;
if (!smi->ops->is_vlan_valid(smi, vlan->vid)) if (!smi->ops->is_vlan_valid(smi, vlan->vid))
return -EINVAL; return -EINVAL;
dev_info(smi->dev, "prepare VLAN %04x\n", vlan->vid);
/* Enable VLAN in the hardware /* Enable VLAN in the hardware
* FIXME: what's with this 4k business? * FIXME: what's with this 4k business?
* Just rtl8366_enable_vlan() seems inconclusive. * Just rtl8366_enable_vlan() seems inconclusive.
*/ */
return rtl8366_enable_vlan4k(smi, true); ret = rtl8366_enable_vlan4k(smi, true);
} if (ret)
EXPORT_SYMBOL_GPL(rtl8366_vlan_prepare); return ret;
void rtl8366_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan)
{
bool untagged = !!(vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED);
bool pvid = !!(vlan->flags & BRIDGE_VLAN_INFO_PVID);
struct realtek_smi *smi = ds->priv;
u32 member = 0;
u32 untag = 0;
int ret;
if (!smi->ops->is_vlan_valid(smi, vlan->vid))
return;
dev_info(smi->dev, "add VLAN %d on port %d, %s, %s\n", dev_info(smi->dev, "add VLAN %d on port %d, %s, %s\n",
vlan->vid, port, untagged ? "untagged" : "tagged", vlan->vid, port, untagged ? "untagged" : "tagged",
...@@ -418,20 +408,22 @@ void rtl8366_vlan_add(struct dsa_switch *ds, int port, ...@@ -418,20 +408,22 @@ void rtl8366_vlan_add(struct dsa_switch *ds, int port,
untag |= BIT(port); untag |= BIT(port);
ret = rtl8366_set_vlan(smi, vlan->vid, member, untag, 0); ret = rtl8366_set_vlan(smi, vlan->vid, member, untag, 0);
if (ret) if (ret) {
dev_err(smi->dev, "failed to set up VLAN %04x", vlan->vid); dev_err(smi->dev, "failed to set up VLAN %04x", vlan->vid);
return ret;
}
if (!pvid) if (!pvid)
return; return 0;
ret = rtl8366_set_pvid(smi, port, vlan->vid); ret = rtl8366_set_pvid(smi, port, vlan->vid);
if (ret) if (ret) {
dev_err(smi->dev, "failed to set PVID on port %d to VLAN %04x", dev_err(smi->dev, "failed to set PVID on port %d to VLAN %04x",
port, vlan->vid); port, vlan->vid);
return ret;
}
if (!ret) return 0;
dev_dbg(smi->dev, "VLAN add: added VLAN %d with PVID on port %d\n",
vlan->vid, port);
} }
EXPORT_SYMBOL_GPL(rtl8366_vlan_add); EXPORT_SYMBOL_GPL(rtl8366_vlan_add);
......
...@@ -1504,7 +1504,6 @@ static const struct dsa_switch_ops rtl8366rb_switch_ops = { ...@@ -1504,7 +1504,6 @@ static const struct dsa_switch_ops rtl8366rb_switch_ops = {
.get_ethtool_stats = rtl8366_get_ethtool_stats, .get_ethtool_stats = rtl8366_get_ethtool_stats,
.get_sset_count = rtl8366_get_sset_count, .get_sset_count = rtl8366_get_sset_count,
.port_vlan_filtering = rtl8366_vlan_filtering, .port_vlan_filtering = rtl8366_vlan_filtering,
.port_vlan_prepare = rtl8366_vlan_prepare,
.port_vlan_add = rtl8366_vlan_add, .port_vlan_add = rtl8366_vlan_add,
.port_vlan_del = rtl8366_vlan_del, .port_vlan_del = rtl8366_vlan_del,
.port_enable = rtl8366rb_port_enable, .port_enable = rtl8366rb_port_enable,
......
...@@ -2600,26 +2600,6 @@ static int sja1105_build_vlan_table(struct sja1105_private *priv, bool notify) ...@@ -2600,26 +2600,6 @@ static int sja1105_build_vlan_table(struct sja1105_private *priv, bool notify)
return rc; return rc;
} }
static int sja1105_vlan_prepare(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan)
{
struct sja1105_private *priv = ds->priv;
if (priv->vlan_state == SJA1105_VLAN_FILTERING_FULL)
return 0;
/* If the user wants best-effort VLAN filtering (aka vlan_filtering
* bridge plus tagging), be sure to at least deny alterations to the
* configuration done by dsa_8021q.
*/
if (vid_is_dsa_8021q(vlan->vid)) {
dev_err(ds->dev, "Range 1024-3071 reserved for dsa_8021q operation\n");
return -EBUSY;
}
return 0;
}
/* The TPID setting belongs to the General Parameters table, /* The TPID setting belongs to the General Parameters table,
* which can only be partially reconfigured at runtime (and not the TPID). * which can only be partially reconfigured at runtime (and not the TPID).
* So a switch reset is required. * So a switch reset is required.
...@@ -2779,26 +2759,34 @@ static int sja1105_vlan_del_one(struct dsa_switch *ds, int port, u16 vid, ...@@ -2779,26 +2759,34 @@ static int sja1105_vlan_del_one(struct dsa_switch *ds, int port, u16 vid,
return 0; return 0;
} }
static void sja1105_vlan_add(struct dsa_switch *ds, int port, static int sja1105_vlan_add(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan) const struct switchdev_obj_port_vlan *vlan)
{ {
struct sja1105_private *priv = ds->priv; struct sja1105_private *priv = ds->priv;
bool vlan_table_changed = false; bool vlan_table_changed = false;
int rc; int rc;
/* If the user wants best-effort VLAN filtering (aka vlan_filtering
* bridge plus tagging), be sure to at least deny alterations to the
* configuration done by dsa_8021q.
*/
if (priv->vlan_state != SJA1105_VLAN_FILTERING_FULL &&
vid_is_dsa_8021q(vlan->vid)) {
dev_err(ds->dev, "Range 1024-3071 reserved for dsa_8021q operation\n");
return -EBUSY;
}
rc = sja1105_vlan_add_one(ds, port, vlan->vid, vlan->flags, rc = sja1105_vlan_add_one(ds, port, vlan->vid, vlan->flags,
&priv->bridge_vlans); &priv->bridge_vlans);
if (rc < 0) if (rc < 0)
return; return rc;
if (rc > 0) if (rc > 0)
vlan_table_changed = true; vlan_table_changed = true;
if (!vlan_table_changed) if (!vlan_table_changed)
return; return 0;
rc = sja1105_build_vlan_table(priv, true); return sja1105_build_vlan_table(priv, true);
if (rc)
dev_err(ds->dev, "Failed to build VLAN table: %d\n", rc);
} }
static int sja1105_vlan_del(struct dsa_switch *ds, int port, static int sja1105_vlan_del(struct dsa_switch *ds, int port,
...@@ -3277,7 +3265,6 @@ static const struct dsa_switch_ops sja1105_switch_ops = { ...@@ -3277,7 +3265,6 @@ static const struct dsa_switch_ops sja1105_switch_ops = {
.port_bridge_join = sja1105_bridge_join, .port_bridge_join = sja1105_bridge_join,
.port_bridge_leave = sja1105_bridge_leave, .port_bridge_leave = sja1105_bridge_leave,
.port_stp_state_set = sja1105_bridge_stp_state_set, .port_stp_state_set = sja1105_bridge_stp_state_set,
.port_vlan_prepare = sja1105_vlan_prepare,
.port_vlan_filtering = sja1105_vlan_filtering, .port_vlan_filtering = sja1105_vlan_filtering,
.port_vlan_add = sja1105_vlan_add, .port_vlan_add = sja1105_vlan_add,
.port_vlan_del = sja1105_vlan_del, .port_vlan_del = sja1105_vlan_del,
......
...@@ -566,9 +566,7 @@ struct dsa_switch_ops { ...@@ -566,9 +566,7 @@ struct dsa_switch_ops {
*/ */
int (*port_vlan_filtering)(struct dsa_switch *ds, int port, int (*port_vlan_filtering)(struct dsa_switch *ds, int port,
bool vlan_filtering); bool vlan_filtering);
int (*port_vlan_prepare)(struct dsa_switch *ds, int port, int (*port_vlan_add)(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan);
void (*port_vlan_add)(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan); const struct switchdev_obj_port_vlan *vlan);
int (*port_vlan_del)(struct dsa_switch *ds, int port, int (*port_vlan_del)(struct dsa_switch *ds, int port,
const struct switchdev_obj_port_vlan *vlan); const struct switchdev_obj_port_vlan *vlan);
......
...@@ -227,21 +227,17 @@ static int dsa_switch_vlan_add(struct dsa_switch *ds, ...@@ -227,21 +227,17 @@ static int dsa_switch_vlan_add(struct dsa_switch *ds,
{ {
int port, err; int port, err;
if (!ds->ops->port_vlan_prepare || !ds->ops->port_vlan_add) if (!ds->ops->port_vlan_add)
return -EOPNOTSUPP; return -EOPNOTSUPP;
for (port = 0; port < ds->num_ports; port++) { for (port = 0; port < ds->num_ports; port++) {
if (dsa_switch_vlan_match(ds, port, info)) { if (dsa_switch_vlan_match(ds, port, info)) {
err = ds->ops->port_vlan_prepare(ds, port, info->vlan); err = ds->ops->port_vlan_add(ds, port, info->vlan);
if (err) if (err)
return err; return err;
} }
} }
for (port = 0; port < ds->num_ports; port++)
if (dsa_switch_vlan_match(ds, port, info))
ds->ops->port_vlan_add(ds, port, info->vlan);
return 0; return 0;
} }
......
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