net/mlx5e: TC, Fix ct_clear overwriting ct action metadata
ct_clear action is translated to clearing reg_c metadata which holds ct state and zone information using mod header actions. These actions are allocated during the actions parsing, as part of the flow attributes main mod header action list. If ct action exists in the rule, the flow's main mod header is used only in the post action table rule, after the ct tables which set the ct info in the reg_c as part of the ct actions. Therefore, if the original rule has a ct_clear action followed by a ct action, the ct action reg_c setting will be done first and will be followed by the ct_clear resetting reg_c and overwriting the ct info. Fix this by moving the ct_clear mod header actions allocation from the ct action parsing stage to the ct action post parsing stage where it is already known if ct_clear is followed by a ct action. In such case, we skip the mod header actions allocation for the ct clear since the ct action will write to reg_c anyway after clearing it. Fixes: 806401c20a0f ("net/mlx5e: CT, Fix multiple allocations and memleak of mod acts") Signed-off-by: Ariel Levkovich <lariel@nvidia.com> Reviewed-by: Paul Blakey <paulb@nvidia.com> Reviewed-by: Roi Dayan <roid@nvidia.com> Reviewed-by: Maor Dickman <maord@nvidia.com> Signed-off-by: Saeed Mahameed <saeedm@nvidia.com>
This commit is contained in:
parent
4a2a664ed8
commit
087032ee70
@ -45,12 +45,41 @@ tc_act_parse_ct(struct mlx5e_tc_act_parse_state *parse_state,
|
||||
if (mlx5e_is_eswitch_flow(parse_state->flow))
|
||||
attr->esw_attr->split_count = attr->esw_attr->out_count;
|
||||
|
||||
if (!clear_action) {
|
||||
if (clear_action) {
|
||||
parse_state->ct_clear = true;
|
||||
} else {
|
||||
attr->flags |= MLX5_ATTR_FLAG_CT;
|
||||
flow_flag_set(parse_state->flow, CT);
|
||||
parse_state->ct = true;
|
||||
}
|
||||
parse_state->ct_clear = clear_action;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int
|
||||
tc_act_post_parse_ct(struct mlx5e_tc_act_parse_state *parse_state,
|
||||
struct mlx5e_priv *priv,
|
||||
struct mlx5_flow_attr *attr)
|
||||
{
|
||||
struct mlx5e_tc_mod_hdr_acts *mod_acts = &attr->parse_attr->mod_hdr_acts;
|
||||
int err;
|
||||
|
||||
/* If ct action exist, we can ignore previous ct_clear actions */
|
||||
if (parse_state->ct)
|
||||
return 0;
|
||||
|
||||
if (parse_state->ct_clear) {
|
||||
err = mlx5_tc_ct_set_ct_clear_regs(parse_state->ct_priv, mod_acts);
|
||||
if (err) {
|
||||
NL_SET_ERR_MSG_MOD(parse_state->extack,
|
||||
"Failed to set registers for ct clear");
|
||||
return err;
|
||||
}
|
||||
attr->action |= MLX5_FLOW_CONTEXT_ACTION_MOD_HDR;
|
||||
|
||||
/* Prevent handling of additional, redundant clear actions */
|
||||
parse_state->ct_clear = false;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@ -70,5 +99,6 @@ struct mlx5e_tc_act mlx5e_tc_act_ct = {
|
||||
.can_offload = tc_act_can_offload_ct,
|
||||
.parse_action = tc_act_parse_ct,
|
||||
.is_multi_table_act = tc_act_is_multi_table_act_ct,
|
||||
.post_parse = tc_act_post_parse_ct,
|
||||
};
|
||||
|
||||
|
@ -582,6 +582,12 @@ mlx5_tc_ct_entry_set_registers(struct mlx5_tc_ct_priv *ct_priv,
|
||||
return 0;
|
||||
}
|
||||
|
||||
int mlx5_tc_ct_set_ct_clear_regs(struct mlx5_tc_ct_priv *priv,
|
||||
struct mlx5e_tc_mod_hdr_acts *mod_acts)
|
||||
{
|
||||
return mlx5_tc_ct_entry_set_registers(priv, mod_acts, 0, 0, 0, 0);
|
||||
}
|
||||
|
||||
static int
|
||||
mlx5_tc_ct_parse_mangle_to_mod_act(struct flow_action_entry *act,
|
||||
char *modact)
|
||||
@ -1410,9 +1416,6 @@ mlx5_tc_ct_parse_action(struct mlx5_tc_ct_priv *priv,
|
||||
const struct flow_action_entry *act,
|
||||
struct netlink_ext_ack *extack)
|
||||
{
|
||||
bool clear_action = act->ct.action & TCA_CT_ACT_CLEAR;
|
||||
int err;
|
||||
|
||||
if (!priv) {
|
||||
NL_SET_ERR_MSG_MOD(extack,
|
||||
"offload of ct action isn't available");
|
||||
@ -1423,17 +1426,6 @@ mlx5_tc_ct_parse_action(struct mlx5_tc_ct_priv *priv,
|
||||
attr->ct_attr.ct_action = act->ct.action;
|
||||
attr->ct_attr.nf_ft = act->ct.flow_table;
|
||||
|
||||
if (!clear_action)
|
||||
goto out;
|
||||
|
||||
err = mlx5_tc_ct_entry_set_registers(priv, mod_acts, 0, 0, 0, 0);
|
||||
if (err) {
|
||||
NL_SET_ERR_MSG_MOD(extack, "Failed to set registers for ct clear");
|
||||
return err;
|
||||
}
|
||||
attr->action |= MLX5_FLOW_CONTEXT_ACTION_MOD_HDR;
|
||||
|
||||
out:
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
@ -129,6 +129,10 @@ bool
|
||||
mlx5e_tc_ct_restore_flow(struct mlx5_tc_ct_priv *ct_priv,
|
||||
struct sk_buff *skb, u8 zone_restore_id);
|
||||
|
||||
int
|
||||
mlx5_tc_ct_set_ct_clear_regs(struct mlx5_tc_ct_priv *priv,
|
||||
struct mlx5e_tc_mod_hdr_acts *mod_acts);
|
||||
|
||||
#else /* CONFIG_MLX5_TC_CT */
|
||||
|
||||
static inline struct mlx5_tc_ct_priv *
|
||||
@ -170,6 +174,13 @@ mlx5_tc_ct_add_no_trk_match(struct mlx5_flow_spec *spec)
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int
|
||||
mlx5_tc_ct_set_ct_clear_regs(struct mlx5_tc_ct_priv *priv,
|
||||
struct mlx5e_tc_mod_hdr_acts *mod_acts)
|
||||
{
|
||||
return -EOPNOTSUPP;
|
||||
}
|
||||
|
||||
static inline int
|
||||
mlx5_tc_ct_parse_action(struct mlx5_tc_ct_priv *priv,
|
||||
struct mlx5_flow_attr *attr,
|
||||
|
Loading…
x
Reference in New Issue
Block a user