Commit 65dfc8b4 authored by Florian Fainelli's avatar Florian Fainelli Committed by Greg Kroah-Hartman

net: dsa: b53: Fix VLAN usage and how we treat CPU port

[ Upstream commit e47112d9 ]

We currently have a fundamental problem in how we treat the CPU port and
its VLAN membership. As soon as a second VLAN is configured to be
untagged, the CPU automatically becomes untagged for that VLAN as well,
and yet, we don't gracefully make sure that the CPU becomes tagged in
the other VLANs it could be a member of. This results in only one VLAN
being effectively usable from the CPU's perspective.

Instead of having some pretty complex logic which tries to maintain the
CPU port's default VLAN and its untagged properties, just do something
very simple which consists in neither altering the CPU port's PVID
settings, nor its untagged settings:

- whenever a VLAN is added, the CPU is automatically a member of this
  VLAN group, as a tagged member
- PVID settings for downstream ports do not alter the CPU port's PVID
  since it now is part of all VLANs in the system

This means that a typical example where e.g: LAN ports are in VLAN1, and
WAN port is in VLAN2, now require having two VLAN interfaces for the
host to properly terminate and send traffic from/to.

Fixes: Fixes: a2482d2c ("net: dsa: b53: Plug in VLAN support")
Reported-by: default avatarHartmut Knaack <knaack.h@gmx.de>
Signed-off-by: default avatarFlorian Fainelli <f.fainelli@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent f959eb50
...@@ -904,9 +904,10 @@ static void b53_vlan_add(struct dsa_switch *ds, int port, ...@@ -904,9 +904,10 @@ static void b53_vlan_add(struct dsa_switch *ds, int port,
vl->members |= BIT(port) | BIT(cpu_port); vl->members |= BIT(port) | BIT(cpu_port);
if (untagged) if (untagged)
vl->untag |= BIT(port) | BIT(cpu_port); vl->untag |= BIT(port);
else else
vl->untag &= ~(BIT(port) | BIT(cpu_port)); vl->untag &= ~BIT(port);
vl->untag &= ~BIT(cpu_port);
b53_set_vlan_entry(dev, vid, vl); b53_set_vlan_entry(dev, vid, vl);
b53_fast_age_vlan(dev, vid); b53_fast_age_vlan(dev, vid);
...@@ -915,8 +916,6 @@ static void b53_vlan_add(struct dsa_switch *ds, int port, ...@@ -915,8 +916,6 @@ static void b53_vlan_add(struct dsa_switch *ds, int port,
if (pvid) { if (pvid) {
b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(port), b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(port),
vlan->vid_end); vlan->vid_end);
b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(cpu_port),
vlan->vid_end);
b53_fast_age_vlan(dev, vid); b53_fast_age_vlan(dev, vid);
} }
} }
...@@ -926,7 +925,6 @@ static int b53_vlan_del(struct dsa_switch *ds, int port, ...@@ -926,7 +925,6 @@ static int b53_vlan_del(struct dsa_switch *ds, int port,
{ {
struct b53_device *dev = ds_to_priv(ds); struct b53_device *dev = ds_to_priv(ds);
bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED; bool untagged = vlan->flags & BRIDGE_VLAN_INFO_UNTAGGED;
unsigned int cpu_port = dev->cpu_port;
struct b53_vlan *vl; struct b53_vlan *vl;
u16 vid; u16 vid;
u16 pvid; u16 pvid;
...@@ -939,8 +937,6 @@ static int b53_vlan_del(struct dsa_switch *ds, int port, ...@@ -939,8 +937,6 @@ static int b53_vlan_del(struct dsa_switch *ds, int port,
b53_get_vlan_entry(dev, vid, vl); b53_get_vlan_entry(dev, vid, vl);
vl->members &= ~BIT(port); vl->members &= ~BIT(port);
if ((vl->members & BIT(cpu_port)) == BIT(cpu_port))
vl->members = 0;
if (pvid == vid) { if (pvid == vid) {
if (is5325(dev) || is5365(dev)) if (is5325(dev) || is5365(dev))
...@@ -949,18 +945,14 @@ static int b53_vlan_del(struct dsa_switch *ds, int port, ...@@ -949,18 +945,14 @@ static int b53_vlan_del(struct dsa_switch *ds, int port,
pvid = 0; pvid = 0;
} }
if (untagged) { if (untagged)
vl->untag &= ~(BIT(port)); vl->untag &= ~(BIT(port));
if ((vl->untag & BIT(cpu_port)) == BIT(cpu_port))
vl->untag = 0;
}
b53_set_vlan_entry(dev, vid, vl); b53_set_vlan_entry(dev, vid, vl);
b53_fast_age_vlan(dev, vid); b53_fast_age_vlan(dev, vid);
} }
b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(port), pvid); b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(port), pvid);
b53_write16(dev, B53_VLAN_PAGE, B53_VLAN_PORT_DEF_TAG(cpu_port), pvid);
b53_fast_age_vlan(dev, pvid); b53_fast_age_vlan(dev, pvid);
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