net/mlx5: update modify field action

Update immediate value/pointer source operand support
for modify field RTE Flow action:

  - source operand data can be presented by byte buffer
    (instead of former uint64_t) or by pointer
  - no host byte ordering is assumed anymore for immediate
    data buffer (not uint64_t anymore)
  - no immediate value offset is expected (the source
    subfield is located at the same offset as in destination)

Signed-off-by: Viacheslav Ovsiienko <viacheslavo@nvidia.com>
This commit is contained in:
Viacheslav Ovsiienko 2021-10-13 21:45:16 +03:00 committed by Ferruh Yigit
parent a2b97fb7f5
commit 40c8fb1fd3

View File

@ -1391,7 +1391,7 @@ flow_dv_convert_action_modify_ipv6_dscp
static int static int
mlx5_flow_item_field_width(struct mlx5_priv *priv, mlx5_flow_item_field_width(struct mlx5_priv *priv,
enum rte_flow_field_id field) enum rte_flow_field_id field, int inherit)
{ {
switch (field) { switch (field) {
case RTE_FLOW_FIELD_START: case RTE_FLOW_FIELD_START:
@ -1442,7 +1442,7 @@ mlx5_flow_item_field_width(struct mlx5_priv *priv,
return __builtin_popcount(priv->sh->dv_meta_mask); return __builtin_popcount(priv->sh->dv_meta_mask);
case RTE_FLOW_FIELD_POINTER: case RTE_FLOW_FIELD_POINTER:
case RTE_FLOW_FIELD_VALUE: case RTE_FLOW_FIELD_VALUE:
return 64; return inherit < 0 ? 0 : inherit;
default: default:
MLX5_ASSERT(false); MLX5_ASSERT(false);
} }
@ -1452,17 +1452,14 @@ mlx5_flow_item_field_width(struct mlx5_priv *priv,
static void static void
mlx5_flow_field_id_to_modify_info mlx5_flow_field_id_to_modify_info
(const struct rte_flow_action_modify_data *data, (const struct rte_flow_action_modify_data *data,
struct field_modify_info *info, struct field_modify_info *info, uint32_t *mask,
uint32_t *mask, uint32_t *value, uint32_t width, uint32_t *shift, struct rte_eth_dev *dev,
uint32_t width, uint32_t dst_width, const struct rte_flow_attr *attr, struct rte_flow_error *error)
uint32_t *shift, struct rte_eth_dev *dev,
const struct rte_flow_attr *attr,
struct rte_flow_error *error)
{ {
struct mlx5_priv *priv = dev->data->dev_private; struct mlx5_priv *priv = dev->data->dev_private;
uint32_t idx = 0; uint32_t idx = 0;
uint32_t off = 0; uint32_t off = 0;
uint64_t val = 0;
switch (data->field) { switch (data->field) {
case RTE_FLOW_FIELD_START: case RTE_FLOW_FIELD_START:
/* not supported yet */ /* not supported yet */
@ -1472,7 +1469,7 @@ mlx5_flow_field_id_to_modify_info
off = data->offset > 16 ? data->offset - 16 : 0; off = data->offset > 16 ? data->offset - 16 : 0;
if (mask) { if (mask) {
if (data->offset < 16) { if (data->offset < 16) {
info[idx] = (struct field_modify_info){2, 0, info[idx] = (struct field_modify_info){2, 4,
MLX5_MODI_OUT_DMAC_15_0}; MLX5_MODI_OUT_DMAC_15_0};
if (width < 16) { if (width < 16) {
mask[idx] = rte_cpu_to_be_16(0xffff >> mask[idx] = rte_cpu_to_be_16(0xffff >>
@ -1486,15 +1483,15 @@ mlx5_flow_field_id_to_modify_info
break; break;
++idx; ++idx;
} }
info[idx] = (struct field_modify_info){4, 4 * idx, info[idx] = (struct field_modify_info){4, 0,
MLX5_MODI_OUT_DMAC_47_16}; MLX5_MODI_OUT_DMAC_47_16};
mask[idx] = rte_cpu_to_be_32((0xffffffff >> mask[idx] = rte_cpu_to_be_32((0xffffffff >>
(32 - width)) << off); (32 - width)) << off);
} else { } else {
if (data->offset < 16) if (data->offset < 16)
info[idx++] = (struct field_modify_info){2, 0, info[idx++] = (struct field_modify_info){2, 4,
MLX5_MODI_OUT_DMAC_15_0}; MLX5_MODI_OUT_DMAC_15_0};
info[idx] = (struct field_modify_info){4, off, info[idx] = (struct field_modify_info){4, 0,
MLX5_MODI_OUT_DMAC_47_16}; MLX5_MODI_OUT_DMAC_47_16};
} }
break; break;
@ -1502,7 +1499,7 @@ mlx5_flow_field_id_to_modify_info
off = data->offset > 16 ? data->offset - 16 : 0; off = data->offset > 16 ? data->offset - 16 : 0;
if (mask) { if (mask) {
if (data->offset < 16) { if (data->offset < 16) {
info[idx] = (struct field_modify_info){2, 0, info[idx] = (struct field_modify_info){2, 4,
MLX5_MODI_OUT_SMAC_15_0}; MLX5_MODI_OUT_SMAC_15_0};
if (width < 16) { if (width < 16) {
mask[idx] = rte_cpu_to_be_16(0xffff >> mask[idx] = rte_cpu_to_be_16(0xffff >>
@ -1516,15 +1513,15 @@ mlx5_flow_field_id_to_modify_info
break; break;
++idx; ++idx;
} }
info[idx] = (struct field_modify_info){4, 4 * idx, info[idx] = (struct field_modify_info){4, 0,
MLX5_MODI_OUT_SMAC_47_16}; MLX5_MODI_OUT_SMAC_47_16};
mask[idx] = rte_cpu_to_be_32((0xffffffff >> mask[idx] = rte_cpu_to_be_32((0xffffffff >>
(32 - width)) << off); (32 - width)) << off);
} else { } else {
if (data->offset < 16) if (data->offset < 16)
info[idx++] = (struct field_modify_info){2, 0, info[idx++] = (struct field_modify_info){2, 4,
MLX5_MODI_OUT_SMAC_15_0}; MLX5_MODI_OUT_SMAC_15_0};
info[idx] = (struct field_modify_info){4, off, info[idx] = (struct field_modify_info){4, 0,
MLX5_MODI_OUT_SMAC_47_16}; MLX5_MODI_OUT_SMAC_47_16};
} }
break; break;
@ -1584,8 +1581,7 @@ mlx5_flow_field_id_to_modify_info
case RTE_FLOW_FIELD_IPV6_SRC: case RTE_FLOW_FIELD_IPV6_SRC:
if (mask) { if (mask) {
if (data->offset < 32) { if (data->offset < 32) {
info[idx] = (struct field_modify_info){4, info[idx] = (struct field_modify_info){4, 12,
4 * idx,
MLX5_MODI_OUT_SIPV6_31_0}; MLX5_MODI_OUT_SIPV6_31_0};
if (width < 32) { if (width < 32) {
mask[idx] = mask[idx] =
@ -1601,8 +1597,7 @@ mlx5_flow_field_id_to_modify_info
++idx; ++idx;
} }
if (data->offset < 64) { if (data->offset < 64) {
info[idx] = (struct field_modify_info){4, info[idx] = (struct field_modify_info){4, 8,
4 * idx,
MLX5_MODI_OUT_SIPV6_63_32}; MLX5_MODI_OUT_SIPV6_63_32};
if (width < 32) { if (width < 32) {
mask[idx] = mask[idx] =
@ -1618,8 +1613,7 @@ mlx5_flow_field_id_to_modify_info
++idx; ++idx;
} }
if (data->offset < 96) { if (data->offset < 96) {
info[idx] = (struct field_modify_info){4, info[idx] = (struct field_modify_info){4, 4,
4 * idx,
MLX5_MODI_OUT_SIPV6_95_64}; MLX5_MODI_OUT_SIPV6_95_64};
if (width < 32) { if (width < 32) {
mask[idx] = mask[idx] =
@ -1634,19 +1628,19 @@ mlx5_flow_field_id_to_modify_info
break; break;
++idx; ++idx;
} }
info[idx] = (struct field_modify_info){4, 4 * idx, info[idx] = (struct field_modify_info){4, 0,
MLX5_MODI_OUT_SIPV6_127_96}; MLX5_MODI_OUT_SIPV6_127_96};
mask[idx] = rte_cpu_to_be_32(0xffffffff >> mask[idx] = rte_cpu_to_be_32(0xffffffff >>
(32 - width)); (32 - width));
} else { } else {
if (data->offset < 32) if (data->offset < 32)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 12,
MLX5_MODI_OUT_SIPV6_31_0}; MLX5_MODI_OUT_SIPV6_31_0};
if (data->offset < 64) if (data->offset < 64)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 8,
MLX5_MODI_OUT_SIPV6_63_32}; MLX5_MODI_OUT_SIPV6_63_32};
if (data->offset < 96) if (data->offset < 96)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 4,
MLX5_MODI_OUT_SIPV6_95_64}; MLX5_MODI_OUT_SIPV6_95_64};
if (data->offset < 128) if (data->offset < 128)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 0,
@ -1656,8 +1650,7 @@ mlx5_flow_field_id_to_modify_info
case RTE_FLOW_FIELD_IPV6_DST: case RTE_FLOW_FIELD_IPV6_DST:
if (mask) { if (mask) {
if (data->offset < 32) { if (data->offset < 32) {
info[idx] = (struct field_modify_info){4, info[idx] = (struct field_modify_info){4, 12,
4 * idx,
MLX5_MODI_OUT_DIPV6_31_0}; MLX5_MODI_OUT_DIPV6_31_0};
if (width < 32) { if (width < 32) {
mask[idx] = mask[idx] =
@ -1673,8 +1666,7 @@ mlx5_flow_field_id_to_modify_info
++idx; ++idx;
} }
if (data->offset < 64) { if (data->offset < 64) {
info[idx] = (struct field_modify_info){4, info[idx] = (struct field_modify_info){4, 8,
4 * idx,
MLX5_MODI_OUT_DIPV6_63_32}; MLX5_MODI_OUT_DIPV6_63_32};
if (width < 32) { if (width < 32) {
mask[idx] = mask[idx] =
@ -1690,8 +1682,7 @@ mlx5_flow_field_id_to_modify_info
++idx; ++idx;
} }
if (data->offset < 96) { if (data->offset < 96) {
info[idx] = (struct field_modify_info){4, info[idx] = (struct field_modify_info){4, 4,
4 * idx,
MLX5_MODI_OUT_DIPV6_95_64}; MLX5_MODI_OUT_DIPV6_95_64};
if (width < 32) { if (width < 32) {
mask[idx] = mask[idx] =
@ -1706,19 +1697,19 @@ mlx5_flow_field_id_to_modify_info
break; break;
++idx; ++idx;
} }
info[idx] = (struct field_modify_info){4, 4 * idx, info[idx] = (struct field_modify_info){4, 0,
MLX5_MODI_OUT_DIPV6_127_96}; MLX5_MODI_OUT_DIPV6_127_96};
mask[idx] = rte_cpu_to_be_32(0xffffffff >> mask[idx] = rte_cpu_to_be_32(0xffffffff >>
(32 - width)); (32 - width));
} else { } else {
if (data->offset < 32) if (data->offset < 32)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 12,
MLX5_MODI_OUT_DIPV6_31_0}; MLX5_MODI_OUT_DIPV6_31_0};
if (data->offset < 64) if (data->offset < 64)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 8,
MLX5_MODI_OUT_DIPV6_63_32}; MLX5_MODI_OUT_DIPV6_63_32};
if (data->offset < 96) if (data->offset < 96)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 4,
MLX5_MODI_OUT_DIPV6_95_64}; MLX5_MODI_OUT_DIPV6_95_64};
if (data->offset < 128) if (data->offset < 128)
info[idx++] = (struct field_modify_info){4, 0, info[idx++] = (struct field_modify_info){4, 0,
@ -1838,35 +1829,6 @@ mlx5_flow_field_id_to_modify_info
break; break;
case RTE_FLOW_FIELD_POINTER: case RTE_FLOW_FIELD_POINTER:
case RTE_FLOW_FIELD_VALUE: case RTE_FLOW_FIELD_VALUE:
if (data->field == RTE_FLOW_FIELD_POINTER)
memcpy(&val, (void *)(uintptr_t)data->value,
sizeof(uint64_t));
else
val = data->value;
for (idx = 0; idx < MLX5_ACT_MAX_MOD_FIELDS; idx++) {
if (mask[idx]) {
if (dst_width == 48) {
/*special case for MAC addresses */
value[idx] = rte_cpu_to_be_16(val);
val >>= 16;
dst_width -= 16;
} else if (dst_width > 16) {
value[idx] = rte_cpu_to_be_32(val);
val >>= 32;
} else if (dst_width > 8) {
value[idx] = rte_cpu_to_be_16(val);
val >>= 16;
} else {
value[idx] = (uint8_t)val;
val >>= 8;
}
if (*shift)
value[idx] <<= *shift;
if (!val)
break;
}
}
break;
default: default:
MLX5_ASSERT(false); MLX5_ASSERT(false);
break; break;
@ -1898,42 +1860,40 @@ flow_dv_convert_action_modify_field
const struct rte_flow_attr *attr, const struct rte_flow_attr *attr,
struct rte_flow_error *error) struct rte_flow_error *error)
{ {
struct mlx5_priv *priv = dev->data->dev_private;
const struct rte_flow_action_modify_field *conf = const struct rte_flow_action_modify_field *conf =
(const struct rte_flow_action_modify_field *)(action->conf); (const struct rte_flow_action_modify_field *)(action->conf);
struct rte_flow_item item; struct rte_flow_item item = {
.spec = NULL,
.mask = NULL
};
struct field_modify_info field[MLX5_ACT_MAX_MOD_FIELDS] = { struct field_modify_info field[MLX5_ACT_MAX_MOD_FIELDS] = {
{0, 0, 0} }; {0, 0, 0} };
struct field_modify_info dcopy[MLX5_ACT_MAX_MOD_FIELDS] = { struct field_modify_info dcopy[MLX5_ACT_MAX_MOD_FIELDS] = {
{0, 0, 0} }; {0, 0, 0} };
uint32_t mask[MLX5_ACT_MAX_MOD_FIELDS] = {0, 0, 0, 0, 0}; uint32_t mask[MLX5_ACT_MAX_MOD_FIELDS] = {0, 0, 0, 0, 0};
uint32_t value[MLX5_ACT_MAX_MOD_FIELDS] = {0, 0, 0, 0, 0};
uint32_t type; uint32_t type;
uint32_t shift = 0; uint32_t shift = 0;
uint32_t dst_width = mlx5_flow_item_field_width(priv, conf->dst.field);
if (conf->src.field == RTE_FLOW_FIELD_POINTER || if (conf->src.field == RTE_FLOW_FIELD_POINTER ||
conf->src.field == RTE_FLOW_FIELD_VALUE) { conf->src.field == RTE_FLOW_FIELD_VALUE) {
type = MLX5_MODIFICATION_TYPE_SET; type = MLX5_MODIFICATION_TYPE_SET;
/** For SET fill the destination field (field) first. */ /** For SET fill the destination field (field) first. */
mlx5_flow_field_id_to_modify_info(&conf->dst, field, mask, mlx5_flow_field_id_to_modify_info(&conf->dst, field, mask,
value, conf->width, dst_width, conf->width, &shift, dev,
&shift, dev, attr, error); attr, error);
/** Then copy immediate value from source as per mask. */ item.spec = conf->src.field == RTE_FLOW_FIELD_POINTER ?
mlx5_flow_field_id_to_modify_info(&conf->src, dcopy, mask, (void *)(uintptr_t)conf->src.pvalue :
value, conf->width, dst_width, (void *)(uintptr_t)&conf->src.value;
&shift, dev, attr, error);
item.spec = &value;
} else { } else {
type = MLX5_MODIFICATION_TYPE_COPY; type = MLX5_MODIFICATION_TYPE_COPY;
/** For COPY fill the destination field (dcopy) without mask. */ /** For COPY fill the destination field (dcopy) without mask. */
mlx5_flow_field_id_to_modify_info(&conf->dst, dcopy, NULL, mlx5_flow_field_id_to_modify_info(&conf->dst, dcopy, NULL,
value, conf->width, dst_width, conf->width, &shift, dev,
&shift, dev, attr, error); attr, error);
/** Then construct the source field (field) with mask. */ /** Then construct the source field (field) with mask. */
mlx5_flow_field_id_to_modify_info(&conf->src, field, mask, mlx5_flow_field_id_to_modify_info(&conf->src, field, mask,
value, conf->width, dst_width, conf->width, &shift,
&shift, dev, attr, error); dev, attr, error);
} }
item.mask = &mask; item.mask = &mask;
return flow_dv_convert_modify_action(&item, return flow_dv_convert_modify_action(&item,
@ -4854,9 +4814,9 @@ flow_dv_validate_action_modify_field(struct rte_eth_dev *dev,
const struct rte_flow_action_modify_field *action_modify_field = const struct rte_flow_action_modify_field *action_modify_field =
action->conf; action->conf;
uint32_t dst_width = mlx5_flow_item_field_width(priv, uint32_t dst_width = mlx5_flow_item_field_width(priv,
action_modify_field->dst.field); action_modify_field->dst.field, -1);
uint32_t src_width = mlx5_flow_item_field_width(priv, uint32_t src_width = mlx5_flow_item_field_width(priv,
action_modify_field->src.field); action_modify_field->src.field, dst_width);
ret = flow_dv_validate_action_modify_hdr(action_flags, action, error); ret = flow_dv_validate_action_modify_hdr(action_flags, action, error);
if (ret) if (ret)