mlxsw: core_acl_flex_actions: Implement flow_offload action cookie offload

Track cookies coming down to driver by flow_offload.
Assign a cookie_index to each unique cookie binary. Use previously
defined "Trap with userdef" flex action to ask HW to pass cookie_index
alongside with the dropped packets.

Signed-off-by: Jiri Pirko <jiri@mellanox.com>
Signed-off-by: Ido Schimmel <idosch@mellanox.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Jiri Pirko 2020-02-25 11:45:23 +01:00 committed by David S. Miller
parent ec12165195
commit 6d19d2bdc8
5 changed files with 257 additions and 6 deletions

View File

@ -7,6 +7,9 @@
#include <linux/errno.h>
#include <linux/rhashtable.h>
#include <linux/list.h>
#include <linux/idr.h>
#include <linux/refcount.h>
#include <net/flow_offload.h>
#include "item.h"
#include "trap.h"
@ -63,6 +66,8 @@ struct mlxsw_afa {
void *ops_priv;
struct rhashtable set_ht;
struct rhashtable fwd_entry_ht;
struct rhashtable cookie_ht;
struct idr cookie_idr;
};
#define MLXSW_AFA_SET_LEN 0xA8
@ -121,6 +126,55 @@ static const struct rhashtable_params mlxsw_afa_fwd_entry_ht_params = {
.automatic_shrinking = true,
};
struct mlxsw_afa_cookie {
struct rhash_head ht_node;
refcount_t ref_count;
struct rcu_head rcu;
u32 cookie_index;
struct flow_action_cookie fa_cookie;
};
static u32 mlxsw_afa_cookie_hash(const struct flow_action_cookie *fa_cookie,
u32 seed)
{
return jhash2((u32 *) fa_cookie->cookie,
fa_cookie->cookie_len / sizeof(u32), seed);
}
static u32 mlxsw_afa_cookie_key_hashfn(const void *data, u32 len, u32 seed)
{
const struct flow_action_cookie *fa_cookie = data;
return mlxsw_afa_cookie_hash(fa_cookie, seed);
}
static u32 mlxsw_afa_cookie_obj_hashfn(const void *data, u32 len, u32 seed)
{
const struct mlxsw_afa_cookie *cookie = data;
return mlxsw_afa_cookie_hash(&cookie->fa_cookie, seed);
}
static int mlxsw_afa_cookie_obj_cmpfn(struct rhashtable_compare_arg *arg,
const void *obj)
{
const struct flow_action_cookie *fa_cookie = arg->key;
const struct mlxsw_afa_cookie *cookie = obj;
if (cookie->fa_cookie.cookie_len == fa_cookie->cookie_len)
return memcmp(cookie->fa_cookie.cookie, fa_cookie->cookie,
fa_cookie->cookie_len);
return 1;
}
static const struct rhashtable_params mlxsw_afa_cookie_ht_params = {
.head_offset = offsetof(struct mlxsw_afa_cookie, ht_node),
.hashfn = mlxsw_afa_cookie_key_hashfn,
.obj_hashfn = mlxsw_afa_cookie_obj_hashfn,
.obj_cmpfn = mlxsw_afa_cookie_obj_cmpfn,
.automatic_shrinking = true,
};
struct mlxsw_afa *mlxsw_afa_create(unsigned int max_acts_per_set,
const struct mlxsw_afa_ops *ops,
void *ops_priv)
@ -138,11 +192,18 @@ struct mlxsw_afa *mlxsw_afa_create(unsigned int max_acts_per_set,
&mlxsw_afa_fwd_entry_ht_params);
if (err)
goto err_fwd_entry_rhashtable_init;
err = rhashtable_init(&mlxsw_afa->cookie_ht,
&mlxsw_afa_cookie_ht_params);
if (err)
goto err_cookie_rhashtable_init;
idr_init(&mlxsw_afa->cookie_idr);
mlxsw_afa->max_acts_per_set = max_acts_per_set;
mlxsw_afa->ops = ops;
mlxsw_afa->ops_priv = ops_priv;
return mlxsw_afa;
err_cookie_rhashtable_init:
rhashtable_destroy(&mlxsw_afa->fwd_entry_ht);
err_fwd_entry_rhashtable_init:
rhashtable_destroy(&mlxsw_afa->set_ht);
err_set_rhashtable_init:
@ -153,6 +214,9 @@ EXPORT_SYMBOL(mlxsw_afa_create);
void mlxsw_afa_destroy(struct mlxsw_afa *mlxsw_afa)
{
WARN_ON(!idr_is_empty(&mlxsw_afa->cookie_idr));
idr_destroy(&mlxsw_afa->cookie_idr);
rhashtable_destroy(&mlxsw_afa->cookie_ht);
rhashtable_destroy(&mlxsw_afa->fwd_entry_ht);
rhashtable_destroy(&mlxsw_afa->set_ht);
kfree(mlxsw_afa);
@ -627,6 +691,135 @@ err_counter_index_get:
return ERR_PTR(err);
}
/* 20 bits is a maximum that hardware can handle in trap with userdef action
* and carry along with the trapped packet.
*/
#define MLXSW_AFA_COOKIE_INDEX_BITS 20
#define MLXSW_AFA_COOKIE_INDEX_MAX ((1 << MLXSW_AFA_COOKIE_INDEX_BITS) - 1)
static struct mlxsw_afa_cookie *
mlxsw_afa_cookie_create(struct mlxsw_afa *mlxsw_afa,
const struct flow_action_cookie *fa_cookie)
{
struct mlxsw_afa_cookie *cookie;
u32 cookie_index;
int err;
cookie = kzalloc(sizeof(*cookie) + fa_cookie->cookie_len, GFP_KERNEL);
if (!cookie)
return ERR_PTR(-ENOMEM);
refcount_set(&cookie->ref_count, 1);
memcpy(&cookie->fa_cookie, fa_cookie,
sizeof(*fa_cookie) + fa_cookie->cookie_len);
err = rhashtable_insert_fast(&mlxsw_afa->cookie_ht, &cookie->ht_node,
mlxsw_afa_cookie_ht_params);
if (err)
goto err_rhashtable_insert;
/* Start cookie indexes with 1. Leave the 0 index unused. Packets
* that come from the HW which are not dropped by drop-with-cookie
* action are going to pass cookie_index 0 to lookup.
*/
cookie_index = 1;
err = idr_alloc_u32(&mlxsw_afa->cookie_idr, cookie, &cookie_index,
MLXSW_AFA_COOKIE_INDEX_MAX, GFP_KERNEL);
if (err)
goto err_idr_alloc;
cookie->cookie_index = cookie_index;
return cookie;
err_idr_alloc:
rhashtable_remove_fast(&mlxsw_afa->cookie_ht, &cookie->ht_node,
mlxsw_afa_cookie_ht_params);
err_rhashtable_insert:
kfree(cookie);
return ERR_PTR(err);
}
static void mlxsw_afa_cookie_destroy(struct mlxsw_afa *mlxsw_afa,
struct mlxsw_afa_cookie *cookie)
{
idr_remove(&mlxsw_afa->cookie_idr, cookie->cookie_index);
rhashtable_remove_fast(&mlxsw_afa->cookie_ht, &cookie->ht_node,
mlxsw_afa_cookie_ht_params);
kfree_rcu(cookie, rcu);
}
static struct mlxsw_afa_cookie *
mlxsw_afa_cookie_get(struct mlxsw_afa *mlxsw_afa,
const struct flow_action_cookie *fa_cookie)
{
struct mlxsw_afa_cookie *cookie;
cookie = rhashtable_lookup_fast(&mlxsw_afa->cookie_ht, fa_cookie,
mlxsw_afa_cookie_ht_params);
if (cookie) {
refcount_inc(&cookie->ref_count);
return cookie;
}
return mlxsw_afa_cookie_create(mlxsw_afa, fa_cookie);
}
static void mlxsw_afa_cookie_put(struct mlxsw_afa *mlxsw_afa,
struct mlxsw_afa_cookie *cookie)
{
if (!refcount_dec_and_test(&cookie->ref_count))
return;
mlxsw_afa_cookie_destroy(mlxsw_afa, cookie);
}
struct mlxsw_afa_cookie_ref {
struct mlxsw_afa_resource resource;
struct mlxsw_afa_cookie *cookie;
};
static void
mlxsw_afa_cookie_ref_destroy(struct mlxsw_afa_block *block,
struct mlxsw_afa_cookie_ref *cookie_ref)
{
mlxsw_afa_resource_del(&cookie_ref->resource);
mlxsw_afa_cookie_put(block->afa, cookie_ref->cookie);
kfree(cookie_ref);
}
static void
mlxsw_afa_cookie_ref_destructor(struct mlxsw_afa_block *block,
struct mlxsw_afa_resource *resource)
{
struct mlxsw_afa_cookie_ref *cookie_ref;
cookie_ref = container_of(resource, struct mlxsw_afa_cookie_ref,
resource);
mlxsw_afa_cookie_ref_destroy(block, cookie_ref);
}
static struct mlxsw_afa_cookie_ref *
mlxsw_afa_cookie_ref_create(struct mlxsw_afa_block *block,
const struct flow_action_cookie *fa_cookie)
{
struct mlxsw_afa_cookie_ref *cookie_ref;
struct mlxsw_afa_cookie *cookie;
int err;
cookie_ref = kzalloc(sizeof(*cookie_ref), GFP_KERNEL);
if (!cookie_ref)
return ERR_PTR(-ENOMEM);
cookie = mlxsw_afa_cookie_get(block->afa, fa_cookie);
if (IS_ERR(cookie)) {
err = PTR_ERR(cookie);
goto err_cookie_get;
}
cookie_ref->cookie = cookie;
cookie_ref->resource.destructor = mlxsw_afa_cookie_ref_destructor;
mlxsw_afa_resource_add(block, &cookie_ref->resource);
return cookie_ref;
err_cookie_get:
kfree(cookie_ref);
return ERR_PTR(err);
}
#define MLXSW_AFA_ONE_ACTION_LEN 32
#define MLXSW_AFA_PAYLOAD_OFFSET 4
@ -839,7 +1032,8 @@ mlxsw_afa_trap_mirror_pack(char *payload, bool mirror_enable,
mlxsw_afa_trap_mirror_agent_set(payload, mirror_agent);
}
int mlxsw_afa_block_append_drop(struct mlxsw_afa_block *block, bool ingress)
static int mlxsw_afa_block_append_drop_plain(struct mlxsw_afa_block *block,
bool ingress)
{
char *act = mlxsw_afa_block_append_action(block, MLXSW_AFA_TRAP_CODE,
MLXSW_AFA_TRAP_SIZE);
@ -852,6 +1046,53 @@ int mlxsw_afa_block_append_drop(struct mlxsw_afa_block *block, bool ingress)
MLXSW_TRAP_ID_DISCARD_EGRESS_ACL);
return 0;
}
static int
mlxsw_afa_block_append_drop_with_cookie(struct mlxsw_afa_block *block,
bool ingress,
const struct flow_action_cookie *fa_cookie,
struct netlink_ext_ack *extack)
{
struct mlxsw_afa_cookie_ref *cookie_ref;
u32 cookie_index;
char *act;
int err;
cookie_ref = mlxsw_afa_cookie_ref_create(block, fa_cookie);
if (IS_ERR(cookie_ref)) {
NL_SET_ERR_MSG_MOD(extack, "Cannot create cookie for drop action");
return PTR_ERR(cookie_ref);
}
cookie_index = cookie_ref->cookie->cookie_index;
act = mlxsw_afa_block_append_action(block, MLXSW_AFA_TRAPWU_CODE,
MLXSW_AFA_TRAPWU_SIZE);
if (IS_ERR(act)) {
NL_SET_ERR_MSG_MOD(extack, "Cannot append drop with cookie action");
err = PTR_ERR(act);
goto err_append_action;
}
mlxsw_afa_trapwu_pack(act, MLXSW_AFA_TRAP_TRAP_ACTION_TRAP,
MLXSW_AFA_TRAP_FORWARD_ACTION_DISCARD,
ingress ? MLXSW_TRAP_ID_DISCARD_INGRESS_ACL :
MLXSW_TRAP_ID_DISCARD_EGRESS_ACL,
cookie_index);
return 0;
err_append_action:
mlxsw_afa_cookie_ref_destroy(block, cookie_ref);
return err;
}
int mlxsw_afa_block_append_drop(struct mlxsw_afa_block *block, bool ingress,
const struct flow_action_cookie *fa_cookie,
struct netlink_ext_ack *extack)
{
return fa_cookie ?
mlxsw_afa_block_append_drop_with_cookie(block, ingress,
fa_cookie, extack) :
mlxsw_afa_block_append_drop_plain(block, ingress);
}
EXPORT_SYMBOL(mlxsw_afa_block_append_drop);
int mlxsw_afa_block_append_trap(struct mlxsw_afa_block *block, u16 trap_id)

