mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-10 07:00:48 +00:00
net: mscc: ocelot: switch from {,un}set to {,un}assign for tag_8021q CPU ports
There is a desire for the felix driver to gain support for multiple tag_8021q CPU ports, but the current model prevents it. This is because ocelot_apply_bridge_fwd_mask() only takes into consideration whether a port is a tag_8021q CPU port, but not whose CPU port it is. We need a model where we can have a direct affinity between an ocelot port and a tag_8021q CPU port. This serves as the basis for multiple CPU ports. Declare a "dsa_8021q_cpu" backpointer in struct ocelot_port which encodes that affinity. Repurpose the "ocelot_set_dsa_8021q_cpu" API to "ocelot_assign_dsa_8021q_cpu" to express the change of paradigm. Note that this change makes the first practical use of the new ocelot_port->index field in ocelot_port_unassign_dsa_8021q_cpu(), where we need to remove the old tag_8021q CPU port from the reserved VLAN range. Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
8c166acb60
commit
c295f9831f
@ -414,21 +414,18 @@ static const struct felix_tag_proto_ops felix_tag_npi_proto_ops = {
|
|||||||
static int felix_tag_8021q_setup(struct dsa_switch *ds)
|
static int felix_tag_8021q_setup(struct dsa_switch *ds)
|
||||||
{
|
{
|
||||||
struct ocelot *ocelot = ds->priv;
|
struct ocelot *ocelot = ds->priv;
|
||||||
struct dsa_port *dp, *cpu_dp;
|
struct dsa_port *dp;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = dsa_tag_8021q_register(ds, htons(ETH_P_8021AD));
|
err = dsa_tag_8021q_register(ds, htons(ETH_P_8021AD));
|
||||||
if (err)
|
if (err)
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
dsa_switch_for_each_cpu_port(cpu_dp, ds) {
|
dsa_switch_for_each_user_port(dp, ds)
|
||||||
ocelot_port_set_dsa_8021q_cpu(ocelot, cpu_dp->index);
|
ocelot_port_assign_dsa_8021q_cpu(ocelot, dp->index,
|
||||||
|
dp->cpu_dp->index);
|
||||||
|
|
||||||
/* TODO we could support multiple CPU ports in tag_8021q mode */
|
dsa_switch_for_each_available_port(dp, ds)
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
dsa_switch_for_each_available_port(dp, ds) {
|
|
||||||
/* This overwrites ocelot_init():
|
/* This overwrites ocelot_init():
|
||||||
* Do not forward BPDU frames to the CPU port module,
|
* Do not forward BPDU frames to the CPU port module,
|
||||||
* for 2 reasons:
|
* for 2 reasons:
|
||||||
@ -442,7 +439,6 @@ static int felix_tag_8021q_setup(struct dsa_switch *ds)
|
|||||||
ocelot_write_gix(ocelot,
|
ocelot_write_gix(ocelot,
|
||||||
ANA_PORT_CPU_FWD_BPDU_CFG_BPDU_REDIR_ENA(0),
|
ANA_PORT_CPU_FWD_BPDU_CFG_BPDU_REDIR_ENA(0),
|
||||||
ANA_PORT_CPU_FWD_BPDU_CFG, dp->index);
|
ANA_PORT_CPU_FWD_BPDU_CFG, dp->index);
|
||||||
}
|
|
||||||
|
|
||||||
/* The ownership of the CPU port module's queues might have just been
|
/* The ownership of the CPU port module's queues might have just been
|
||||||
* transferred to the tag_8021q tagger from the NPI-based tagger.
|
* transferred to the tag_8021q tagger from the NPI-based tagger.
|
||||||
@ -459,9 +455,9 @@ static int felix_tag_8021q_setup(struct dsa_switch *ds)
|
|||||||
static void felix_tag_8021q_teardown(struct dsa_switch *ds)
|
static void felix_tag_8021q_teardown(struct dsa_switch *ds)
|
||||||
{
|
{
|
||||||
struct ocelot *ocelot = ds->priv;
|
struct ocelot *ocelot = ds->priv;
|
||||||
struct dsa_port *dp, *cpu_dp;
|
struct dsa_port *dp;
|
||||||
|
|
||||||
dsa_switch_for_each_available_port(dp, ds) {
|
dsa_switch_for_each_available_port(dp, ds)
|
||||||
/* Restore the logic from ocelot_init:
|
/* Restore the logic from ocelot_init:
|
||||||
* do not forward BPDU frames to the front ports.
|
* do not forward BPDU frames to the front ports.
|
||||||
*/
|
*/
|
||||||
@ -469,14 +465,9 @@ static void felix_tag_8021q_teardown(struct dsa_switch *ds)
|
|||||||
ANA_PORT_CPU_FWD_BPDU_CFG_BPDU_REDIR_ENA(0xffff),
|
ANA_PORT_CPU_FWD_BPDU_CFG_BPDU_REDIR_ENA(0xffff),
|
||||||
ANA_PORT_CPU_FWD_BPDU_CFG,
|
ANA_PORT_CPU_FWD_BPDU_CFG,
|
||||||
dp->index);
|
dp->index);
|
||||||
}
|
|
||||||
|
|
||||||
dsa_switch_for_each_cpu_port(cpu_dp, ds) {
|
dsa_switch_for_each_user_port(dp, ds)
|
||||||
ocelot_port_unset_dsa_8021q_cpu(ocelot, cpu_dp->index);
|
ocelot_port_unassign_dsa_8021q_cpu(ocelot, dp->index);
|
||||||
|
|
||||||
/* TODO we could support multiple CPU ports in tag_8021q mode */
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
dsa_tag_8021q_unregister(ds);
|
dsa_tag_8021q_unregister(ds);
|
||||||
}
|
}
|
||||||
|
@ -2162,7 +2162,8 @@ static void vsc9959_cut_through_fwd(struct ocelot *ocelot)
|
|||||||
if (ocelot->npi >= 0)
|
if (ocelot->npi >= 0)
|
||||||
mask |= BIT(ocelot->npi);
|
mask |= BIT(ocelot->npi);
|
||||||
else
|
else
|
||||||
mask |= ocelot_get_dsa_8021q_cpu_mask(ocelot);
|
mask |= ocelot_port_assigned_dsa_8021q_cpu_mask(ocelot,
|
||||||
|
port);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Calculate the minimum link speed, among the ports that are
|
/* Calculate the minimum link speed, among the ports that are
|
||||||
|
@ -2046,6 +2046,37 @@ static int ocelot_bond_get_id(struct ocelot *ocelot, struct net_device *bond)
|
|||||||
return __ffs(bond_mask);
|
return __ffs(bond_mask);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static u32 ocelot_dsa_8021q_cpu_assigned_ports(struct ocelot *ocelot,
|
||||||
|
struct ocelot_port *cpu)
|
||||||
|
{
|
||||||
|
u32 mask = 0;
|
||||||
|
int port;
|
||||||
|
|
||||||
|
for (port = 0; port < ocelot->num_phys_ports; port++) {
|
||||||
|
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||||
|
|
||||||
|
if (!ocelot_port)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (ocelot_port->dsa_8021q_cpu == cpu)
|
||||||
|
mask |= BIT(port);
|
||||||
|
}
|
||||||
|
|
||||||
|
return mask;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 ocelot_port_assigned_dsa_8021q_cpu_mask(struct ocelot *ocelot, int port)
|
||||||
|
{
|
||||||
|
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||||
|
struct ocelot_port *cpu_port = ocelot_port->dsa_8021q_cpu;
|
||||||
|
|
||||||
|
if (!cpu_port)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
return BIT(cpu_port->index);
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(ocelot_port_assigned_dsa_8021q_cpu_mask);
|
||||||
|
|
||||||
u32 ocelot_get_bridge_fwd_mask(struct ocelot *ocelot, int src_port)
|
u32 ocelot_get_bridge_fwd_mask(struct ocelot *ocelot, int src_port)
|
||||||
{
|
{
|
||||||
struct ocelot_port *ocelot_port = ocelot->ports[src_port];
|
struct ocelot_port *ocelot_port = ocelot->ports[src_port];
|
||||||
@ -2075,28 +2106,8 @@ u32 ocelot_get_bridge_fwd_mask(struct ocelot *ocelot, int src_port)
|
|||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(ocelot_get_bridge_fwd_mask);
|
EXPORT_SYMBOL_GPL(ocelot_get_bridge_fwd_mask);
|
||||||
|
|
||||||
u32 ocelot_get_dsa_8021q_cpu_mask(struct ocelot *ocelot)
|
|
||||||
{
|
|
||||||
u32 mask = 0;
|
|
||||||
int port;
|
|
||||||
|
|
||||||
for (port = 0; port < ocelot->num_phys_ports; port++) {
|
|
||||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
|
||||||
|
|
||||||
if (!ocelot_port)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
if (ocelot_port->is_dsa_8021q_cpu)
|
|
||||||
mask |= BIT(port);
|
|
||||||
}
|
|
||||||
|
|
||||||
return mask;
|
|
||||||
}
|
|
||||||
EXPORT_SYMBOL_GPL(ocelot_get_dsa_8021q_cpu_mask);
|
|
||||||
|
|
||||||
static void ocelot_apply_bridge_fwd_mask(struct ocelot *ocelot, bool joining)
|
static void ocelot_apply_bridge_fwd_mask(struct ocelot *ocelot, bool joining)
|
||||||
{
|
{
|
||||||
unsigned long cpu_fwd_mask;
|
|
||||||
int port;
|
int port;
|
||||||
|
|
||||||
lockdep_assert_held(&ocelot->fwd_domain_lock);
|
lockdep_assert_held(&ocelot->fwd_domain_lock);
|
||||||
@ -2108,15 +2119,6 @@ static void ocelot_apply_bridge_fwd_mask(struct ocelot *ocelot, bool joining)
|
|||||||
if (joining && ocelot->ops->cut_through_fwd)
|
if (joining && ocelot->ops->cut_through_fwd)
|
||||||
ocelot->ops->cut_through_fwd(ocelot);
|
ocelot->ops->cut_through_fwd(ocelot);
|
||||||
|
|
||||||
/* If a DSA tag_8021q CPU exists, it needs to be included in the
|
|
||||||
* regular forwarding path of the front ports regardless of whether
|
|
||||||
* those are bridged or standalone.
|
|
||||||
* If DSA tag_8021q is not used, this returns 0, which is fine because
|
|
||||||
* the hardware-based CPU port module can be a destination for packets
|
|
||||||
* even if it isn't part of PGID_SRC.
|
|
||||||
*/
|
|
||||||
cpu_fwd_mask = ocelot_get_dsa_8021q_cpu_mask(ocelot);
|
|
||||||
|
|
||||||
/* Apply FWD mask. The loop is needed to add/remove the current port as
|
/* Apply FWD mask. The loop is needed to add/remove the current port as
|
||||||
* a source for the other ports.
|
* a source for the other ports.
|
||||||
*/
|
*/
|
||||||
@ -2129,17 +2131,19 @@ static void ocelot_apply_bridge_fwd_mask(struct ocelot *ocelot, bool joining)
|
|||||||
mask = 0;
|
mask = 0;
|
||||||
} else if (ocelot_port->is_dsa_8021q_cpu) {
|
} else if (ocelot_port->is_dsa_8021q_cpu) {
|
||||||
/* The DSA tag_8021q CPU ports need to be able to
|
/* The DSA tag_8021q CPU ports need to be able to
|
||||||
* forward packets to all other ports except for
|
* forward packets to all ports assigned to them.
|
||||||
* themselves
|
|
||||||
*/
|
*/
|
||||||
mask = GENMASK(ocelot->num_phys_ports - 1, 0);
|
mask = ocelot_dsa_8021q_cpu_assigned_ports(ocelot,
|
||||||
mask &= ~cpu_fwd_mask;
|
ocelot_port);
|
||||||
} else if (ocelot_port->bridge) {
|
} else if (ocelot_port->bridge) {
|
||||||
struct net_device *bond = ocelot_port->bond;
|
struct net_device *bond = ocelot_port->bond;
|
||||||
|
|
||||||
mask = ocelot_get_bridge_fwd_mask(ocelot, port);
|
mask = ocelot_get_bridge_fwd_mask(ocelot, port);
|
||||||
mask |= cpu_fwd_mask;
|
|
||||||
mask &= ~BIT(port);
|
mask &= ~BIT(port);
|
||||||
|
|
||||||
|
mask |= ocelot_port_assigned_dsa_8021q_cpu_mask(ocelot,
|
||||||
|
port);
|
||||||
|
|
||||||
if (bond)
|
if (bond)
|
||||||
mask &= ~ocelot_get_bond_mask(ocelot, bond);
|
mask &= ~ocelot_get_bond_mask(ocelot, bond);
|
||||||
} else {
|
} else {
|
||||||
@ -2147,7 +2151,8 @@ static void ocelot_apply_bridge_fwd_mask(struct ocelot *ocelot, bool joining)
|
|||||||
* ports (if those exist), or to the hardware CPU port
|
* ports (if those exist), or to the hardware CPU port
|
||||||
* module otherwise.
|
* module otherwise.
|
||||||
*/
|
*/
|
||||||
mask = cpu_fwd_mask;
|
mask = ocelot_port_assigned_dsa_8021q_cpu_mask(ocelot,
|
||||||
|
port);
|
||||||
}
|
}
|
||||||
|
|
||||||
ocelot_write_rix(ocelot, mask, ANA_PGID_PGID, PGID_SRC + port);
|
ocelot_write_rix(ocelot, mask, ANA_PGID_PGID, PGID_SRC + port);
|
||||||
@ -2191,43 +2196,66 @@ static void ocelot_update_pgid_cpu(struct ocelot *ocelot)
|
|||||||
ocelot_write_rix(ocelot, pgid_cpu, ANA_PGID_PGID, PGID_CPU);
|
ocelot_write_rix(ocelot, pgid_cpu, ANA_PGID_PGID, PGID_CPU);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ocelot_port_set_dsa_8021q_cpu(struct ocelot *ocelot, int port)
|
void ocelot_port_assign_dsa_8021q_cpu(struct ocelot *ocelot, int port,
|
||||||
|
int cpu)
|
||||||
{
|
{
|
||||||
|
struct ocelot_port *cpu_port = ocelot->ports[cpu];
|
||||||
u16 vid;
|
u16 vid;
|
||||||
|
|
||||||
mutex_lock(&ocelot->fwd_domain_lock);
|
mutex_lock(&ocelot->fwd_domain_lock);
|
||||||
|
|
||||||
ocelot->ports[port]->is_dsa_8021q_cpu = true;
|
ocelot->ports[port]->dsa_8021q_cpu = cpu_port;
|
||||||
|
|
||||||
for (vid = OCELOT_RSV_VLAN_RANGE_START; vid < VLAN_N_VID; vid++)
|
if (!cpu_port->is_dsa_8021q_cpu) {
|
||||||
ocelot_vlan_member_add(ocelot, port, vid, true);
|
cpu_port->is_dsa_8021q_cpu = true;
|
||||||
|
|
||||||
ocelot_update_pgid_cpu(ocelot);
|
for (vid = OCELOT_RSV_VLAN_RANGE_START; vid < VLAN_N_VID; vid++)
|
||||||
|
ocelot_vlan_member_add(ocelot, cpu, vid, true);
|
||||||
|
|
||||||
|
ocelot_update_pgid_cpu(ocelot);
|
||||||
|
}
|
||||||
|
|
||||||
ocelot_apply_bridge_fwd_mask(ocelot, true);
|
ocelot_apply_bridge_fwd_mask(ocelot, true);
|
||||||
|
|
||||||
mutex_unlock(&ocelot->fwd_domain_lock);
|
mutex_unlock(&ocelot->fwd_domain_lock);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(ocelot_port_set_dsa_8021q_cpu);
|
EXPORT_SYMBOL_GPL(ocelot_port_assign_dsa_8021q_cpu);
|
||||||
|
|
||||||
void ocelot_port_unset_dsa_8021q_cpu(struct ocelot *ocelot, int port)
|
void ocelot_port_unassign_dsa_8021q_cpu(struct ocelot *ocelot, int port)
|
||||||
{
|
{
|
||||||
|
struct ocelot_port *cpu_port = ocelot->ports[port]->dsa_8021q_cpu;
|
||||||
|
bool keep = false;
|
||||||
u16 vid;
|
u16 vid;
|
||||||
|
int p;
|
||||||
|
|
||||||
mutex_lock(&ocelot->fwd_domain_lock);
|
mutex_lock(&ocelot->fwd_domain_lock);
|
||||||
|
|
||||||
ocelot->ports[port]->is_dsa_8021q_cpu = false;
|
ocelot->ports[port]->dsa_8021q_cpu = NULL;
|
||||||
|
|
||||||
for (vid = OCELOT_RSV_VLAN_RANGE_START; vid < VLAN_N_VID; vid++)
|
for (p = 0; p < ocelot->num_phys_ports; p++) {
|
||||||
ocelot_vlan_member_del(ocelot, port, vid);
|
if (!ocelot->ports[p])
|
||||||
|
continue;
|
||||||
|
|
||||||
ocelot_update_pgid_cpu(ocelot);
|
if (ocelot->ports[p]->dsa_8021q_cpu == cpu_port) {
|
||||||
|
keep = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!keep) {
|
||||||
|
cpu_port->is_dsa_8021q_cpu = false;
|
||||||
|
|
||||||
|
for (vid = OCELOT_RSV_VLAN_RANGE_START; vid < VLAN_N_VID; vid++)
|
||||||
|
ocelot_vlan_member_del(ocelot, cpu_port->index, vid);
|
||||||
|
|
||||||
|
ocelot_update_pgid_cpu(ocelot);
|
||||||
|
}
|
||||||
|
|
||||||
ocelot_apply_bridge_fwd_mask(ocelot, true);
|
ocelot_apply_bridge_fwd_mask(ocelot, true);
|
||||||
|
|
||||||
mutex_unlock(&ocelot->fwd_domain_lock);
|
mutex_unlock(&ocelot->fwd_domain_lock);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(ocelot_port_unset_dsa_8021q_cpu);
|
EXPORT_SYMBOL_GPL(ocelot_port_unassign_dsa_8021q_cpu);
|
||||||
|
|
||||||
void ocelot_bridge_stp_state_set(struct ocelot *ocelot, int port, u8 state)
|
void ocelot_bridge_stp_state_set(struct ocelot *ocelot, int port, u8 state)
|
||||||
{
|
{
|
||||||
|
@ -654,6 +654,8 @@ struct ocelot_mirror {
|
|||||||
int to;
|
int to;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct ocelot_port;
|
||||||
|
|
||||||
struct ocelot_port {
|
struct ocelot_port {
|
||||||
struct ocelot *ocelot;
|
struct ocelot *ocelot;
|
||||||
|
|
||||||
@ -662,6 +664,8 @@ struct ocelot_port {
|
|||||||
struct net_device *bond;
|
struct net_device *bond;
|
||||||
struct net_device *bridge;
|
struct net_device *bridge;
|
||||||
|
|
||||||
|
struct ocelot_port *dsa_8021q_cpu;
|
||||||
|
|
||||||
/* VLAN that untagged frames are classified to, on ingress */
|
/* VLAN that untagged frames are classified to, on ingress */
|
||||||
const struct ocelot_bridge_vlan *pvid_vlan;
|
const struct ocelot_bridge_vlan *pvid_vlan;
|
||||||
|
|
||||||
@ -865,8 +869,9 @@ void ocelot_deinit(struct ocelot *ocelot);
|
|||||||
void ocelot_init_port(struct ocelot *ocelot, int port);
|
void ocelot_init_port(struct ocelot *ocelot, int port);
|
||||||
void ocelot_deinit_port(struct ocelot *ocelot, int port);
|
void ocelot_deinit_port(struct ocelot *ocelot, int port);
|
||||||
|
|
||||||
void ocelot_port_set_dsa_8021q_cpu(struct ocelot *ocelot, int port);
|
void ocelot_port_assign_dsa_8021q_cpu(struct ocelot *ocelot, int port, int cpu);
|
||||||
void ocelot_port_unset_dsa_8021q_cpu(struct ocelot *ocelot, int port);
|
void ocelot_port_unassign_dsa_8021q_cpu(struct ocelot *ocelot, int port);
|
||||||
|
u32 ocelot_port_assigned_dsa_8021q_cpu_mask(struct ocelot *ocelot, int port);
|
||||||
|
|
||||||
/* DSA callbacks */
|
/* DSA callbacks */
|
||||||
void ocelot_get_strings(struct ocelot *ocelot, int port, u32 sset, u8 *data);
|
void ocelot_get_strings(struct ocelot *ocelot, int port, u32 sset, u8 *data);
|
||||||
@ -878,7 +883,6 @@ void ocelot_set_ageing_time(struct ocelot *ocelot, unsigned int msecs);
|
|||||||
int ocelot_port_vlan_filtering(struct ocelot *ocelot, int port, bool enabled,
|
int ocelot_port_vlan_filtering(struct ocelot *ocelot, int port, bool enabled,
|
||||||
struct netlink_ext_ack *extack);
|
struct netlink_ext_ack *extack);
|
||||||
void ocelot_bridge_stp_state_set(struct ocelot *ocelot, int port, u8 state);
|
void ocelot_bridge_stp_state_set(struct ocelot *ocelot, int port, u8 state);
|
||||||
u32 ocelot_get_dsa_8021q_cpu_mask(struct ocelot *ocelot);
|
|
||||||
u32 ocelot_get_bridge_fwd_mask(struct ocelot *ocelot, int src_port);
|
u32 ocelot_get_bridge_fwd_mask(struct ocelot *ocelot, int src_port);
|
||||||
int ocelot_port_pre_bridge_flags(struct ocelot *ocelot, int port,
|
int ocelot_port_pre_bridge_flags(struct ocelot *ocelot, int port,
|
||||||
struct switchdev_brport_flags val);
|
struct switchdev_brport_flags val);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user