]> www.infradead.org Git - nvme.git/commitdiff
net: dsa: b53: fix toggling vlan_filtering
authorJonas Gorski <jonas.gorski@gmail.com>
Tue, 29 Apr 2025 20:17:08 +0000 (22:17 +0200)
committerJakub Kicinski <kuba@kernel.org>
Thu, 8 May 2025 02:30:35 +0000 (19:30 -0700)
To allow runtime switching between vlan aware and vlan non-aware mode,
we need to properly keep track of any bridge VLAN configuration.
Likewise, we need to know when we actually switch between both modes, to
not have to rewrite the full VLAN table every time we update the VLANs.

So keep track of the current vlan_filtering mode, and on changes, apply
the appropriate VLAN configuration.

Fixes: 0ee2af4ebbe3 ("net: dsa: set configure_vlan_while_not_filtering to true by default")
Signed-off-by: Jonas Gorski <jonas.gorski@gmail.com>
Tested-by: Florian Fainelli <florian.fainelli@broadcom.com>
Reviewed-by: Florian Fainelli <florian.fainelli@broadcom.com>
Link: https://patch.msgid.link/20250429201710.330937-10-jonas.gorski@gmail.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/dsa/b53/b53_common.c
drivers/net/dsa/b53/b53_priv.h

index ee2f1be626184c09524795dbee3f99c7a18be7fc..0a7749b416f7775700624496bb5427d8dbe4b14b 100644 (file)
@@ -763,6 +763,22 @@ static bool b53_vlan_port_needs_forced_tagged(struct dsa_switch *ds, int port)
        return dev->tag_protocol == DSA_TAG_PROTO_NONE && dsa_is_cpu_port(ds, port);
 }
 
+static bool b53_vlan_port_may_join_untagged(struct dsa_switch *ds, int port)
+{
+       struct b53_device *dev = ds->priv;
+       struct dsa_port *dp;
+
+       if (!dev->vlan_filtering)
+               return true;
+
+       dp = dsa_to_port(ds, port);
+
+       if (dsa_port_is_cpu(dp))
+               return true;
+
+       return dp->bridge == NULL;
+}
+
 int b53_configure_vlan(struct dsa_switch *ds)
 {
        struct b53_device *dev = ds->priv;
@@ -781,7 +797,7 @@ int b53_configure_vlan(struct dsa_switch *ds)
                b53_do_vlan_op(dev, VTA_CMD_CLEAR);
        }
 
-       b53_enable_vlan(dev, -1, dev->vlan_enabled, ds->vlan_filtering);
+       b53_enable_vlan(dev, -1, dev->vlan_enabled, dev->vlan_filtering);
 
        /* Create an untagged VLAN entry for the default PVID in case
         * CONFIG_VLAN_8021Q is disabled and there are no calls to
@@ -789,26 +805,39 @@ int b53_configure_vlan(struct dsa_switch *ds)
         * entry. Do this only when the tagging protocol is not
         * DSA_TAG_PROTO_NONE
         */
+       v = &dev->vlans[def_vid];
        b53_for_each_port(dev, i) {
-               v = &dev->vlans[def_vid];
-               v->members |= BIT(i);
+               if (!b53_vlan_port_may_join_untagged(ds, i))
+                       continue;
+
+               vl.members |= BIT(i);
                if (!b53_vlan_port_needs_forced_tagged(ds, i))
-                       v->untag = v->members;
-               b53_write16(dev, B53_VLAN_PAGE,
-                           B53_VLAN_PORT_DEF_TAG(i), def_vid);
+                       vl.untag = vl.members;
+               b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(i),
+                           def_vid);
        }
+       b53_set_vlan_entry(dev, def_vid, &vl);
 
