Skip to content

Commit ea08dfc

Browse files
vladimirolteankuba-moo
authored andcommitted
net: dsa: mv88e6xxx: fix -ENOENT when deleting VLANs and MST is unsupported
Russell King reports that on the ZII dev rev B, deleting a bridge VLAN from a user port fails with -ENOENT: https://lore.kernel.org/netdev/[email protected]/ This comes from mv88e6xxx_port_vlan_leave() -> mv88e6xxx_mst_put(), which tries to find an MST entry in &chip->msts associated with the SID, but fails and returns -ENOENT as such. But we know that this chip does not support MST at all, so that is not surprising. The question is why does the guard in mv88e6xxx_mst_put() not exit early: if (!sid) return 0; And the answer seems to be simple: the sid comes from vlan.sid which supposedly was previously populated by mv88e6xxx_vtu_get(). But some chip->info->ops->vtu_getnext() implementations do not populate vlan.sid, for example see mv88e6185_g1_vtu_getnext(). In that case, later in mv88e6xxx_port_vlan_leave() we are using a garbage sid which is just residual stack memory. Testing for sid == 0 covers all cases of a non-bridge VLAN or a bridge VLAN mapped to the default MSTI. For some chips, SID 0 is valid and installed by mv88e6xxx_stu_setup(). A chip which does not support the STU would implicitly only support mapping all VLANs to the default MSTI, so although SID 0 is not valid, it would be sufficient, if we were to zero-initialize the vlan structure, to fix the bug, due to the coincidence that a test for vlan.sid == 0 already exists and leads to the same (correct) behavior. Another option which would be sufficient would be to add a test for mv88e6xxx_has_stu() inside mv88e6xxx_mst_put(), symmetric to the one which already exists in mv88e6xxx_mst_get(). But that placement means the caller will have to dereference vlan.sid, which means it will access uninitialized memory, which is not nice even if it ignores it later. So we end up making both modifications, in order to not rely just on the sid == 0 coincidence, but also to avoid having uninitialized structure fields which might get temporarily accessed. Fixes: acaf4d2 ("net: dsa: mv88e6xxx: MST Offloading") Signed-off-by: Vladimir Oltean <[email protected]> Link: https://patch.msgid.link/[email protected] Signed-off-by: Jakub Kicinski <[email protected]>
1 parent c84f6ce commit ea08dfc

File tree

1 file changed

+12
-1
lines changed
  • drivers/net/dsa/mv88e6xxx

1 file changed

+12
-1
lines changed

drivers/net/dsa/mv88e6xxx/chip.c

Lines changed: 12 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1852,6 +1852,8 @@ static int mv88e6xxx_vtu_get(struct mv88e6xxx_chip *chip, u16 vid,
18521852
if (!chip->info->ops->vtu_getnext)
18531853
return -EOPNOTSUPP;
18541854

1855+
memset(entry, 0, sizeof(*entry));
1856+
18551857
entry->vid = vid ? vid - 1 : mv88e6xxx_max_vid(chip);
18561858
entry->valid = false;
18571859

@@ -1960,7 +1962,16 @@ static int mv88e6xxx_mst_put(struct mv88e6xxx_chip *chip, u8 sid)
19601962
struct mv88e6xxx_mst *mst, *tmp;
19611963
int err;
19621964

1963-
if (!sid)
1965+
/* If the SID is zero, it is for a VLAN mapped to the default MSTI,
1966+
* and mv88e6xxx_stu_setup() made sure it is always present, and thus,
1967+
* should not be removed here.
1968+
*
1969+
* If the chip lacks STU support, numerically the "sid" variable will
1970+
* happen to also be zero, but we don't want to rely on that fact, so
1971+
* we explicitly test that first. In that case, there is also nothing
1972+
* to do here.
1973+
*/
1974+
if (!mv88e6xxx_has_stu(chip) || !sid)
19641975
return 0;
19651976

19661977
list_for_each_entry_safe(mst, tmp, &chip->msts, node) {

0 commit comments

Comments
 (0)