View File

@ -6,6 +6,7 @@
#include <linux/types.h>
#include <linux/netdevice.h>
#include <net/flow_offload.h>
struct mlxsw_afa;
struct mlxsw_afa_block;
@ -42,7 +43,9 @@ int mlxsw_afa_block_activity_get(struct mlxsw_afa_block *block, bool *activity);
int mlxsw_afa_block_continue(struct mlxsw_afa_block *block);
int mlxsw_afa_block_jump(struct mlxsw_afa_block *block, u16 group_id);
int mlxsw_afa_block_terminate(struct mlxsw_afa_block *block);
int mlxsw_afa_block_append_drop(struct mlxsw_afa_block *block, bool ingress);
int mlxsw_afa_block_append_drop(struct mlxsw_afa_block *block, bool ingress,
const struct flow_action_cookie *fa_cookie,
struct netlink_ext_ack *extack);
int mlxsw_afa_block_append_trap(struct mlxsw_afa_block *block, u16 trap_id);
int mlxsw_afa_block_append_trap_and_forward(struct mlxsw_afa_block *block,
u16 trap_id);

View File

@ -19,6 +19,7 @@
#include <net/pkt_cls.h>
#include <net/red.h>
#include <net/vxlan.h>
#include <net/flow_offload.h>
#include "port.h"
#include "core.h"
@ -726,7 +727,9 @@ int mlxsw_sp_acl_rulei_act_jump(struct mlxsw_sp_acl_rule_info *rulei,
u16 group_id);
int mlxsw_sp_acl_rulei_act_terminate(struct mlxsw_sp_acl_rule_info *rulei);
int mlxsw_sp_acl_rulei_act_drop(struct mlxsw_sp_acl_rule_info *rulei,
bool ingress);
bool ingress,
const struct flow_action_cookie *fa_cookie,
struct netlink_ext_ack *extack);
int mlxsw_sp_acl_rulei_act_trap(struct mlxsw_sp_acl_rule_info *rulei);
int mlxsw_sp_acl_rulei_act_mirror(struct mlxsw_sp *mlxsw_sp,
struct mlxsw_sp_acl_rule_info *rulei,

View File

@ -536,9 +536,12 @@ int mlxsw_sp_acl_rulei_act_terminate(struct mlxsw_sp_acl_rule_info *rulei)
}
int mlxsw_sp_acl_rulei_act_drop(struct mlxsw_sp_acl_rule_info *rulei,
bool ingress)
bool ingress,
const struct flow_action_cookie *fa_cookie,
struct netlink_ext_ack *extack)
{
return mlxsw_afa_block_append_drop(rulei->act_block, ingress);
return mlxsw_afa_block_append_drop(rulei->act_block, ingress,
fa_cookie, extack);
}
int mlxsw_sp_acl_rulei_act_trap(struct mlxsw_sp_acl_rule_info *rulei)

View File

@ -49,7 +49,8 @@ static int mlxsw_sp_flower_parse_actions(struct mlxsw_sp *mlxsw_sp,
return -EOPNOTSUPP;
}
ingress = mlxsw_sp_acl_block_is_ingress_bound(block);
err = mlxsw_sp_acl_rulei_act_drop(rulei, ingress);
err = mlxsw_sp_acl_rulei_act_drop(rulei, ingress,
act->cookie, extack);
if (err) {
NL_SET_ERR_MSG_MOD(extack, "Cannot append drop action");
return err;