Commit 5e0202eb authored by Shay Drory's avatar Shay Drory Committed by Saeed Mahameed

net/mlx5: E-switch, Handle multiple master egress rules

Currently, whenever a shared FDB is created, the slave eswitch is
creating master egress rule to the master eswitch.
In order to support more than two ports, which means there will be
more than one slave eswitch, enlarge bounce_rule, which is used to
create master egress rule, to an xarray.
Signed-off-by: default avatarShay Drory <shayd@nvidia.com>
Reviewed-by: default avatarRoi Dayan <roid@nvidia.com>
Signed-off-by: default avatarSaeed Mahameed <saeedm@nvidia.com>
parent 9bee385a
...@@ -15,13 +15,15 @@ static void esw_acl_egress_ofld_fwd2vport_destroy(struct mlx5_vport *vport) ...@@ -15,13 +15,15 @@ static void esw_acl_egress_ofld_fwd2vport_destroy(struct mlx5_vport *vport)
vport->egress.offloads.fwd_rule = NULL; vport->egress.offloads.fwd_rule = NULL;
} }
static void esw_acl_egress_ofld_bounce_rule_destroy(struct mlx5_vport *vport) static void esw_acl_egress_ofld_bounce_rules_destroy(struct mlx5_vport *vport)
{ {
if (!vport->egress.offloads.bounce_rule) struct mlx5_flow_handle *bounce_rule;
return; unsigned long i;
mlx5_del_flow_rules(vport->egress.offloads.bounce_rule); xa_for_each(&vport->egress.offloads.bounce_rules, i, bounce_rule) {
vport->egress.offloads.bounce_rule = NULL; mlx5_del_flow_rules(bounce_rule);
xa_erase(&vport->egress.offloads.bounce_rules, i);
}
} }
static int esw_acl_egress_ofld_fwd2vport_create(struct mlx5_eswitch *esw, static int esw_acl_egress_ofld_fwd2vport_create(struct mlx5_eswitch *esw,
...@@ -96,7 +98,7 @@ static void esw_acl_egress_ofld_rules_destroy(struct mlx5_vport *vport) ...@@ -96,7 +98,7 @@ static void esw_acl_egress_ofld_rules_destroy(struct mlx5_vport *vport)
{ {
esw_acl_egress_vlan_destroy(vport); esw_acl_egress_vlan_destroy(vport);
esw_acl_egress_ofld_fwd2vport_destroy(vport); esw_acl_egress_ofld_fwd2vport_destroy(vport);
esw_acl_egress_ofld_bounce_rule_destroy(vport); esw_acl_egress_ofld_bounce_rules_destroy(vport);
} }
static int esw_acl_egress_ofld_groups_create(struct mlx5_eswitch *esw, static int esw_acl_egress_ofld_groups_create(struct mlx5_eswitch *esw,
...@@ -194,6 +196,7 @@ int esw_acl_egress_ofld_setup(struct mlx5_eswitch *esw, struct mlx5_vport *vport ...@@ -194,6 +196,7 @@ int esw_acl_egress_ofld_setup(struct mlx5_eswitch *esw, struct mlx5_vport *vport
vport->egress.acl = NULL; vport->egress.acl = NULL;
return err; return err;
} }
vport->egress.type = VPORT_EGRESS_ACL_TYPE_DEFAULT;
err = esw_acl_egress_ofld_groups_create(esw, vport); err = esw_acl_egress_ofld_groups_create(esw, vport);
if (err) if (err)
......
...@@ -123,8 +123,14 @@ struct vport_ingress { ...@@ -123,8 +123,14 @@ struct vport_ingress {
} offloads; } offloads;
}; };
enum vport_egress_acl_type {
VPORT_EGRESS_ACL_TYPE_DEFAULT,
VPORT_EGRESS_ACL_TYPE_SHARED_FDB,
};
struct vport_egress { struct vport_egress {
struct mlx5_flow_table *acl; struct mlx5_flow_table *acl;
enum vport_egress_acl_type type;
struct mlx5_flow_handle *allowed_vlan; struct mlx5_flow_handle *allowed_vlan;
struct mlx5_flow_group *vlan_grp; struct mlx5_flow_group *vlan_grp;
union { union {
...@@ -136,7 +142,7 @@ struct vport_egress { ...@@ -136,7 +142,7 @@ struct vport_egress {
struct { struct {
struct mlx5_flow_group *fwd_grp; struct mlx5_flow_group *fwd_grp;
struct mlx5_flow_handle *fwd_rule; struct mlx5_flow_handle *fwd_rule;
struct mlx5_flow_handle *bounce_rule; struct xarray bounce_rules;
struct mlx5_flow_group *bounce_grp; struct mlx5_flow_group *bounce_grp;
} offloads; } offloads;
}; };
......
...@@ -2512,6 +2512,7 @@ static int __esw_set_master_egress_rule(struct mlx5_core_dev *master, ...@@ -2512,6 +2512,7 @@ static int __esw_set_master_egress_rule(struct mlx5_core_dev *master,
struct mlx5_vport *vport, struct mlx5_vport *vport,
struct mlx5_flow_table *acl) struct mlx5_flow_table *acl)
{ {
u16 slave_index = MLX5_CAP_GEN(slave, vhca_id);
struct mlx5_flow_handle *flow_rule = NULL; struct mlx5_flow_handle *flow_rule = NULL;
struct mlx5_flow_destination dest = {}; struct mlx5_flow_destination dest = {};
struct mlx5_flow_act flow_act = {}; struct mlx5_flow_act flow_act = {};
...@@ -2527,8 +2528,7 @@ static int __esw_set_master_egress_rule(struct mlx5_core_dev *master, ...@@ -2527,8 +2528,7 @@ static int __esw_set_master_egress_rule(struct mlx5_core_dev *master,
misc = MLX5_ADDR_OF(fte_match_param, spec->match_value, misc = MLX5_ADDR_OF(fte_match_param, spec->match_value,
misc_parameters); misc_parameters);
MLX5_SET(fte_match_set_misc, misc, source_port, MLX5_VPORT_UPLINK); MLX5_SET(fte_match_set_misc, misc, source_port, MLX5_VPORT_UPLINK);
MLX5_SET(fte_match_set_misc, misc, source_eswitch_owner_vhca_id, MLX5_SET(fte_match_set_misc, misc, source_eswitch_owner_vhca_id, slave_index);
MLX5_CAP_GEN(slave, vhca_id));
misc = MLX5_ADDR_OF(fte_match_param, spec->match_criteria, misc_parameters); misc = MLX5_ADDR_OF(fte_match_param, spec->match_criteria, misc_parameters);
MLX5_SET_TO_ONES(fte_match_set_misc, misc, source_port); MLX5_SET_TO_ONES(fte_match_set_misc, misc, source_port);
...@@ -2543,44 +2543,35 @@ static int __esw_set_master_egress_rule(struct mlx5_core_dev *master, ...@@ -2543,44 +2543,35 @@ static int __esw_set_master_egress_rule(struct mlx5_core_dev *master,
flow_rule = mlx5_add_flow_rules(acl, spec, &flow_act, flow_rule = mlx5_add_flow_rules(acl, spec, &flow_act,
&dest, 1); &dest, 1);
if (IS_ERR(flow_rule)) if (IS_ERR(flow_rule)) {
err = PTR_ERR(flow_rule); err = PTR_ERR(flow_rule);
else } else {
vport->egress.offloads.bounce_rule = flow_rule; err = xa_insert(&vport->egress.offloads.bounce_rules,
slave_index, flow_rule, GFP_KERNEL);
if (err)
mlx5_del_flow_rules(flow_rule);
}
kvfree(spec); kvfree(spec);
return err; return err;
} }
static int esw_set_master_egress_rule(struct mlx5_core_dev *master, static int esw_master_egress_create_resources(struct mlx5_flow_namespace *egress_ns,
struct mlx5_core_dev *slave) struct mlx5_vport *vport)
{ {
int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in); int inlen = MLX5_ST_SZ_BYTES(create_flow_group_in);
struct mlx5_eswitch *esw = master->priv.eswitch;
struct mlx5_flow_table_attr ft_attr = { struct mlx5_flow_table_attr ft_attr = {
.max_fte = 1, .prio = 0, .level = 0, .max_fte = MLX5_MAX_PORTS, .prio = 0, .level = 0,
.flags = MLX5_FLOW_TABLE_OTHER_VPORT, .flags = MLX5_FLOW_TABLE_OTHER_VPORT,
}; };
struct mlx5_flow_namespace *egress_ns;
struct mlx5_flow_table *acl; struct mlx5_flow_table *acl;
struct mlx5_flow_group *g; struct mlx5_flow_group *g;
struct mlx5_vport *vport;
void *match_criteria; void *match_criteria;
u32 *flow_group_in; u32 *flow_group_in;
int err; int err;
vport = mlx5_eswitch_get_vport(esw, esw->manager_vport);
if (IS_ERR(vport))
return PTR_ERR(vport);
egress_ns = mlx5_get_flow_vport_acl_namespace(master,
MLX5_FLOW_NAMESPACE_ESW_EGRESS,
vport->index);
if (!egress_ns)
return -EINVAL;
if (vport->egress.acl) if (vport->egress.acl)
return -EINVAL; return 0;
flow_group_in = kvzalloc(inlen, GFP_KERNEL); flow_group_in = kvzalloc(inlen, GFP_KERNEL);
if (!flow_group_in) if (!flow_group_in)
...@@ -2604,7 +2595,7 @@ static int esw_set_master_egress_rule(struct mlx5_core_dev *master, ...@@ -2604,7 +2595,7 @@ static int esw_set_master_egress_rule(struct mlx5_core_dev *master,
MLX5_SET(create_flow_group_in, flow_group_in, MLX5_SET(create_flow_group_in, flow_group_in,
source_eswitch_owner_vhca_id_valid, 1); source_eswitch_owner_vhca_id_valid, 1);
MLX5_SET(create_flow_group_in, flow_group_in, start_flow_index, 0); MLX5_SET(create_flow_group_in, flow_group_in, start_flow_index, 0);
MLX5_SET(create_flow_group_in, flow_group_in, end_flow_index, 0); MLX5_SET(create_flow_group_in, flow_group_in, end_flow_index, MLX5_MAX_PORTS);
g = mlx5_create_flow_group(acl, flow_group_in); g = mlx5_create_flow_group(acl, flow_group_in);
if (IS_ERR(g)) { if (IS_ERR(g)) {
...@@ -2612,19 +2603,15 @@ static int esw_set_master_egress_rule(struct mlx5_core_dev *master, ...@@ -2612,19 +2603,15 @@ static int esw_set_master_egress_rule(struct mlx5_core_dev *master,
goto err_group; goto err_group;
} }
err = __esw_set_master_egress_rule(master, slave, vport, acl);
if (err)
goto err_rule;
vport->egress.acl = acl; vport->egress.acl = acl;
vport->egress.offloads.bounce_grp = g; vport->egress.offloads.bounce_grp = g;
vport->egress.type = VPORT_EGRESS_ACL_TYPE_SHARED_FDB;
xa_init_flags(&vport->egress.offloads.bounce_rules, XA_FLAGS_ALLOC);
kvfree(flow_group_in); kvfree(flow_group_in);
return 0; return 0;
err_rule:
mlx5_destroy_flow_group(g);
err_group: err_group:
mlx5_destroy_flow_table(acl); mlx5_destroy_flow_table(acl);
out: out:
...@@ -2632,6 +2619,52 @@ static int esw_set_master_egress_rule(struct mlx5_core_dev *master, ...@@ -2632,6 +2619,52 @@ static int esw_set_master_egress_rule(struct mlx5_core_dev *master,
return err; return err;
} }
static void esw_master_egress_destroy_resources(struct mlx5_vport *vport)
{
mlx5_destroy_flow_group(vport->egress.offloads.bounce_grp);
mlx5_destroy_flow_table(vport->egress.acl);
}
static int esw_set_master_egress_rule(struct mlx5_core_dev *master,
struct mlx5_core_dev *slave)
{
struct mlx5_eswitch *esw = master->priv.eswitch;
u16 slave_index = MLX5_CAP_GEN(slave, vhca_id);
struct mlx5_flow_namespace *egress_ns;
struct mlx5_vport *vport;
int err;
vport = mlx5_eswitch_get_vport(esw, esw->manager_vport);
if (IS_ERR(vport))
return PTR_ERR(vport);
egress_ns = mlx5_get_flow_vport_acl_namespace(master,
MLX5_FLOW_NAMESPACE_ESW_EGRESS,
vport->index);
if (!egress_ns)
return -EINVAL;
if (vport->egress.acl && vport->egress.type != VPORT_EGRESS_ACL_TYPE_SHARED_FDB)
return 0;
err = esw_master_egress_create_resources(egress_ns, vport);
if (err)
return err;
if (xa_load(&vport->egress.offloads.bounce_rules, slave_index))
return -EINVAL;
err = __esw_set_master_egress_rule(master, slave, vport, vport->egress.acl);
if (err)
goto err_rule;
return 0;
err_rule:
esw_master_egress_destroy_resources(vport);
return err;
}
static void esw_unset_master_egress_rule(struct mlx5_core_dev *dev) static void esw_unset_master_egress_rule(struct mlx5_core_dev *dev)
{ {
struct mlx5_vport *vport; struct mlx5_vport *vport;
...@@ -2640,6 +2673,7 @@ static void esw_unset_master_egress_rule(struct mlx5_core_dev *dev) ...@@ -2640,6 +2673,7 @@ static void esw_unset_master_egress_rule(struct mlx5_core_dev *dev)
dev->priv.eswitch->manager_vport); dev->priv.eswitch->manager_vport);
esw_acl_egress_ofld_cleanup(vport); esw_acl_egress_ofld_cleanup(vport);
xa_destroy(&vport->egress.offloads.bounce_rules);
} }
int mlx5_eswitch_offloads_config_single_fdb(struct mlx5_eswitch *master_esw, int mlx5_eswitch_offloads_config_single_fdb(struct mlx5_eswitch *master_esw,
......
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