-       /* Upon initial call we have not set-up any VLANs, but upon
-        * system resume, we need to restore all VLAN entries.
-        */
-       for (vid = def_vid; vid < dev->num_vlans; vid++) {
-               v = &dev->vlans[vid];
+       if (dev->vlan_filtering) {
+               /* Upon initial call we have not set-up any VLANs, but upon
+                * system resume, we need to restore all VLAN entries.
+                */
+               for (vid = def_vid + 1; vid < dev->num_vlans; vid++) {
+                       v = &dev->vlans[vid];
 
-               if (!v->members)
-                       continue;
+                       if (!v->members)
+                               continue;
+
+                       b53_set_vlan_entry(dev, vid, v);
+                       b53_fast_age_vlan(dev, vid);
+               }
 
-               b53_set_vlan_entry(dev, vid, v);
-               b53_fast_age_vlan(dev, vid);
+               b53_for_each_port(dev, i) {
+                       if (!dsa_is_cpu_port(ds, i))
+                               b53_write16(dev, B53_VLAN_PAGE,
+                                           B53_VLAN_PORT_DEF_TAG(i),
+                                           dev->ports[i].pvid);
+               }
        }
 
        return 0;
@@ -1127,7 +1156,9 @@ EXPORT_SYMBOL(b53_setup_devlink_resources);
 static int b53_setup(struct dsa_switch *ds)
 {
        struct b53_device *dev = ds->priv;
+       struct b53_vlan *vl;
        unsigned int port;
+       u16 pvid;
        int ret;
 
        /* Request bridge PVID untagged when DSA_TAG_PROTO_NONE is set
@@ -1146,6 +1177,15 @@ static int b53_setup(struct dsa_switch *ds)
                return ret;
        }
 
+       /* setup default vlan for filtering mode */
+       pvid = b53_default_pvid(dev);
+       vl = &dev->vlans[pvid];
+       b53_for_each_port(dev, port) {
+               vl->members |= BIT(port);
+               if (!b53_vlan_port_needs_forced_tagged(ds, port))
+                       vl->untag |= BIT(port);
+       }
+
        b53_reset_mib(dev);
 
        ret = b53_apply_config(dev);
@@ -1499,7 +1539,10 @@ int b53_vlan_filtering(struct dsa_switch *ds, int port, bool vlan_filtering,
 {
        struct b53_device *dev = ds->priv;
 
-       b53_enable_vlan(dev, port, dev->vlan_enabled, vlan_filtering);
+       if (dev->vlan_filtering != vlan_filtering) {
+               dev->vlan_filtering = vlan_filtering;
+               b53_apply_config(dev);
+       }
 
        return 0;
 }
@@ -1524,7 +1567,7 @@ static int b53_vlan_prepare(struct dsa_switch *ds, int port,
        if (vlan->vid >= dev->num_vlans)
                return -ERANGE;
 
-       b53_enable_vlan(dev, port, true, ds->vlan_filtering);
+       b53_enable_vlan(dev, port, true, dev->vlan_filtering);
 
        return 0;
 }
@@ -1547,21 +1590,17 @@ int b53_vlan_add(struct dsa_switch *ds, int port,
        if (vlan->vid == 0)
                return 0;
 
-       if (!ds->vlan_filtering)
-               return 0;
-
-       b53_read16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(port), &old_pvid);
+       old_pvid = dev->ports[port].pvid;
        if (pvid)
                new_pvid = vlan->vid;
        else if (!pvid && vlan->vid == old_pvid)
                new_pvid = b53_default_pvid(dev);
        else
                new_pvid = old_pvid;
+       dev->ports[port].pvid = new_pvid;
 
        vl = &dev->vlans[vlan->vid];
 
-       b53_get_vlan_entry(dev, vlan->vid, vl);
-
        if (dsa_is_cpu_port(ds, port))
                untagged = false;
 
@@ -1571,6 +1610,9 @@ int b53_vlan_add(struct dsa_switch *ds, int port,
        else
                vl->untag &= ~BIT(port);
 
+       if (!dev->vlan_filtering)
+               return 0;
+
        b53_set_vlan_entry(dev, vlan->vid, vl);
        b53_fast_age_vlan(dev, vlan->vid);
 
@@ -1595,23 +1637,22 @@ int b53_vlan_del(struct dsa_switch *ds, int port,
        if (vlan->vid == 0)
                return 0;
 
-       if (!ds->vlan_filtering)
-               return 0;
-
-       b53_read16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(port), &pvid);
+       pvid = dev->ports[port].pvid;
 
        vl = &dev->vlans[vlan->vid];
 
-       b53_get_vlan_entry(dev, vlan->vid, vl);
-
        vl->members &= ~BIT(port);
 
        if (pvid == vlan->vid)
                pvid = b53_default_pvid(dev);
+       dev->ports[port].pvid = pvid;
 
        if (untagged && !b53_vlan_port_needs_forced_tagged(ds, port))
                vl->untag &= ~(BIT(port));
 
+       if (!dev->vlan_filtering)
+               return 0;
+
        b53_set_vlan_entry(dev, vlan->vid, vl);
        b53_fast_age_vlan(dev, vlan->vid);
 
@@ -1958,7 +1999,7 @@ int b53_br_join(struct dsa_switch *ds, int port, struct dsa_bridge bridge,
        pvid = b53_default_pvid(dev);
        vl = &dev->vlans[pvid];
 
-       if (ds->vlan_filtering) {
+       if (dev->vlan_filtering) {
                /* Make this port leave the all VLANs join since we will have
                 * proper VLAN entries from now on
                 */
@@ -2038,7 +2079,7 @@ void b53_br_leave(struct dsa_switch *ds, int port, struct dsa_bridge bridge)
        pvid = b53_default_pvid(dev);
        vl = &dev->vlans[pvid];
 
-       if (ds->vlan_filtering) {
+       if (dev->vlan_filtering) {
                /* Make this port join all VLANs without VLAN entries */
                if (is58xx(dev)) {
                        b53_read16(dev, B53_VLAN_PAGE, B53_JOIN_ALL_VLAN_EN, &reg);
@@ -2803,6 +2844,7 @@ struct b53_device *b53_switch_alloc(struct device *base,
        ds->ops = &b53_switch_ops;
        ds->phylink_mac_ops = &b53_phylink_mac_ops;
        dev->vlan_enabled = true;
+       dev->vlan_filtering = false;
        /* Let DSA handle the case were multiple bridges span the same switch
         * device and different VLAN awareness settings are requested, which
         * would be breaking filtering semantics for any of the other bridge
index 0166c37a13a7fcbbe7c009bc6250ea0afbf1072d..4636e27fd1ee91cfb611f9952059f85a868551d2 100644 (file)
@@ -96,6 +96,7 @@ struct b53_pcs {
 
 struct b53_port {
        u16             vlan_ctl_mask;
+       u16             pvid;
        struct ethtool_keee eee;
 };
 
@@ -147,6 +148,7 @@ struct b53_device {
        unsigned int num_vlans;
        struct b53_vlan *vlans;
        bool vlan_enabled;
+       bool vlan_filtering;
        unsigned int num_ports;
        struct b53_port *ports;