c5b006493c1dc297f50027dadff11552613b0ab7
[openwrt/staging/svanheule.git] /
1 From 675992be6f7b603b8cfda4678f173e1021fc1ab6 Mon Sep 17 00:00:00 2001
2 From: Vladimir Oltean <vladimir.oltean@nxp.com>
3 Date: Thu, 7 Oct 2021 19:47:10 +0300
4 Subject: [PATCH] net: dsa: mv88e6xxx: keep the pvid at 0 when VLAN-unaware
5
6 The VLAN support in mv88e6xxx has a loaded history. Commit 2ea7a679ca2a
7 ("net: dsa: Don't add vlans when vlan filtering is disabled") noticed
8 some issues with VLAN and decided the best way to deal with them was to
9 make the DSA core ignore VLANs added by the bridge while VLAN awareness
10 is turned off. Those issues were never explained, just presented as
11 "at least one corner case".
12
13 That approach had problems of its own, presented by
14 commit 54a0ed0df496 ("net: dsa: provide an option for drivers to always
15 receive bridge VLANs") for the DSA core, followed by
16 commit 1fb74191988f ("net: dsa: mv88e6xxx: fix vlan setup") which
17 applied ds->configure_vlan_while_not_filtering = true for mv88e6xxx in
18 particular.
19
20 We still don't know what corner case Andrew saw when he wrote
21 commit 2ea7a679ca2a ("net: dsa: Don't add vlans when vlan filtering is
22 disabled"), but Tobias now reports that when we use TX forwarding
23 offload, pinging an external station from the bridge device is broken if
24 the front-facing DSA user port has flooding turned off. The full
25 description is in the link below, but for short, when a mv88e6xxx port
26 is under a VLAN-unaware bridge, it inherits that bridge's pvid.
27 So packets ingressing a user port will be classified to e.g. VID 1
28 (assuming that value for the bridge_default_pvid), whereas when
29 tag_dsa.c xmits towards a user port, it always sends packets using a VID
30 of 0 if that port is standalone or under a VLAN-unaware bridge - or at
31 least it did so prior to commit d82f8ab0d874 ("net: dsa: tag_dsa:
32 offload the bridge forwarding process").
33
34 In any case, when there is a conversation between the CPU and a station
35 connected to a user port, the station's MAC address is learned in VID 1
36 but the CPU tries to transmit through VID 0. The packets reach the
37 intended station, but via flooding and not by virtue of matching the
38 existing ATU entry.
39
40 DSA has established (and enforced in other drivers: sja1105, felix,
41 mt7530) that a VLAN-unaware port should use a private pvid, and not
42 inherit the one from the bridge. The bridge's pvid should only be
43 inherited when that bridge is VLAN-aware, so all state transitions need
44 to be handled. On the other hand, all bridge VLANs should sit in the VTU
45 starting with the moment when the bridge offloads them via switchdev,
46 they are just not used.
47
48 This solves the problem that Tobias sees because packets ingressing on
49 VLAN-unaware user ports now get classified to VID 0, which is also the
50 VID used by tag_dsa.c on xmit.
51
52 Fixes: d82f8ab0d874 ("net: dsa: tag_dsa: offload the bridge forwarding process")
53 Link: https://patchwork.kernel.org/project/netdevbpf/patch/20211003222312.284175-2-vladimir.oltean@nxp.com/#24491503
54 Reported-by: Tobias Waldekranz <tobias@waldekranz.com>
55 Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com>
56 Signed-off-by: Jakub Kicinski <kuba@kernel.org>
57 ---
58 drivers/net/dsa/mv88e6xxx/chip.c | 56 +++++++++++++++++++++++++++++---
59 drivers/net/dsa/mv88e6xxx/chip.h | 6 ++++
60 drivers/net/dsa/mv88e6xxx/port.c | 21 ++++++++++++
61 drivers/net/dsa/mv88e6xxx/port.h | 2 ++
62 4 files changed, 81 insertions(+), 4 deletions(-)
63
64 diff --git a/drivers/net/dsa/mv88e6xxx/chip.c b/drivers/net/dsa/mv88e6xxx/chip.c
65 index 1992be77522a..a77c86e0321d 100644
66 --- a/drivers/net/dsa/mv88e6xxx/chip.c
67 +++ b/drivers/net/dsa/mv88e6xxx/chip.c
68 @@ -1586,6 +1586,26 @@ static int mv88e6xxx_port_check_hw_vlan(struct dsa_switch *ds, int port,
69 return 0;
70 }
71
72 +static int mv88e6xxx_port_commit_pvid(struct mv88e6xxx_chip *chip, int port)
73 +{
74 + struct dsa_port *dp = dsa_to_port(chip->ds, port);
75 + struct mv88e6xxx_port *p = &chip->ports[port];
76 + bool drop_untagged = false;
77 + u16 pvid = 0;
78 + int err;
79 +
80 + if (dp->bridge_dev && br_vlan_enabled(dp->bridge_dev)) {
81 + pvid = p->bridge_pvid.vid;
82 + drop_untagged = !p->bridge_pvid.valid;
83 + }
84 +
85 + err = mv88e6xxx_port_set_pvid(chip, port, pvid);
86 + if (err)
87 + return err;
88 +
89 + return mv88e6xxx_port_drop_untagged(chip, port, drop_untagged);
90 +}
91 +
92 static int mv88e6xxx_port_vlan_filtering(struct dsa_switch *ds, int port,
93 bool vlan_filtering,
94 struct switchdev_trans *trans)
95 @@ -1599,7 +1619,16 @@ static int mv88e6xxx_port_vlan_filtering(struct dsa_switch *ds, int port,
96 return chip->info->max_vid ? 0 : -EOPNOTSUPP;
97
98 mv88e6xxx_reg_lock(chip);
99 +
100 err = mv88e6xxx_port_set_8021q_mode(chip, port, mode);
101 + if (err)
102 + goto unlock;
103 +
104 + err = mv88e6xxx_port_commit_pvid(chip, port);
105 + if (err)
106 + goto unlock;
107 +
108 +unlock:
109 mv88e6xxx_reg_unlock(chip);
110
111 return err;
112 @@ -1982,8 +2011,10 @@ static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port,
113 struct mv88e6xxx_chip *chip = ds->priv;
114 bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
115 bool pvid = vlan->flags & BRIDGE_VLAN_INFO_PVID;
116 + struct mv88e6xxx_port *p = &chip->ports[port];
117 bool warn;
118 u8 member;
119 + int err;
120 u16 vid;
121
122 if (!chip->info->max_vid)
123 @@ -2008,9 +2039,23 @@ static void mv88e6xxx_port_vlan_add(struct dsa_switch *ds, int port,
124 dev_err(ds->dev, "p%d: failed to add VLAN %d%c\n", port,
125 vid, untagged ? 'u' : 't');
126
127 - if (pvid && mv88e6xxx_port_set_pvid(chip, port, vlan->vid_end))
128 - dev_err(ds->dev, "p%d: failed to set PVID %d\n", port,
129 - vlan->vid_end);
130 + if (pvid) {
131 + p->bridge_pvid.vid = vlan->vid_end;
132 + p->bridge_pvid.valid = true;
133 +
134 + err = mv88e6xxx_port_commit_pvid(chip, port);
135 + if (err)
136 + dev_err(ds->dev, "p%d: failed to set PVID %d", port,
137 + vlan->vid_end);
138 + } else if (vlan->vid_end && p->bridge_pvid.vid == vlan->vid_end) {
139 + /* The old pvid was reinstalled as a non-pvid VLAN */
140 + p->bridge_pvid.valid = false;
141 +
142 + err = mv88e6xxx_port_commit_pvid(chip, port);
143 + if (err)
144 + dev_err(ds->dev, "p%d: failed to unset PVID %d", port,
145 + vlan->vid_end);
146 + }
147
148 mv88e6xxx_reg_unlock(chip);
149 }
150 @@ -2061,6 +2106,7 @@ static int mv88e6xxx_port_vlan_del(struct dsa_switch *ds, int port,
151 const struct switchdev_obj_port_vlan *vlan)
152 {
153 struct mv88e6xxx_chip *chip = ds->priv;
154 + struct mv88e6xxx_port *p = &chip->ports[port];
155 u16 pvid, vid;
156 int err = 0;
157
158 @@ -2079,7 +2125,9 @@ static int mv88e6xxx_port_vlan_del(struct dsa_switch *ds, int port,
159 goto unlock;
160
161 if (vid == pvid) {
162 - err = mv88e6xxx_port_set_pvid(chip, port, 0);
163 + p->bridge_pvid.valid = false;
164 +
165 + err = mv88e6xxx_port_commit_pvid(chip, port);
166 if (err)
167 goto unlock;
168 }
169 diff --git a/drivers/net/dsa/mv88e6xxx/chip.h b/drivers/net/dsa/mv88e6xxx/chip.h
170 index 51a7ff44478e..843803c3adf2 100644
171 --- a/drivers/net/dsa/mv88e6xxx/chip.h
172 +++ b/drivers/net/dsa/mv88e6xxx/chip.h
173 @@ -224,9 +224,15 @@ struct mv88e6xxx_policy {
174 u16 vid;
175 };
176
177 +struct mv88e6xxx_vlan {
178 + u16 vid;
179 + bool valid;
180 +};
181 +
182 struct mv88e6xxx_port {
183 struct mv88e6xxx_chip *chip;
184 int port;
185 + struct mv88e6xxx_vlan bridge_pvid;
186 u64 serdes_stats[2];
187 u64 atu_member_violation;
188 u64 atu_miss_violation;
189 diff --git a/drivers/net/dsa/mv88e6xxx/port.c b/drivers/net/dsa/mv88e6xxx/port.c
190 index dfd9e8292e9a..a7177aa254a8 100644
191 --- a/drivers/net/dsa/mv88e6xxx/port.c
192 +++ b/drivers/net/dsa/mv88e6xxx/port.c
193 @@ -1062,6 +1062,27 @@ int mv88e6xxx_port_set_8021q_mode(struct mv88e6xxx_chip *chip, int port,
194 return 0;
195 }
196
197 +int mv88e6xxx_port_drop_untagged(struct mv88e6xxx_chip *chip, int port,
198 + bool drop_untagged)
199 +{
200 + u16 old, new;
201 + int err;
202 +
203 + err = mv88e6xxx_port_read(chip, port, MV88E6XXX_PORT_CTL2, &old);
204 + if (err)
205 + return err;
206 +
207 + if (drop_untagged)
208 + new = old | MV88E6XXX_PORT_CTL2_DISCARD_UNTAGGED;
209 + else
210 + new = old & ~MV88E6XXX_PORT_CTL2_DISCARD_UNTAGGED;
211 +
212 + if (new == old)
213 + return 0;
214 +
215 + return mv88e6xxx_port_write(chip, port, MV88E6XXX_PORT_CTL2, new);
216 +}
217 +
218 int mv88e6xxx_port_set_map_da(struct mv88e6xxx_chip *chip, int port)
219 {
220 u16 reg;
221 diff --git a/drivers/net/dsa/mv88e6xxx/port.h b/drivers/net/dsa/mv88e6xxx/port.h
222 index 44d76ac973f6..3390517df42e 100644
223 --- a/drivers/net/dsa/mv88e6xxx/port.h
224 +++ b/drivers/net/dsa/mv88e6xxx/port.h
225 @@ -364,6 +364,8 @@ int mv88e6390x_port_set_cmode(struct mv88e6xxx_chip *chip, int port,
226 phy_interface_t mode);
227 int mv88e6185_port_get_cmode(struct mv88e6xxx_chip *chip, int port, u8 *cmode);
228 int mv88e6352_port_get_cmode(struct mv88e6xxx_chip *chip, int port, u8 *cmode);
229 +int mv88e6xxx_port_drop_untagged(struct mv88e6xxx_chip *chip, int port,
230 + bool drop_untagged);
231 int mv88e6xxx_port_set_map_da(struct mv88e6xxx_chip *chip, int port);
232 int mv88e6095_port_set_upstream_port(struct mv88e6xxx_chip *chip, int port,
233 int upstream_port);
234 --
235 2.34.1
236