net: mscc: ocelot: offload VLAN mangle action to VCAP IS1
The VCAP_IS1_ACT_VID_REPLACE_ENA action, from the VCAP IS1 ingress TCAM, changes the classified VLAN. We are only exposing this ability for switch ports that are under VLAN aware bridges. This is because in standalone ports mode and under a bridge with vlan_filtering=0, the ocelot driver configures the switch to operate as VLAN-unaware, so the classified VLAN is not derived from the 802.1Q header from the packet, but instead is always equal to the port-based VLAN ID of the ingress port. We _can_ still change the classified VLAN for packets when operating in this mode, but the end result will most likely be a drop, since both the ingress and the egress port need to be members of the modified VLAN. And even if we install the new classified VLAN into the VLAN table of the switch, the result would still not be as expected: we wouldn't see, on the output port, the modified VLAN tag, but the original one, even though the classified VLAN was indeed modified. This is because of how the hardware works: on egress, what is pushed to the frame is a "port tag", which gives us the following options: - Tag all frames with port tag (derived from the classified VLAN) - Tag all frames with port tag, except if the classified VLAN is 0 or equal to the native VLAN of the egress port - No port tag Needless to say, in VLAN-unaware mode we are disabling the port tag. Otherwise, the existing VLAN tag would be ignored, and a second VLAN tag (the port tag), holding the classified VLAN, would be pushed (instead of replacing the existing 802.1Q tag). This is definitely not what the user wanted when installing a "vlan modify" action. So it is simply not worth bothering with VLAN modify rules under other configurations except when the ports are fully VLAN-aware. Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com> Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
parent
bea4b3095b
commit
70edfae15a
|
@ -205,8 +205,21 @@ int ocelot_port_vlan_filtering(struct ocelot *ocelot, int port,
|
||||||
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||||
u32 val;
|
u32 val;
|
||||||
|
|
||||||
if (switchdev_trans_ph_prepare(trans))
|
if (switchdev_trans_ph_prepare(trans)) {
|
||||||
|
struct ocelot_vcap_block *block = &ocelot->block[VCAP_IS1];
|
||||||
|
struct ocelot_vcap_filter *filter;
|
||||||
|
|
||||||
|
list_for_each_entry(filter, &block->rules, list) {
|
||||||
|
if (filter->ingress_port_mask & BIT(port) &&
|
||||||
|
filter->action.vid_replace_ena) {
|
||||||
|
dev_err(ocelot->dev,
|
||||||
|
"Cannot change VLAN state with vlan modify rules active\n");
|
||||||
|
return -EBUSY;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
ocelot_port->vlan_aware = vlan_aware;
|
ocelot_port->vlan_aware = vlan_aware;
|
||||||
|
|
||||||
|
|
|
@ -142,10 +142,11 @@ ocelot_find_vcap_filter_that_points_at(struct ocelot *ocelot, int chain)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ocelot_flower_parse_action(struct ocelot *ocelot, bool ingress,
|
static int ocelot_flower_parse_action(struct ocelot *ocelot, int port,
|
||||||
struct flow_cls_offload *f,
|
bool ingress, struct flow_cls_offload *f,
|
||||||
struct ocelot_vcap_filter *filter)
|
struct ocelot_vcap_filter *filter)
|
||||||
{
|
{
|
||||||
|
struct ocelot_port *ocelot_port = ocelot->ports[port];
|
||||||
struct netlink_ext_ack *extack = f->common.extack;
|
struct netlink_ext_ack *extack = f->common.extack;
|
||||||
bool allow_missing_goto_target = false;
|
bool allow_missing_goto_target = false;
|
||||||
const struct flow_action_entry *a;
|
const struct flow_action_entry *a;
|
||||||
|
@ -266,6 +267,28 @@ static int ocelot_flower_parse_action(struct ocelot *ocelot, bool ingress,
|
||||||
}
|
}
|
||||||
filter->type = OCELOT_VCAP_FILTER_OFFLOAD;
|
filter->type = OCELOT_VCAP_FILTER_OFFLOAD;
|
||||||
break;
|
break;
|
||||||
|
case FLOW_ACTION_VLAN_MANGLE:
|
||||||
|
if (filter->block_id != VCAP_IS1) {
|
||||||
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
|
"VLAN modify action can only be offloaded to VCAP IS1");
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
if (filter->goto_target != -1) {
|
||||||
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
|
"Last action must be GOTO");
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
if (!ocelot_port->vlan_aware) {
|
||||||
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
|
"Can only modify VLAN under VLAN aware bridge");
|
||||||
|
return -EOPNOTSUPP;
|
||||||
|
}
|
||||||
|
filter->action.vid_replace_ena = true;
|
||||||
|
filter->action.pcp_dei_ena = true;
|
||||||
|
filter->action.vid = a->vlan.vid;
|
||||||
|
filter->action.pcp = a->vlan.prio;
|
||||||
|
filter->type = OCELOT_VCAP_FILTER_OFFLOAD;
|
||||||
|
break;
|
||||||
case FLOW_ACTION_PRIORITY:
|
case FLOW_ACTION_PRIORITY:
|
||||||
if (filter->block_id != VCAP_IS1) {
|
if (filter->block_id != VCAP_IS1) {
|
||||||
NL_SET_ERR_MSG_MOD(extack,
|
NL_SET_ERR_MSG_MOD(extack,
|
||||||
|
@ -601,7 +624,7 @@ static int ocelot_flower_parse(struct ocelot *ocelot, int port, bool ingress,
|
||||||
filter->prio = f->common.prio;
|
filter->prio = f->common.prio;
|
||||||
filter->id = f->cookie;
|
filter->id = f->cookie;
|
||||||
|
|
||||||
ret = ocelot_flower_parse_action(ocelot, ingress, f, filter);
|
ret = ocelot_flower_parse_action(ocelot, port, ingress, f, filter);
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue