Commit 87b08a8c authored by David S. Miller's avatar David S. Miller

Merge branch 'net-microchip-vcap-rules'

Steen Hegelund says:

====================
net: microchip: Add support for two classes of VCAP rules

This adds support for two classes of VCAP rules:

- Permanent rules (added e.g. for PTP support)
- TC user rules (added by the TC userspace tool)

For this to work the VCAP Loopups must be enabled from boot, so that the
"internal" clients like PTP can add rules that are always active.

When the TC tool add a flower filter the VCAP rule corresponding to this
filter will be disabled (kept in memory) until a TC matchall filter creates
a link from chain 0 to the chain (lookup) where the flower filter was
added.

When the flower filter is enabled it will be written to the appropriate
VCAP lookup and become active in HW.

Likewise the flower filter will be disabled if there is no link from chain
0 to the chain of the filter (lookup), and when that happens the
corresponding VCAP rule will be read from the VCAP instance and stored in
memory until it is deleted or enabled again.

Version History:
================
v4      Removed a leftover 'Fixes' tag from v2.  No functional changes.

v3      Removed the change that allowed rules to always be added in the
        LAN996x even though the lookups are not enabled (Horatiu Vultur).
        This was sent separately to net instead.

        Removed the 'Fixes' tags due to the patch sent to net by Horatiu
        Vultur.

        Added a check for validity of the chain source when enabling a
        lookup.

v2      Adding a missing goto exit in vcap_add_rule (Dan Carpenter).
        Added missing checks for error returns in vcap_enable_rule.

v1      Initial version
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents ce870af3 18a15c76
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
#include "vcap_api_client.h" #include "vcap_api_client.h"
int lan966x_goto_port_add(struct lan966x_port *port, int lan966x_goto_port_add(struct lan966x_port *port,
struct flow_action_entry *act, int from_cid, int to_cid,
unsigned long goto_id, unsigned long goto_id,
struct netlink_ext_ack *extack) struct netlink_ext_ack *extack)
{ {
...@@ -12,7 +12,7 @@ int lan966x_goto_port_add(struct lan966x_port *port, ...@@ -12,7 +12,7 @@ int lan966x_goto_port_add(struct lan966x_port *port,
int err; int err;
err = vcap_enable_lookups(lan966x->vcap_ctrl, port->dev, err = vcap_enable_lookups(lan966x->vcap_ctrl, port->dev,
act->chain_index, goto_id, from_cid, to_cid, goto_id,
true); true);
if (err == -EFAULT) { if (err == -EFAULT) {
NL_SET_ERR_MSG_MOD(extack, "Unsupported goto chain"); NL_SET_ERR_MSG_MOD(extack, "Unsupported goto chain");
...@@ -29,8 +29,6 @@ int lan966x_goto_port_add(struct lan966x_port *port, ...@@ -29,8 +29,6 @@ int lan966x_goto_port_add(struct lan966x_port *port,
return err; return err;
} }
port->tc.goto_id = goto_id;
return 0; return 0;
} }
...@@ -41,14 +39,12 @@ int lan966x_goto_port_del(struct lan966x_port *port, ...@@ -41,14 +39,12 @@ int lan966x_goto_port_del(struct lan966x_port *port,
struct lan966x *lan966x = port->lan966x; struct lan966x *lan966x = port->lan966x;
int err; int err;
err = vcap_enable_lookups(lan966x->vcap_ctrl, port->dev, 0, err = vcap_enable_lookups(lan966x->vcap_ctrl, port->dev, 0, 0,
goto_id, false); goto_id, false);
if (err) { if (err) {
NL_SET_ERR_MSG_MOD(extack, "Could not disable VCAP lookups"); NL_SET_ERR_MSG_MOD(extack, "Could not disable VCAP lookups");
return err; return err;
} }
port->tc.goto_id = 0;
return 0; return 0;
} }
...@@ -332,7 +332,6 @@ struct lan966x_port_tc { ...@@ -332,7 +332,6 @@ struct lan966x_port_tc {
unsigned long police_id; unsigned long police_id;
unsigned long ingress_mirror_id; unsigned long ingress_mirror_id;
unsigned long egress_mirror_id; unsigned long egress_mirror_id;
unsigned long goto_id;
struct flow_stats police_stat; struct flow_stats police_stat;
struct flow_stats mirror_stat; struct flow_stats mirror_stat;
}; };
...@@ -607,7 +606,7 @@ int lan966x_tc_flower(struct lan966x_port *port, ...@@ -607,7 +606,7 @@ int lan966x_tc_flower(struct lan966x_port *port,
struct flow_cls_offload *f); struct flow_cls_offload *f);
int lan966x_goto_port_add(struct lan966x_port *port, int lan966x_goto_port_add(struct lan966x_port *port,
struct flow_action_entry *act, int from_cid, int to_cid,
unsigned long goto_id, unsigned long goto_id,
struct netlink_ext_ack *extack); struct netlink_ext_ack *extack);
int lan966x_goto_port_del(struct lan966x_port *port, int lan966x_goto_port_del(struct lan966x_port *port,
......
...@@ -82,8 +82,8 @@ static int lan966x_tc_flower_use_dissectors(struct flow_cls_offload *f, ...@@ -82,8 +82,8 @@ static int lan966x_tc_flower_use_dissectors(struct flow_cls_offload *f,
} }
static int lan966x_tc_flower_action_check(struct vcap_control *vctrl, static int lan966x_tc_flower_action_check(struct vcap_control *vctrl,
struct flow_cls_offload *fco, struct net_device *dev,
struct vcap_admin *admin) struct flow_cls_offload *fco)
{ {
struct flow_rule *rule = flow_cls_offload_flow_rule(fco); struct flow_rule *rule = flow_cls_offload_flow_rule(fco);
struct flow_action_entry *actent, *last_actent = NULL; struct flow_action_entry *actent, *last_actent = NULL;
...@@ -109,21 +109,23 @@ static int lan966x_tc_flower_action_check(struct vcap_control *vctrl, ...@@ -109,21 +109,23 @@ static int lan966x_tc_flower_action_check(struct vcap_control *vctrl,
last_actent = actent; /* Save last action for later check */ last_actent = actent; /* Save last action for later check */
} }
/* Check that last action is a goto */ /* Check that last action is a goto
if (last_actent->id != FLOW_ACTION_GOTO) { * The last chain/lookup does not need to have goto action
*/
if (last_actent->id == FLOW_ACTION_GOTO) {
/* Check if the destination chain is in one of the VCAPs */
if (!vcap_is_next_lookup(vctrl, fco->common.chain_index,
last_actent->chain_index)) {
NL_SET_ERR_MSG_MOD(fco->common.extack,
"Invalid goto chain");
return -EINVAL;
}
} else if (!vcap_is_last_chain(vctrl, fco->common.chain_index)) {
NL_SET_ERR_MSG_MOD(fco->common.extack, NL_SET_ERR_MSG_MOD(fco->common.extack,
"Last action must be 'goto'"); "Last action must be 'goto'");
return -EINVAL; return -EINVAL;
} }
/* Check if the goto chain is in the next lookup */
if (!vcap_is_next_lookup(vctrl, fco->common.chain_index,
last_actent->chain_index)) {
NL_SET_ERR_MSG_MOD(fco->common.extack,
"Invalid goto chain");
return -EINVAL;
}
/* Catch unsupported combinations of actions */ /* Catch unsupported combinations of actions */
if (action_mask & BIT(FLOW_ACTION_TRAP) && if (action_mask & BIT(FLOW_ACTION_TRAP) &&
action_mask & BIT(FLOW_ACTION_ACCEPT)) { action_mask & BIT(FLOW_ACTION_ACCEPT)) {
...@@ -145,8 +147,8 @@ static int lan966x_tc_flower_add(struct lan966x_port *port, ...@@ -145,8 +147,8 @@ static int lan966x_tc_flower_add(struct lan966x_port *port,
struct vcap_rule *vrule; struct vcap_rule *vrule;
int err, idx; int err, idx;
err = lan966x_tc_flower_action_check(port->lan966x->vcap_ctrl, f, err = lan966x_tc_flower_action_check(port->lan966x->vcap_ctrl,
admin); port->dev, f);
if (err) if (err)
return err; return err;
......
...@@ -24,7 +24,8 @@ static int lan966x_tc_matchall_add(struct lan966x_port *port, ...@@ -24,7 +24,8 @@ static int lan966x_tc_matchall_add(struct lan966x_port *port,
return lan966x_mirror_port_add(port, act, f->cookie, return lan966x_mirror_port_add(port, act, f->cookie,
ingress, f->common.extack); ingress, f->common.extack);
case FLOW_ACTION_GOTO: case FLOW_ACTION_GOTO:
return lan966x_goto_port_add(port, act, f->cookie, return lan966x_goto_port_add(port, f->common.chain_index,
act->chain_index, f->cookie,
f->common.extack); f->common.extack);
default: default:
NL_SET_ERR_MSG_MOD(f->common.extack, NL_SET_ERR_MSG_MOD(f->common.extack,
...@@ -46,13 +47,8 @@ static int lan966x_tc_matchall_del(struct lan966x_port *port, ...@@ -46,13 +47,8 @@ static int lan966x_tc_matchall_del(struct lan966x_port *port,
f->cookie == port->tc.egress_mirror_id) { f->cookie == port->tc.egress_mirror_id) {
return lan966x_mirror_port_del(port, ingress, return lan966x_mirror_port_del(port, ingress,
f->common.extack); f->common.extack);
} else if (f->cookie == port->tc.goto_id) {
return lan966x_goto_port_del(port, f->cookie,
f->common.extack);
} else { } else {
NL_SET_ERR_MSG_MOD(f->common.extack, return lan966x_goto_port_del(port, f->cookie, f->common.extack);
"Unsupported action");
return -EOPNOTSUPP;
} }
return 0; return 0;
...@@ -80,12 +76,6 @@ int lan966x_tc_matchall(struct lan966x_port *port, ...@@ -80,12 +76,6 @@ int lan966x_tc_matchall(struct lan966x_port *port,
struct tc_cls_matchall_offload *f, struct tc_cls_matchall_offload *f,
bool ingress) bool ingress)
{ {
if (!tc_cls_can_offload_and_chain0(port->dev, &f->common)) {
NL_SET_ERR_MSG_MOD(f->common.extack,
"Only chain zero is supported");
return -EOPNOTSUPP;
}
switch (f->command) { switch (f->command) {
case TC_CLSMATCHALL_REPLACE: case TC_CLSMATCHALL_REPLACE:
return lan966x_tc_matchall_add(port, f, ingress); return lan966x_tc_matchall_add(port, f, ingress);
......
...@@ -390,20 +390,6 @@ static int lan966x_vcap_port_info(struct net_device *dev, ...@@ -390,20 +390,6 @@ static int lan966x_vcap_port_info(struct net_device *dev,
return 0; return 0;
} }
static int lan966x_vcap_enable(struct net_device *dev,
struct vcap_admin *admin,
bool enable)
{
struct lan966x_port *port = netdev_priv(dev);
struct lan966x *lan966x = port->lan966x;
lan_rmw(ANA_VCAP_S2_CFG_ENA_SET(enable),
ANA_VCAP_S2_CFG_ENA,
lan966x, ANA_VCAP_S2_CFG(port->chip_port));
return 0;
}
static struct vcap_operations lan966x_vcap_ops = { static struct vcap_operations lan966x_vcap_ops = {
.validate_keyset = lan966x_vcap_validate_keyset, .validate_keyset = lan966x_vcap_validate_keyset,
.add_default_fields = lan966x_vcap_add_default_fields, .add_default_fields = lan966x_vcap_add_default_fields,
...@@ -414,7 +400,6 @@ static struct vcap_operations lan966x_vcap_ops = { ...@@ -414,7 +400,6 @@ static struct vcap_operations lan966x_vcap_ops = {
.update = lan966x_vcap_update, .update = lan966x_vcap_update,
.move = lan966x_vcap_move, .move = lan966x_vcap_move,
.port_info = lan966x_vcap_port_info, .port_info = lan966x_vcap_port_info,
.enable = lan966x_vcap_enable,
}; };
static void lan966x_vcap_admin_free(struct vcap_admin *admin) static void lan966x_vcap_admin_free(struct vcap_admin *admin)
...@@ -521,6 +506,12 @@ int lan966x_vcap_init(struct lan966x *lan966x) ...@@ -521,6 +506,12 @@ int lan966x_vcap_init(struct lan966x *lan966x)
list_add_tail(&admin->list, &ctrl->list); list_add_tail(&admin->list, &ctrl->list);
} }
for (int p = 0; p < lan966x->num_phys_ports; ++p)
if (lan966x->ports[p])
lan_rmw(ANA_VCAP_S2_CFG_ENA_SET(true),
ANA_VCAP_S2_CFG_ENA, lan966x,
ANA_VCAP_S2_CFG(lan966x->ports[p]->chip_port));
lan966x->vcap_ctrl = ctrl; lan966x->vcap_ctrl = ctrl;
return 0; return 0;
......
...@@ -573,8 +573,8 @@ static int sparx5_tc_use_dissectors(struct flow_cls_offload *fco, ...@@ -573,8 +573,8 @@ static int sparx5_tc_use_dissectors(struct flow_cls_offload *fco,
} }
static int sparx5_tc_flower_action_check(struct vcap_control *vctrl, static int sparx5_tc_flower_action_check(struct vcap_control *vctrl,
struct flow_cls_offload *fco, struct net_device *ndev,
struct vcap_admin *admin) struct flow_cls_offload *fco)
{ {
struct flow_rule *rule = flow_cls_offload_flow_rule(fco); struct flow_rule *rule = flow_cls_offload_flow_rule(fco);
struct flow_action_entry *actent, *last_actent = NULL; struct flow_action_entry *actent, *last_actent = NULL;
...@@ -600,21 +600,23 @@ static int sparx5_tc_flower_action_check(struct vcap_control *vctrl, ...@@ -600,21 +600,23 @@ static int sparx5_tc_flower_action_check(struct vcap_control *vctrl,
last_actent = actent; /* Save last action for later check */ last_actent = actent; /* Save last action for later check */
} }
/* Check that last action is a goto */ /* Check if last action is a goto
if (last_actent->id != FLOW_ACTION_GOTO) { * The last chain/lookup does not need to have a goto action
*/
if (last_actent->id == FLOW_ACTION_GOTO) {
/* Check if the destination chain is in one of the VCAPs */
if (!vcap_is_next_lookup(vctrl, fco->common.chain_index,
last_actent->chain_index)) {
NL_SET_ERR_MSG_MOD(fco->common.extack,
"Invalid goto chain");
return -EINVAL;
}
} else if (!vcap_is_last_chain(vctrl, fco->common.chain_index)) {
NL_SET_ERR_MSG_MOD(fco->common.extack, NL_SET_ERR_MSG_MOD(fco->common.extack,
"Last action must be 'goto'"); "Last action must be 'goto'");
return -EINVAL; return -EINVAL;
} }
/* Check if the goto chain is in the next lookup */
if (!vcap_is_next_lookup(vctrl, fco->common.chain_index,
last_actent->chain_index)) {
NL_SET_ERR_MSG_MOD(fco->common.extack,
"Invalid goto chain");
return -EINVAL;
}
/* Catch unsupported combinations of actions */ /* Catch unsupported combinations of actions */
if (action_mask & BIT(FLOW_ACTION_TRAP) && if (action_mask & BIT(FLOW_ACTION_TRAP) &&
action_mask & BIT(FLOW_ACTION_ACCEPT)) { action_mask & BIT(FLOW_ACTION_ACCEPT)) {
...@@ -833,7 +835,7 @@ static int sparx5_tc_flower_replace(struct net_device *ndev, ...@@ -833,7 +835,7 @@ static int sparx5_tc_flower_replace(struct net_device *ndev,
vctrl = port->sparx5->vcap_ctrl; vctrl = port->sparx5->vcap_ctrl;
err = sparx5_tc_flower_action_check(vctrl, fco, admin); err = sparx5_tc_flower_action_check(vctrl, ndev, fco);
if (err) if (err)
return err; return err;
......
...@@ -31,6 +31,7 @@ static int sparx5_tc_matchall_replace(struct net_device *ndev, ...@@ -31,6 +31,7 @@ static int sparx5_tc_matchall_replace(struct net_device *ndev,
switch (action->id) { switch (action->id) {
case FLOW_ACTION_GOTO: case FLOW_ACTION_GOTO:
err = vcap_enable_lookups(sparx5->vcap_ctrl, ndev, err = vcap_enable_lookups(sparx5->vcap_ctrl, ndev,
tmo->common.chain_index,
action->chain_index, tmo->cookie, action->chain_index, tmo->cookie,
true); true);
if (err == -EFAULT) { if (err == -EFAULT) {
...@@ -43,6 +44,11 @@ static int sparx5_tc_matchall_replace(struct net_device *ndev, ...@@ -43,6 +44,11 @@ static int sparx5_tc_matchall_replace(struct net_device *ndev,
"VCAP already enabled"); "VCAP already enabled");
return -EOPNOTSUPP; return -EOPNOTSUPP;
} }
if (err == -EADDRNOTAVAIL) {
NL_SET_ERR_MSG_MOD(tmo->common.extack,
"Already matching this chain");
return -EOPNOTSUPP;
}
if (err) { if (err) {
NL_SET_ERR_MSG_MOD(tmo->common.extack, NL_SET_ERR_MSG_MOD(tmo->common.extack,
"Could not enable VCAP lookups"); "Could not enable VCAP lookups");
...@@ -66,8 +72,8 @@ static int sparx5_tc_matchall_destroy(struct net_device *ndev, ...@@ -66,8 +72,8 @@ static int sparx5_tc_matchall_destroy(struct net_device *ndev,
sparx5 = port->sparx5; sparx5 = port->sparx5;
if (!tmo->rule && tmo->cookie) { if (!tmo->rule && tmo->cookie) {
err = vcap_enable_lookups(sparx5->vcap_ctrl, ndev, 0, err = vcap_enable_lookups(sparx5->vcap_ctrl, ndev,
tmo->cookie, false); 0, 0, tmo->cookie, false);
if (err) if (err)
return err; return err;
return 0; return 0;
...@@ -80,12 +86,6 @@ int sparx5_tc_matchall(struct net_device *ndev, ...@@ -80,12 +86,6 @@ int sparx5_tc_matchall(struct net_device *ndev,
struct tc_cls_matchall_offload *tmo, struct tc_cls_matchall_offload *tmo,
bool ingress) bool ingress)
{ {
if (!tc_cls_can_offload_and_chain0(ndev, &tmo->common)) {
NL_SET_ERR_MSG_MOD(tmo->common.extack,
"Only chain zero is supported");
return -EOPNOTSUPP;
}
switch (tmo->command) { switch (tmo->command) {
case TC_CLSMATCHALL_REPLACE: case TC_CLSMATCHALL_REPLACE:
return sparx5_tc_matchall_replace(ndev, tmo, ingress); return sparx5_tc_matchall_replace(ndev, tmo, ingress);
......
...@@ -29,7 +29,7 @@ static void sparx5_vcap_port_keys(struct sparx5 *sparx5, ...@@ -29,7 +29,7 @@ static void sparx5_vcap_port_keys(struct sparx5 *sparx5,
/* Get lookup state */ /* Get lookup state */
value = spx5_rd(sparx5, ANA_ACL_VCAP_S2_CFG(port->portno)); value = spx5_rd(sparx5, ANA_ACL_VCAP_S2_CFG(port->portno));
out->prf(out->dst, "\n state: "); out->prf(out->dst, "\n state: ");
if (ANA_ACL_VCAP_S2_CFG_SEC_ENA_GET(value)) if (ANA_ACL_VCAP_S2_CFG_SEC_ENA_GET(value) & BIT(lookup))
out->prf(out->dst, "on"); out->prf(out->dst, "on");
else else
out->prf(out->dst, "off"); out->prf(out->dst, "off");
......
...@@ -510,28 +510,6 @@ static void sparx5_vcap_move(struct net_device *ndev, struct vcap_admin *admin, ...@@ -510,28 +510,6 @@ static void sparx5_vcap_move(struct net_device *ndev, struct vcap_admin *admin,
sparx5_vcap_wait_super_update(sparx5); sparx5_vcap_wait_super_update(sparx5);
} }
/* Enable all lookups in the VCAP instance */
static int sparx5_vcap_enable(struct net_device *ndev,
struct vcap_admin *admin,
bool enable)
{
struct sparx5_port *port = netdev_priv(ndev);
struct sparx5 *sparx5;
int portno;
sparx5 = port->sparx5;
portno = port->portno;
/* For now we only consider IS2 */
if (enable)
spx5_wr(ANA_ACL_VCAP_S2_CFG_SEC_ENA_SET(0xf), sparx5,
ANA_ACL_VCAP_S2_CFG(portno));
else
spx5_wr(ANA_ACL_VCAP_S2_CFG_SEC_ENA_SET(0), sparx5,
ANA_ACL_VCAP_S2_CFG(portno));
return 0;
}
/* API callback operations: only IS2 is supported for now */ /* API callback operations: only IS2 is supported for now */
static struct vcap_operations sparx5_vcap_ops = { static struct vcap_operations sparx5_vcap_ops = {
.validate_keyset = sparx5_vcap_validate_keyset, .validate_keyset = sparx5_vcap_validate_keyset,
...@@ -543,7 +521,6 @@ static struct vcap_operations sparx5_vcap_ops = { ...@@ -543,7 +521,6 @@ static struct vcap_operations sparx5_vcap_ops = {
.update = sparx5_vcap_update, .update = sparx5_vcap_update,
.move = sparx5_vcap_move, .move = sparx5_vcap_move,
.port_info = sparx5_port_info, .port_info = sparx5_port_info,
.enable = sparx5_vcap_enable,
}; };
/* Enable lookups per port and set the keyset generation: only IS2 for now */ /* Enable lookups per port and set the keyset generation: only IS2 for now */
...@@ -568,6 +545,12 @@ static void sparx5_vcap_port_key_selection(struct sparx5 *sparx5, ...@@ -568,6 +545,12 @@ static void sparx5_vcap_port_key_selection(struct sparx5 *sparx5,
ANA_ACL_VCAP_S2_KEY_SEL(portno, lookup)); ANA_ACL_VCAP_S2_KEY_SEL(portno, lookup));
} }
} }
/* IS2 lookups are in bit 0:3 */
for (portno = 0; portno < SPX5_PORTS; ++portno)
spx5_rmw(ANA_ACL_VCAP_S2_CFG_SEC_ENA_SET(0xf),
ANA_ACL_VCAP_S2_CFG_SEC_ENA,
sparx5,
ANA_ACL_VCAP_S2_CFG(portno));
} }
/* Disable lookups per port and set the keyset generation: only IS2 for now */ /* Disable lookups per port and set the keyset generation: only IS2 for now */
......
...@@ -259,11 +259,6 @@ struct vcap_operations { ...@@ -259,11 +259,6 @@ struct vcap_operations {
(struct net_device *ndev, (struct net_device *ndev,
struct vcap_admin *admin, struct vcap_admin *admin,
struct vcap_output_print *out); struct vcap_output_print *out);
/* enable/disable the lookups in a vcap instance */
int (*enable)
(struct net_device *ndev,
struct vcap_admin *admin,
bool enable);
}; };
/* VCAP API Client control interface */ /* VCAP API Client control interface */
......
...@@ -148,9 +148,10 @@ struct vcap_counter { ...@@ -148,9 +148,10 @@ struct vcap_counter {
bool sticky; bool sticky;
}; };
/* Enable/Disable the VCAP instance lookups. Chain id 0 means disable */ /* Enable/Disable the VCAP instance lookups */
int vcap_enable_lookups(struct vcap_control *vctrl, struct net_device *ndev, int vcap_enable_lookups(struct vcap_control *vctrl, struct net_device *ndev,
int chain_id, unsigned long cookie, bool enable); int from_cid, int to_cid, unsigned long cookie,
bool enable);
/* VCAP rule operations */ /* VCAP rule operations */
/* Allocate a rule and fill in the basic information */ /* Allocate a rule and fill in the basic information */
...@@ -216,6 +217,8 @@ const struct vcap_field *vcap_lookup_keyfield(struct vcap_rule *rule, ...@@ -216,6 +217,8 @@ const struct vcap_field *vcap_lookup_keyfield(struct vcap_rule *rule,
int vcap_lookup_rule_by_cookie(struct vcap_control *vctrl, u64 cookie); int vcap_lookup_rule_by_cookie(struct vcap_control *vctrl, u64 cookie);
/* Is the next chain id in the following lookup, possible in another VCAP */ /* Is the next chain id in the following lookup, possible in another VCAP */
bool vcap_is_next_lookup(struct vcap_control *vctrl, int cur_cid, int next_cid); bool vcap_is_next_lookup(struct vcap_control *vctrl, int cur_cid, int next_cid);
/* Is this chain id the last lookup of all VCAPs */
bool vcap_is_last_chain(struct vcap_control *vctrl, int cid);
/* Provide all rules via a callback interface */ /* Provide all rules via a callback interface */
int vcap_rule_iter(struct vcap_control *vctrl, int vcap_rule_iter(struct vcap_control *vctrl,
int (*callback)(void *, struct vcap_rule *), void *arg); int (*callback)(void *, struct vcap_rule *), void *arg);
...@@ -261,5 +264,4 @@ int vcap_rule_mod_action_u32(struct vcap_rule *rule, ...@@ -261,5 +264,4 @@ int vcap_rule_mod_action_u32(struct vcap_rule *rule,
/* Get a 32 bit key field value and mask from the rule */ /* Get a 32 bit key field value and mask from the rule */
int vcap_rule_get_key_u32(struct vcap_rule *rule, enum vcap_key_field key, int vcap_rule_get_key_u32(struct vcap_rule *rule, enum vcap_key_field key,
u32 *value, u32 *mask); u32 *value, u32 *mask);
#endif /* __VCAP_API_CLIENT__ */ #endif /* __VCAP_API_CLIENT__ */
...@@ -152,37 +152,48 @@ vcap_debugfs_show_rule_actionfield(struct vcap_control *vctrl, ...@@ -152,37 +152,48 @@ vcap_debugfs_show_rule_actionfield(struct vcap_control *vctrl,
out->prf(out->dst, "\n"); out->prf(out->dst, "\n");
} }
static int vcap_debugfs_show_rule_keyset(struct vcap_rule_internal *ri, static int vcap_debugfs_show_keysets(struct vcap_rule_internal *ri,
struct vcap_output_print *out) struct vcap_output_print *out)
{ {
struct vcap_control *vctrl = ri->vctrl;
struct vcap_admin *admin = ri->admin; struct vcap_admin *admin = ri->admin;
enum vcap_keyfield_set keysets[10]; enum vcap_keyfield_set keysets[10];
const struct vcap_field *keyfield;
enum vcap_type vt = admin->vtype;
struct vcap_client_keyfield *ckf;
struct vcap_keyset_list matches; struct vcap_keyset_list matches;
u32 *maskstream; int err;
u32 *keystream;
int res;
keystream = admin->cache.keystream;
maskstream = admin->cache.maskstream;
matches.keysets = keysets; matches.keysets = keysets;
matches.cnt = 0; matches.cnt = 0;
matches.max = ARRAY_SIZE(keysets); matches.max = ARRAY_SIZE(keysets);
res = vcap_find_keystream_keysets(vctrl, vt, keystream, maskstream,
false, 0, &matches); if (ri->state == VCAP_RS_DISABLED)
if (res < 0) { err = vcap_rule_get_keysets(ri, &matches);
else
err = vcap_find_keystream_keysets(ri->vctrl, admin->vtype,
admin->cache.keystream,
admin->cache.maskstream,
false, 0, &matches);
if (err) {
pr_err("%s:%d: could not find valid keysets: %d\n", pr_err("%s:%d: could not find valid keysets: %d\n",
__func__, __LINE__, res); __func__, __LINE__, err);
return -EINVAL; return err;
} }
out->prf(out->dst, " keysets:"); out->prf(out->dst, " keysets:");
for (int idx = 0; idx < matches.cnt; ++idx) for (int idx = 0; idx < matches.cnt; ++idx)
out->prf(out->dst, " %s", out->prf(out->dst, " %s",
vcap_keyset_name(vctrl, matches.keysets[idx])); vcap_keyset_name(ri->vctrl, matches.keysets[idx]));
out->prf(out->dst, "\n"); out->prf(out->dst, "\n");
return 0;
}
static int vcap_debugfs_show_rule_keyset(struct vcap_rule_internal *ri,
struct vcap_output_print *out)
{
struct vcap_control *vctrl = ri->vctrl;
struct vcap_admin *admin = ri->admin;
const struct vcap_field *keyfield;
struct vcap_client_keyfield *ckf;
vcap_debugfs_show_keysets(ri, out);
out->prf(out->dst, " keyset_sw: %d\n", ri->keyset_sw); out->prf(out->dst, " keyset_sw: %d\n", ri->keyset_sw);
out->prf(out->dst, " keyset_sw_regs: %d\n", ri->keyset_sw_regs); out->prf(out->dst, " keyset_sw_regs: %d\n", ri->keyset_sw_regs);
...@@ -233,6 +244,18 @@ static void vcap_show_admin_rule(struct vcap_control *vctrl, ...@@ -233,6 +244,18 @@ static void vcap_show_admin_rule(struct vcap_control *vctrl,
out->prf(out->dst, " chain_id: %d\n", ri->data.vcap_chain_id); out->prf(out->dst, " chain_id: %d\n", ri->data.vcap_chain_id);
out->prf(out->dst, " user: %d\n", ri->data.user); out->prf(out->dst, " user: %d\n", ri->data.user);
out->prf(out->dst, " priority: %d\n", ri->data.priority); out->prf(out->dst, " priority: %d\n", ri->data.priority);
out->prf(out->dst, " state: ");
switch (ri->state) {
case VCAP_RS_PERMANENT:
out->prf(out->dst, "permanent\n");
break;
case VCAP_RS_DISABLED:
out->prf(out->dst, "disabled\n");
break;
case VCAP_RS_ENABLED:
out->prf(out->dst, "enabled\n");
break;
}
vcap_debugfs_show_rule_keyset(ri, out); vcap_debugfs_show_rule_keyset(ri, out);
vcap_debugfs_show_rule_actionset(ri, out); vcap_debugfs_show_rule_actionset(ri, out);
} }
......
...@@ -221,13 +221,6 @@ static int vcap_test_port_info(struct net_device *ndev, ...@@ -221,13 +221,6 @@ static int vcap_test_port_info(struct net_device *ndev,
return 0; return 0;
} }
static int vcap_test_enable(struct net_device *ndev,
struct vcap_admin *admin,
bool enable)
{
return 0;
}
static struct vcap_operations test_callbacks = { static struct vcap_operations test_callbacks = {
.validate_keyset = test_val_keyset, .validate_keyset = test_val_keyset,
.add_default_fields = test_add_def_fields, .add_default_fields = test_add_def_fields,
...@@ -238,7 +231,6 @@ static struct vcap_operations test_callbacks = { ...@@ -238,7 +231,6 @@ static struct vcap_operations test_callbacks = {
.update = test_cache_update, .update = test_cache_update,
.move = test_cache_move, .move = test_cache_move,
.port_info = vcap_test_port_info, .port_info = vcap_test_port_info,
.enable = vcap_test_enable,
}; };
static struct vcap_control test_vctrl = { static struct vcap_control test_vctrl = {
...@@ -253,6 +245,7 @@ static void vcap_test_api_init(struct vcap_admin *admin) ...@@ -253,6 +245,7 @@ static void vcap_test_api_init(struct vcap_admin *admin)
INIT_LIST_HEAD(&test_vctrl.list); INIT_LIST_HEAD(&test_vctrl.list);
INIT_LIST_HEAD(&admin->list); INIT_LIST_HEAD(&admin->list);
INIT_LIST_HEAD(&admin->rules); INIT_LIST_HEAD(&admin->rules);
INIT_LIST_HEAD(&admin->enabled);
list_add_tail(&admin->list, &test_vctrl.list); list_add_tail(&admin->list, &test_vctrl.list);
memset(test_updateaddr, 0, sizeof(test_updateaddr)); memset(test_updateaddr, 0, sizeof(test_updateaddr));
test_updateaddridx = 0; test_updateaddridx = 0;
...@@ -452,6 +445,7 @@ static const char * const test_admin_expect[] = { ...@@ -452,6 +445,7 @@ static const char * const test_admin_expect[] = {
" chain_id: 0\n", " chain_id: 0\n",
" user: 0\n", " user: 0\n",
" priority: 0\n", " priority: 0\n",
" state: permanent\n",
" keysets: VCAP_KFS_MAC_ETYPE\n", " keysets: VCAP_KFS_MAC_ETYPE\n",
" keyset_sw: 6\n", " keyset_sw: 6\n",
" keyset_sw_regs: 2\n", " keyset_sw_regs: 2\n",
......
...@@ -211,13 +211,6 @@ static int vcap_test_port_info(struct net_device *ndev, ...@@ -211,13 +211,6 @@ static int vcap_test_port_info(struct net_device *ndev,
return 0; return 0;
} }
static int vcap_test_enable(struct net_device *ndev,
struct vcap_admin *admin,
bool enable)
{
return 0;
}
static struct vcap_operations test_callbacks = { static struct vcap_operations test_callbacks = {
.validate_keyset = test_val_keyset, .validate_keyset = test_val_keyset,
.add_default_fields = test_add_def_fields, .add_default_fields = test_add_def_fields,
...@@ -228,7 +221,6 @@ static struct vcap_operations test_callbacks = { ...@@ -228,7 +221,6 @@ static struct vcap_operations test_callbacks = {
.update = test_cache_update, .update = test_cache_update,
.move = test_cache_move, .move = test_cache_move,
.port_info = vcap_test_port_info, .port_info = vcap_test_port_info,
.enable = vcap_test_enable,
}; };
static struct vcap_control test_vctrl = { static struct vcap_control test_vctrl = {
...@@ -243,6 +235,7 @@ static void vcap_test_api_init(struct vcap_admin *admin) ...@@ -243,6 +235,7 @@ static void vcap_test_api_init(struct vcap_admin *admin)
INIT_LIST_HEAD(&test_vctrl.list); INIT_LIST_HEAD(&test_vctrl.list);
INIT_LIST_HEAD(&admin->list); INIT_LIST_HEAD(&admin->list);
INIT_LIST_HEAD(&admin->rules); INIT_LIST_HEAD(&admin->rules);
INIT_LIST_HEAD(&admin->enabled);
list_add_tail(&admin->list, &test_vctrl.list); list_add_tail(&admin->list, &test_vctrl.list);
memset(test_updateaddr, 0, sizeof(test_updateaddr)); memset(test_updateaddr, 0, sizeof(test_updateaddr));
test_updateaddridx = 0; test_updateaddridx = 0;
...@@ -1312,8 +1305,8 @@ static void vcap_api_encode_rule_test(struct kunit *test) ...@@ -1312,8 +1305,8 @@ static void vcap_api_encode_rule_test(struct kunit *test)
struct vcap_admin is2_admin = { struct vcap_admin is2_admin = {
.vtype = VCAP_TYPE_IS2, .vtype = VCAP_TYPE_IS2,
.first_cid = 10000, .first_cid = 8000000,
.last_cid = 19999, .last_cid = 8099999,
.lookups = 4, .lookups = 4,
.last_valid_addr = 3071, .last_valid_addr = 3071,
.first_valid_addr = 0, .first_valid_addr = 0,
...@@ -1326,7 +1319,7 @@ static void vcap_api_encode_rule_test(struct kunit *test) ...@@ -1326,7 +1319,7 @@ static void vcap_api_encode_rule_test(struct kunit *test)
}; };
struct vcap_rule *rule; struct vcap_rule *rule;
struct vcap_rule_internal *ri; struct vcap_rule_internal *ri;
int vcap_chain_id = 10005; int vcap_chain_id = 8000000;
enum vcap_user user = VCAP_USER_VCAP_UTIL; enum vcap_user user = VCAP_USER_VCAP_UTIL;
u16 priority = 10; u16 priority = 10;
int id = 100; int id = 100;
...@@ -1343,8 +1336,8 @@ static void vcap_api_encode_rule_test(struct kunit *test) ...@@ -1343,8 +1336,8 @@ static void vcap_api_encode_rule_test(struct kunit *test)
u32 port_mask_rng_mask = 0x0f; u32 port_mask_rng_mask = 0x0f;
u32 igr_port_mask_value = 0xffabcd01; u32 igr_port_mask_value = 0xffabcd01;
u32 igr_port_mask_mask = ~0; u32 igr_port_mask_mask = ~0;
/* counter is not written yet, so it is not in expwriteaddr */ /* counter is written as the last operation */
u32 expwriteaddr[] = {792, 793, 794, 795, 796, 797, 0}; u32 expwriteaddr[] = {792, 793, 794, 795, 796, 797, 792};
int idx; int idx;
vcap_test_api_init(&is2_admin); vcap_test_api_init(&is2_admin);
...@@ -1398,6 +1391,11 @@ static void vcap_api_encode_rule_test(struct kunit *test) ...@@ -1398,6 +1391,11 @@ static void vcap_api_encode_rule_test(struct kunit *test)
KUNIT_EXPECT_EQ(test, 2, ri->keyset_sw_regs); KUNIT_EXPECT_EQ(test, 2, ri->keyset_sw_regs);
KUNIT_EXPECT_EQ(test, 4, ri->actionset_sw_regs); KUNIT_EXPECT_EQ(test, 4, ri->actionset_sw_regs);
/* Enable lookup, so the rule will be written */
ret = vcap_enable_lookups(&test_vctrl, &test_netdev, 0,
rule->vcap_chain_id, rule->cookie, true);
KUNIT_EXPECT_EQ(test, 0, ret);
/* Add rule with write callback */ /* Add rule with write callback */
ret = vcap_add_rule(rule); ret = vcap_add_rule(rule);
KUNIT_EXPECT_EQ(test, 0, ret); KUNIT_EXPECT_EQ(test, 0, ret);
...@@ -1872,7 +1870,7 @@ static void vcap_api_next_lookup_basic_test(struct kunit *test) ...@@ -1872,7 +1870,7 @@ static void vcap_api_next_lookup_basic_test(struct kunit *test)
ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8301000); ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8301000);
KUNIT_EXPECT_EQ(test, false, ret); KUNIT_EXPECT_EQ(test, false, ret);
ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8401000); ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8401000);
KUNIT_EXPECT_EQ(test, true, ret); KUNIT_EXPECT_EQ(test, false, ret);
} }
static void vcap_api_next_lookup_advanced_test(struct kunit *test) static void vcap_api_next_lookup_advanced_test(struct kunit *test)
...@@ -1933,9 +1931,9 @@ static void vcap_api_next_lookup_advanced_test(struct kunit *test) ...@@ -1933,9 +1931,9 @@ static void vcap_api_next_lookup_advanced_test(struct kunit *test)
ret = vcap_is_next_lookup(&test_vctrl, 1100000, 1201000); ret = vcap_is_next_lookup(&test_vctrl, 1100000, 1201000);
KUNIT_EXPECT_EQ(test, true, ret); KUNIT_EXPECT_EQ(test, true, ret);
ret = vcap_is_next_lookup(&test_vctrl, 1100000, 1301000); ret = vcap_is_next_lookup(&test_vctrl, 1100000, 1301000);
KUNIT_EXPECT_EQ(test, false, ret); KUNIT_EXPECT_EQ(test, true, ret);
ret = vcap_is_next_lookup(&test_vctrl, 1100000, 8101000); ret = vcap_is_next_lookup(&test_vctrl, 1100000, 8101000);
KUNIT_EXPECT_EQ(test, false, ret); KUNIT_EXPECT_EQ(test, true, ret);
ret = vcap_is_next_lookup(&test_vctrl, 1300000, 1401000); ret = vcap_is_next_lookup(&test_vctrl, 1300000, 1401000);
KUNIT_EXPECT_EQ(test, true, ret); KUNIT_EXPECT_EQ(test, true, ret);
ret = vcap_is_next_lookup(&test_vctrl, 1400000, 1501000); ret = vcap_is_next_lookup(&test_vctrl, 1400000, 1501000);
...@@ -1951,7 +1949,7 @@ static void vcap_api_next_lookup_advanced_test(struct kunit *test) ...@@ -1951,7 +1949,7 @@ static void vcap_api_next_lookup_advanced_test(struct kunit *test)
ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8301000); ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8301000);
KUNIT_EXPECT_EQ(test, false, ret); KUNIT_EXPECT_EQ(test, false, ret);
ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8401000); ret = vcap_is_next_lookup(&test_vctrl, 8300000, 8401000);
KUNIT_EXPECT_EQ(test, true, ret); KUNIT_EXPECT_EQ(test, false, ret);
} }
static void vcap_api_filter_unsupported_keys_test(struct kunit *test) static void vcap_api_filter_unsupported_keys_test(struct kunit *test)
......
...@@ -13,6 +13,12 @@ ...@@ -13,6 +13,12 @@
#define to_intrule(rule) container_of((rule), struct vcap_rule_internal, data) #define to_intrule(rule) container_of((rule), struct vcap_rule_internal, data)
enum vcap_rule_state {
VCAP_RS_PERMANENT, /* the rule is always stored in HW */
VCAP_RS_ENABLED, /* enabled in HW but can be disabled */
VCAP_RS_DISABLED, /* disabled (stored in SW) and can be enabled */
};
/* Private VCAP API rule data */ /* Private VCAP API rule data */
struct vcap_rule_internal { struct vcap_rule_internal {
struct vcap_rule data; /* provided by the client */ struct vcap_rule data; /* provided by the client */
...@@ -29,6 +35,7 @@ struct vcap_rule_internal { ...@@ -29,6 +35,7 @@ struct vcap_rule_internal {
u32 addr; /* address in the VCAP at insertion */ u32 addr; /* address in the VCAP at insertion */
u32 counter_id; /* counter id (if a dedicated counter is available) */ u32 counter_id; /* counter id (if a dedicated counter is available) */
struct vcap_counter counter; /* last read counter value */ struct vcap_counter counter; /* last read counter value */
enum vcap_rule_state state; /* rule storage state */
}; };
/* Bit iterator for the VCAP cache streams */ /* Bit iterator for the VCAP cache streams */
...@@ -43,8 +50,6 @@ struct vcap_stream_iter { ...@@ -43,8 +50,6 @@ struct vcap_stream_iter {
/* Check that the control has a valid set of callbacks */ /* Check that the control has a valid set of callbacks */
int vcap_api_check(struct vcap_control *ctrl); int vcap_api_check(struct vcap_control *ctrl);
/* Make a shallow copy of the rule without the fields */
struct vcap_rule_internal *vcap_dup_rule(struct vcap_rule_internal *ri);
/* Erase the VCAP cache area used or encoding and decoding */ /* Erase the VCAP cache area used or encoding and decoding */
void vcap_erase_cache(struct vcap_rule_internal *ri); void vcap_erase_cache(struct vcap_rule_internal *ri);
...@@ -110,4 +115,7 @@ int vcap_find_keystream_keysets(struct vcap_control *vctrl, enum vcap_type vt, ...@@ -110,4 +115,7 @@ int vcap_find_keystream_keysets(struct vcap_control *vctrl, enum vcap_type vt,
u32 *keystream, u32 *mskstream, bool mask, u32 *keystream, u32 *mskstream, bool mask,
int sw_max, struct vcap_keyset_list *kslist); int sw_max, struct vcap_keyset_list *kslist);
/* Get the keysets that matches the rule key type/mask */
int vcap_rule_get_keysets(struct vcap_rule_internal *ri,
struct vcap_keyset_list *matches);
#endif /* __VCAP_API_PRIVATE__ */ #endif /* __VCAP_API_PRIVATE__ */
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