f-stack/dpdk/drivers/net/mlx5/mlx5_flow_flex.c

1367 lines
40 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* SPDX-License-Identifier: BSD-3-Clause
* Copyright (c) 2021 NVIDIA Corporation & Affiliates
*/
#include <rte_malloc.h>
#include <mlx5_devx_cmds.h>
#include <mlx5_malloc.h>
#include "mlx5.h"
#include "mlx5_flow.h"
static_assert(sizeof(uint32_t) * CHAR_BIT >= MLX5_PORT_FLEX_ITEM_NUM,
"Flex item maximal number exceeds uint32_t bit width");
/**
* Routine called once on port initialization to init flex item
* related infrastructure initialization
*
* @param dev
* Ethernet device to perform flex item initialization
*
* @return
* 0 on success, a negative errno value otherwise and rte_errno is set.
*/
int
mlx5_flex_item_port_init(struct rte_eth_dev *dev)
{
struct mlx5_priv *priv = dev->data->dev_private;
rte_spinlock_init(&priv->flex_item_sl);
MLX5_ASSERT(!priv->flex_item_map);
return 0;
}
/**
* Routine called once on port close to perform flex item
* related infrastructure cleanup.
*
* @param dev
* Ethernet device to perform cleanup
*/
void
mlx5_flex_item_port_cleanup(struct rte_eth_dev *dev)
{
struct mlx5_priv *priv = dev->data->dev_private;
uint32_t i;
for (i = 0; i < MLX5_PORT_FLEX_ITEM_NUM && priv->flex_item_map ; i++) {
if (priv->flex_item_map & (1 << i)) {
struct mlx5_flex_item *flex = &priv->flex_item[i];
claim_zero(mlx5_list_unregister
(priv->sh->flex_parsers_dv,
&flex->devx_fp->entry));
flex->devx_fp = NULL;
flex->refcnt = 0;
priv->flex_item_map &= ~(1 << i);
}
}
}
static int
mlx5_flex_index(struct mlx5_priv *priv, struct mlx5_flex_item *item)
{
uintptr_t start = (uintptr_t)&priv->flex_item[0];
uintptr_t entry = (uintptr_t)item;
uintptr_t idx = (entry - start) / sizeof(struct mlx5_flex_item);
if (entry < start ||
idx >= MLX5_PORT_FLEX_ITEM_NUM ||
(entry - start) % sizeof(struct mlx5_flex_item) ||
!(priv->flex_item_map & (1u << idx)))
return -1;
return (int)idx;
}
static struct mlx5_flex_item *
mlx5_flex_alloc(struct mlx5_priv *priv)
{
struct mlx5_flex_item *item = NULL;
rte_spinlock_lock(&priv->flex_item_sl);
if (~priv->flex_item_map) {
uint32_t idx = rte_bsf32(~priv->flex_item_map);
if (idx < MLX5_PORT_FLEX_ITEM_NUM) {
item = &priv->flex_item[idx];
MLX5_ASSERT(!item->refcnt);
MLX5_ASSERT(!item->devx_fp);
item->devx_fp = NULL;
__atomic_store_n(&item->refcnt, 0, __ATOMIC_RELEASE);
priv->flex_item_map |= 1u << idx;
}
}
rte_spinlock_unlock(&priv->flex_item_sl);
return item;
}
static void
mlx5_flex_free(struct mlx5_priv *priv, struct mlx5_flex_item *item)
{
int idx = mlx5_flex_index(priv, item);
MLX5_ASSERT(idx >= 0 &&
idx < MLX5_PORT_FLEX_ITEM_NUM &&
(priv->flex_item_map & (1u << idx)));
if (idx >= 0) {
rte_spinlock_lock(&priv->flex_item_sl);
MLX5_ASSERT(!item->refcnt);
MLX5_ASSERT(!item->devx_fp);
item->devx_fp = NULL;
__atomic_store_n(&item->refcnt, 0, __ATOMIC_RELEASE);
priv->flex_item_map &= ~(1u << idx);
rte_spinlock_unlock(&priv->flex_item_sl);
}
}
static uint32_t
mlx5_flex_get_bitfield(const struct rte_flow_item_flex *item,
uint32_t pos, uint32_t width, uint32_t shift)
{
const uint8_t *ptr = item->pattern + pos / CHAR_BIT;
uint32_t val, vbits;
/* Proceed the bitfield start byte. */
MLX5_ASSERT(width <= sizeof(uint32_t) * CHAR_BIT && width);
MLX5_ASSERT(width + shift <= sizeof(uint32_t) * CHAR_BIT);
if (item->length <= pos / CHAR_BIT)
return 0;
val = *ptr++ >> (pos % CHAR_BIT);
vbits = CHAR_BIT - pos % CHAR_BIT;
pos = (pos + vbits) / CHAR_BIT;
vbits = RTE_MIN(vbits, width);
val &= RTE_BIT32(vbits) - 1;
while (vbits < width && pos < item->length) {
uint32_t part = RTE_MIN(width - vbits, (uint32_t)CHAR_BIT);
uint32_t tmp = *ptr++;
pos++;
tmp &= RTE_BIT32(part) - 1;
val |= tmp << vbits;
vbits += part;
}
return rte_bswap32(val <<= shift);
}
#define SET_FP_MATCH_SAMPLE_ID(x, def, msk, val, sid) \
do { \
uint32_t tmp, out = (def); \
tmp = MLX5_GET(fte_match_set_misc4, misc4_v, \
prog_sample_field_value_##x); \
tmp = (tmp & ~out) | (val); \
MLX5_SET(fte_match_set_misc4, misc4_v, \
prog_sample_field_value_##x, tmp); \
tmp = MLX5_GET(fte_match_set_misc4, misc4_m, \
prog_sample_field_value_##x); \
tmp = (tmp & ~out) | (msk); \
MLX5_SET(fte_match_set_misc4, misc4_m, \
prog_sample_field_value_##x, tmp); \
tmp = tmp ? (sid) : 0; \
MLX5_SET(fte_match_set_misc4, misc4_v, \
prog_sample_field_id_##x, tmp);\
MLX5_SET(fte_match_set_misc4, misc4_m, \
prog_sample_field_id_##x, tmp); \
} while (0)
__rte_always_inline static void
mlx5_flex_set_match_sample(void *misc4_m, void *misc4_v,
uint32_t def, uint32_t mask, uint32_t value,
uint32_t sample_id, uint32_t id)
{
switch (id) {
case 0:
SET_FP_MATCH_SAMPLE_ID(0, def, mask, value, sample_id);
break;
case 1:
SET_FP_MATCH_SAMPLE_ID(1, def, mask, value, sample_id);
break;
case 2:
SET_FP_MATCH_SAMPLE_ID(2, def, mask, value, sample_id);
break;
case 3:
SET_FP_MATCH_SAMPLE_ID(3, def, mask, value, sample_id);
break;
case 4:
SET_FP_MATCH_SAMPLE_ID(4, def, mask, value, sample_id);
break;
case 5:
SET_FP_MATCH_SAMPLE_ID(5, def, mask, value, sample_id);
break;
case 6:
SET_FP_MATCH_SAMPLE_ID(6, def, mask, value, sample_id);
break;
case 7:
SET_FP_MATCH_SAMPLE_ID(7, def, mask, value, sample_id);
break;
default:
MLX5_ASSERT(false);
break;
}
#undef SET_FP_MATCH_SAMPLE_ID
}
/**
* Translate item pattern into matcher fields according to translation
* array.
*
* @param dev
* Ethernet device to translate flex item on.
* @param[in, out] matcher
* Flow matcher to configure
* @param[in, out] key
* Flow matcher value.
* @param[in] item
* Flow pattern to translate.
* @param[in] is_inner
* Inner Flex Item (follows after tunnel header).
*
* @return
* 0 on success, a negative errno value otherwise and rte_errno is set.
*/
void
mlx5_flex_flow_translate_item(struct rte_eth_dev *dev,
void *matcher, void *key,
const struct rte_flow_item *item,
bool is_inner)
{
const struct rte_flow_item_flex *spec, *mask;
void *misc4_m = MLX5_ADDR_OF(fte_match_param, matcher,
misc_parameters_4);
void *misc4_v = MLX5_ADDR_OF(fte_match_param, key, misc_parameters_4);
struct mlx5_flex_item *tp;
uint32_t i, pos = 0;
RTE_SET_USED(dev);
MLX5_ASSERT(item->spec && item->mask);
spec = item->spec;
mask = item->mask;
tp = (struct mlx5_flex_item *)spec->handle;
MLX5_ASSERT(mlx5_flex_index(dev->data->dev_private, tp) >= 0);
for (i = 0; i < tp->mapnum; i++) {
struct mlx5_flex_pattern_field *map = tp->map + i;
uint32_t id = map->reg_id;
uint32_t def = (RTE_BIT64(map->width) - 1) << map->shift;
uint32_t val, msk;
/* Skip placeholders for DUMMY fields. */
if (id == MLX5_INVALID_SAMPLE_REG_ID) {
pos += map->width;
continue;
}
val = mlx5_flex_get_bitfield(spec, pos, map->width, map->shift);
msk = mlx5_flex_get_bitfield(mask, pos, map->width, map->shift);
MLX5_ASSERT(map->width);
MLX5_ASSERT(id < tp->devx_fp->num_samples);
if (tp->tunnel_mode == FLEX_TUNNEL_MODE_MULTI && is_inner) {
uint32_t num_samples = tp->devx_fp->num_samples / 2;
MLX5_ASSERT(tp->devx_fp->num_samples % 2 == 0);
MLX5_ASSERT(id < num_samples);
id += num_samples;
}
mlx5_flex_set_match_sample(misc4_m, misc4_v,
def, msk & def, val & msk & def,
tp->devx_fp->sample_ids[id], id);
pos += map->width;
}
}
/**
* Convert flex item handle (from the RTE flow) to flex item index on port.
* Optionally can increment flex item object reference count.
*
* @param dev
* Ethernet device to acquire flex item on.
* @param[in] handle
* Flow item handle from item spec.
* @param[in] acquire
* If set - increment reference counter.
*
* @return
* >=0 - index on success, a negative errno value otherwise
* and rte_errno is set.
*/
int
mlx5_flex_acquire_index(struct rte_eth_dev *dev,
struct rte_flow_item_flex_handle *handle,
bool acquire)
{
struct mlx5_priv *priv = dev->data->dev_private;
struct mlx5_flex_item *flex = (struct mlx5_flex_item *)handle;
int ret = mlx5_flex_index(priv, flex);
if (ret < 0) {
errno = -EINVAL;
rte_errno = EINVAL;
return ret;
}
if (acquire)
__atomic_add_fetch(&flex->refcnt, 1, __ATOMIC_RELEASE);
return ret;
}
/**
* Release flex item index on port - decrements reference counter by index.
*
* @param dev
* Ethernet device to acquire flex item on.
* @param[in] index
* Flow item index.
*
* @return
* 0 - on success, a negative errno value otherwise and rte_errno is set.
*/
int
mlx5_flex_release_index(struct rte_eth_dev *dev,
int index)
{
struct mlx5_priv *priv = dev->data->dev_private;
struct mlx5_flex_item *flex;
if (index >= MLX5_PORT_FLEX_ITEM_NUM ||
!(priv->flex_item_map & (1u << index))) {
errno = EINVAL;
rte_errno = -EINVAL;
return -EINVAL;
}
flex = priv->flex_item + index;
if (flex->refcnt <= 1) {
MLX5_ASSERT(false);
errno = EINVAL;
rte_errno = -EINVAL;
return -EINVAL;
}
__atomic_sub_fetch(&flex->refcnt, 1, __ATOMIC_RELEASE);
return 0;
}
/*
* Calculate largest mask value for a given shift.
*
* shift mask
* ------- ---------------
* 0 b111100 0x3C
* 1 b111110 0x3E
* 2 b111111 0x3F
* 3 b011111 0x1F
* 4 b001111 0x0F
* 5 b000111 0x07
*/
static uint8_t
mlx5_flex_hdr_len_mask(uint8_t shift,
const struct mlx5_hca_flex_attr *attr)
{
uint32_t base_mask;
int diff = shift - MLX5_PARSE_GRAPH_NODE_HDR_LEN_SHIFT_DWORD;
base_mask = mlx5_hca_parse_graph_node_base_hdr_len_mask(attr);
return diff == 0 ? base_mask :
diff < 0 ? (base_mask << -diff) & base_mask : base_mask >> diff;
}
static int
mlx5_flex_translate_length(struct mlx5_hca_flex_attr *attr,
const struct rte_flow_item_flex_conf *conf,
struct mlx5_flex_parser_devx *devx,
struct rte_flow_error *error)
{
const struct rte_flow_item_flex_field *field = &conf->next_header;
struct mlx5_devx_graph_node_attr *node = &devx->devx_conf;
uint32_t len_width, mask;
if (field->field_base % CHAR_BIT)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"not byte aligned header length field");
switch (field->field_mode) {
case FIELD_MODE_DUMMY:
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid header length field mode (DUMMY)");
case FIELD_MODE_FIXED:
if (!(attr->header_length_mode &
RTE_BIT32(MLX5_GRAPH_NODE_LEN_FIXED)))
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported header length field mode (FIXED)");
if (field->field_size ||
field->offset_mask || field->offset_shift)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid fields for fixed mode");
if (field->field_base < 0)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"negative header length field base (FIXED)");
node->header_length_mode = MLX5_GRAPH_NODE_LEN_FIXED;
break;
case FIELD_MODE_OFFSET:
if (!(attr->header_length_mode &
RTE_BIT32(MLX5_GRAPH_NODE_LEN_FIELD)))
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported header length field mode (OFFSET)");
node->header_length_mode = MLX5_GRAPH_NODE_LEN_FIELD;
if (field->offset_mask == 0 ||
!rte_is_power_of_2(field->offset_mask + 1))
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid length field offset mask (OFFSET)");
len_width = rte_fls_u32(field->offset_mask);
if (len_width > attr->header_length_mask_width)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"length field offset mask too wide (OFFSET)");
mask = mlx5_flex_hdr_len_mask(field->offset_shift, attr);
if (mask < field->offset_mask)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"length field shift too big (OFFSET)");
node->header_length_field_mask = RTE_MIN(mask,
field->offset_mask);
break;
case FIELD_MODE_BITMASK:
if (!(attr->header_length_mode &
RTE_BIT32(MLX5_GRAPH_NODE_LEN_BITMASK)))
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported header length field mode (BITMASK)");
if (attr->header_length_mask_width < field->field_size)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"header length field width exceeds limit");
node->header_length_mode = MLX5_GRAPH_NODE_LEN_BITMASK;
mask = mlx5_flex_hdr_len_mask(field->offset_shift, attr);
if (mask < field->offset_mask)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"length field shift too big (BITMASK)");
node->header_length_field_mask = RTE_MIN(mask,
field->offset_mask);
break;
default:
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unknown header length field mode");
}
if (field->field_base / CHAR_BIT >= 0 &&
field->field_base / CHAR_BIT > attr->max_base_header_length)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"header length field base exceeds limit");
node->header_length_base_value = field->field_base / CHAR_BIT;
if (field->field_mode == FIELD_MODE_OFFSET ||
field->field_mode == FIELD_MODE_BITMASK) {
if (field->offset_shift > 15 || field->offset_shift < 0)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"header length field shift exceeds limit");
node->header_length_field_shift = field->offset_shift;
node->header_length_field_offset = field->offset_base;
}
return 0;
}
static int
mlx5_flex_translate_next(struct mlx5_hca_flex_attr *attr,
const struct rte_flow_item_flex_conf *conf,
struct mlx5_flex_parser_devx *devx,
struct rte_flow_error *error)
{
const struct rte_flow_item_flex_field *field = &conf->next_protocol;
struct mlx5_devx_graph_node_attr *node = &devx->devx_conf;
switch (field->field_mode) {
case FIELD_MODE_DUMMY:
if (conf->nb_outputs)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"next protocol field is required (DUMMY)");
return 0;
case FIELD_MODE_FIXED:
break;
case FIELD_MODE_OFFSET:
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported next protocol field mode (OFFSET)");
break;
case FIELD_MODE_BITMASK:
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported next protocol field mode (BITMASK)");
default:
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unknown next protocol field mode");
}
MLX5_ASSERT(field->field_mode == FIELD_MODE_FIXED);
if (!conf->nb_outputs)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"out link(s) is required if next field present");
if (attr->max_next_header_offset < field->field_base)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"next protocol field base exceeds limit");
if (field->offset_shift)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported next protocol field shift");
node->next_header_field_offset = field->field_base;
node->next_header_field_size = field->field_size;
return 0;
}
/* Helper structure to handle field bit intervals. */
struct mlx5_flex_field_cover {
uint16_t num;
int32_t start[MLX5_FLEX_ITEM_MAPPING_NUM];
int32_t end[MLX5_FLEX_ITEM_MAPPING_NUM];
uint8_t mapped[MLX5_FLEX_ITEM_MAPPING_NUM / CHAR_BIT + 1];
};
static void
mlx5_flex_insert_field(struct mlx5_flex_field_cover *cover,
uint16_t num, int32_t start, int32_t end)
{
MLX5_ASSERT(num < MLX5_FLEX_ITEM_MAPPING_NUM);
MLX5_ASSERT(num <= cover->num);
if (num < cover->num) {
memmove(&cover->start[num + 1], &cover->start[num],
(cover->num - num) * sizeof(int32_t));
memmove(&cover->end[num + 1], &cover->end[num],
(cover->num - num) * sizeof(int32_t));
}
cover->start[num] = start;
cover->end[num] = end;
cover->num++;
}
static void
mlx5_flex_merge_field(struct mlx5_flex_field_cover *cover, uint16_t num)
{
uint32_t i, del = 0;
int32_t end;
MLX5_ASSERT(num < MLX5_FLEX_ITEM_MAPPING_NUM);
MLX5_ASSERT(num < (cover->num - 1));
end = cover->end[num];
for (i = num + 1; i < cover->num; i++) {
if (end < cover->start[i])
break;
del++;
if (end <= cover->end[i]) {
cover->end[num] = cover->end[i];
break;
}
}
if (del) {
MLX5_ASSERT(del < (cover->num - 1u - num));
cover->num -= del;
MLX5_ASSERT(cover->num > num);
if ((cover->num - num) > 1) {
memmove(&cover->start[num + 1],
&cover->start[num + 1 + del],
(cover->num - num - 1) * sizeof(int32_t));
memmove(&cover->end[num + 1],
&cover->end[num + 1 + del],
(cover->num - num - 1) * sizeof(int32_t));
}
}
}
/*
* Validate the sample field and update interval array
* if parameters match with the 'match" field.
* Returns:
* < 0 - error
* == 0 - no match, interval array not updated
* > 0 - match, interval array updated
*/
static int
mlx5_flex_cover_sample(struct mlx5_flex_field_cover *cover,
struct rte_flow_item_flex_field *field,
struct rte_flow_item_flex_field *match,
struct mlx5_hca_flex_attr *attr,
struct rte_flow_error *error)
{
int32_t start, end;
uint32_t i;
switch (field->field_mode) {
case FIELD_MODE_DUMMY:
return 0;
case FIELD_MODE_FIXED:
if (!(attr->sample_offset_mode &
RTE_BIT32(MLX5_GRAPH_SAMPLE_OFFSET_FIXED)))
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported sample field mode (FIXED)");
if (field->offset_shift)
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid sample field shift (FIXED");
if (field->field_base < 0)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid sample field base (FIXED)");
if (field->field_base / CHAR_BIT > attr->max_sample_base_offset)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"sample field base exceeds limit (FIXED)");
break;
case FIELD_MODE_OFFSET:
if (!(attr->sample_offset_mode &
RTE_BIT32(MLX5_GRAPH_SAMPLE_OFFSET_FIELD)))
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported sample field mode (OFFSET)");
if (field->field_base / CHAR_BIT >= 0 &&
field->field_base / CHAR_BIT > attr->max_sample_base_offset)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"sample field base exceeds limit");
break;
case FIELD_MODE_BITMASK:
if (!(attr->sample_offset_mode &
RTE_BIT32(MLX5_GRAPH_SAMPLE_OFFSET_BITMASK)))
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported sample field mode (BITMASK)");
if (field->field_base / CHAR_BIT >= 0 &&
field->field_base / CHAR_BIT > attr->max_sample_base_offset)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"sample field base exceeds limit");
break;
default:
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unknown data sample field mode");
}
if (!match) {
if (!field->field_size)
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"zero sample field width");
if (field->field_id)
DRV_LOG(DEBUG, "sample field id hint ignored");
} else {
if (field->field_mode != match->field_mode ||
field->offset_base | match->offset_base ||
field->offset_mask | match->offset_mask ||
field->offset_shift | match->offset_shift)
return 0;
}
start = field->field_base;
end = start + field->field_size;
/* Add the new or similar field to interval array. */
if (!cover->num) {
cover->start[cover->num] = start;
cover->end[cover->num] = end;
cover->num = 1;
return 1;
}
for (i = 0; i < cover->num; i++) {
if (start > cover->end[i]) {
if (i >= (cover->num - 1u)) {
mlx5_flex_insert_field(cover, cover->num,
start, end);
break;
}
continue;
}
if (end < cover->start[i]) {
mlx5_flex_insert_field(cover, i, start, end);
break;
}
if (start < cover->start[i])
cover->start[i] = start;
if (end > cover->end[i]) {
cover->end[i] = end;
if (i < (cover->num - 1u))
mlx5_flex_merge_field(cover, i);
}
break;
}
return 1;
}
static void
mlx5_flex_config_sample(struct mlx5_devx_match_sample_attr *na,
struct rte_flow_item_flex_field *field,
enum rte_flow_item_flex_tunnel_mode tunnel_mode)
{
memset(na, 0, sizeof(struct mlx5_devx_match_sample_attr));
na->flow_match_sample_en = 1;
switch (field->field_mode) {
case FIELD_MODE_FIXED:
na->flow_match_sample_offset_mode =
MLX5_GRAPH_SAMPLE_OFFSET_FIXED;
break;
case FIELD_MODE_OFFSET:
na->flow_match_sample_offset_mode =
MLX5_GRAPH_SAMPLE_OFFSET_FIELD;
na->flow_match_sample_field_offset = field->offset_base;
na->flow_match_sample_field_offset_mask = field->offset_mask;
na->flow_match_sample_field_offset_shift = field->offset_shift;
break;
case FIELD_MODE_BITMASK:
na->flow_match_sample_offset_mode =
MLX5_GRAPH_SAMPLE_OFFSET_BITMASK;
na->flow_match_sample_field_offset = field->offset_base;
na->flow_match_sample_field_offset_mask = field->offset_mask;
na->flow_match_sample_field_offset_shift = field->offset_shift;
break;
default:
MLX5_ASSERT(false);
break;
}
switch (tunnel_mode) {
case FLEX_TUNNEL_MODE_SINGLE:
/* Fallthrough */
case FLEX_TUNNEL_MODE_TUNNEL:
na->flow_match_sample_tunnel_mode =
MLX5_GRAPH_SAMPLE_TUNNEL_FIRST;
break;
case FLEX_TUNNEL_MODE_MULTI:
/* Fallthrough */
case FLEX_TUNNEL_MODE_OUTER:
na->flow_match_sample_tunnel_mode =
MLX5_GRAPH_SAMPLE_TUNNEL_OUTER;
break;
case FLEX_TUNNEL_MODE_INNER:
na->flow_match_sample_tunnel_mode =
MLX5_GRAPH_SAMPLE_TUNNEL_INNER;
break;
default:
MLX5_ASSERT(false);
break;
}
}
/* Map specified field to set/subset of allocated sample registers. */
static int
mlx5_flex_map_sample(struct rte_flow_item_flex_field *field,
struct mlx5_flex_parser_devx *parser,
struct mlx5_flex_item *item,
struct rte_flow_error *error)
{
struct mlx5_devx_match_sample_attr node;
int32_t start = field->field_base;
int32_t end = start + field->field_size;
struct mlx5_flex_pattern_field *trans;
uint32_t i, done_bits = 0;
if (field->field_mode == FIELD_MODE_DUMMY) {
done_bits = field->field_size;
while (done_bits) {
uint32_t part = RTE_MIN(done_bits,
sizeof(uint32_t) * CHAR_BIT);
if (item->mapnum >= MLX5_FLEX_ITEM_MAPPING_NUM)
return rte_flow_error_set
(error,
EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"too many flex item pattern translations");
trans = &item->map[item->mapnum];
trans->reg_id = MLX5_INVALID_SAMPLE_REG_ID;
trans->shift = 0;
trans->width = part;
item->mapnum++;
done_bits -= part;
}
return 0;
}
mlx5_flex_config_sample(&node, field, item->tunnel_mode);
for (i = 0; i < parser->num_samples; i++) {
struct mlx5_devx_match_sample_attr *sample =
&parser->devx_conf.sample[i];
int32_t reg_start, reg_end;
int32_t cov_start, cov_end;
MLX5_ASSERT(sample->flow_match_sample_en);
if (!sample->flow_match_sample_en)
break;
node.flow_match_sample_field_base_offset =
sample->flow_match_sample_field_base_offset;
if (memcmp(&node, sample, sizeof(node)))
continue;
reg_start = (int8_t)sample->flow_match_sample_field_base_offset;
reg_start *= CHAR_BIT;
reg_end = reg_start + 32;
if (end <= reg_start || start >= reg_end)
continue;
cov_start = RTE_MAX(reg_start, start);
cov_end = RTE_MIN(reg_end, end);
MLX5_ASSERT(cov_end > cov_start);
done_bits += cov_end - cov_start;
if (item->mapnum >= MLX5_FLEX_ITEM_MAPPING_NUM)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"too many flex item pattern translations");
trans = &item->map[item->mapnum];
item->mapnum++;
trans->reg_id = i;
trans->shift = cov_start - reg_start;
trans->width = cov_end - cov_start;
}
if (done_bits != field->field_size) {
MLX5_ASSERT(false);
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"failed to map field to sample register");
}
return 0;
}
/* Allocate sample registers for the specified field type and interval array. */
static int
mlx5_flex_alloc_sample(struct mlx5_flex_field_cover *cover,
struct mlx5_flex_parser_devx *parser,
struct mlx5_flex_item *item,
struct rte_flow_item_flex_field *field,
struct mlx5_hca_flex_attr *attr,
struct rte_flow_error *error)
{
struct mlx5_devx_match_sample_attr node;
uint32_t idx = 0;
mlx5_flex_config_sample(&node, field, item->tunnel_mode);
while (idx < cover->num) {
int32_t start, end;
/*
* Sample base offsets are in bytes, should be aligned
* to 32-bit as required by firmware for samples.
*/
start = RTE_ALIGN_FLOOR(cover->start[idx],
sizeof(uint32_t) * CHAR_BIT);
node.flow_match_sample_field_base_offset =
(start / CHAR_BIT) & 0xFF;
/* Allocate sample register. */
if (parser->num_samples >= MLX5_GRAPH_NODE_SAMPLE_NUM ||
parser->num_samples >= attr->max_num_sample ||
parser->num_samples >= attr->max_num_prog_sample)
return rte_flow_error_set
(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"no sample registers to handle all flex item fields");
parser->devx_conf.sample[parser->num_samples] = node;
parser->num_samples++;
/* Remove or update covered intervals. */
end = start + 32;
while (idx < cover->num) {
if (end >= cover->end[idx]) {
idx++;
continue;
}
if (end > cover->start[idx])
cover->start[idx] = end;
break;
}
}
return 0;
}
static int
mlx5_flex_translate_sample(struct mlx5_hca_flex_attr *attr,
const struct rte_flow_item_flex_conf *conf,
struct mlx5_flex_parser_devx *parser,
struct mlx5_flex_item *item,
struct rte_flow_error *error)
{
struct mlx5_flex_field_cover cover;
uint32_t i, j;
int ret;
switch (conf->tunnel) {
case FLEX_TUNNEL_MODE_SINGLE:
/* Fallthrough */
case FLEX_TUNNEL_MODE_OUTER:
/* Fallthrough */
case FLEX_TUNNEL_MODE_INNER:
/* Fallthrough */
case FLEX_TUNNEL_MODE_MULTI:
/* Fallthrough */
case FLEX_TUNNEL_MODE_TUNNEL:
break;
default:
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unrecognized tunnel mode");
}
item->tunnel_mode = conf->tunnel;
if (conf->nb_samples > MLX5_FLEX_ITEM_MAPPING_NUM)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"sample field number exceeds limit");
/*
* The application can specify fields smaller or bigger than 32 bits
* covered with single sample register and it can specify field
* offsets in any order.
*
* Gather all similar fields together, build array of bit intervals
* in asсending order and try to cover with the smallest set of sample
* registers.
*/
memset(&cover, 0, sizeof(cover));
for (i = 0; i < conf->nb_samples; i++) {
struct rte_flow_item_flex_field *fl = conf->sample_data + i;
/* Check whether field was covered in the previous iteration. */
if (cover.mapped[i / CHAR_BIT] & (1u << (i % CHAR_BIT)))
continue;
if (fl->field_mode == FIELD_MODE_DUMMY)
continue;
/* Build an interval array for the field and similar ones */
cover.num = 0;
/* Add the first field to array unconditionally. */
ret = mlx5_flex_cover_sample(&cover, fl, NULL, attr, error);
if (ret < 0)
return ret;
MLX5_ASSERT(ret > 0);
cover.mapped[i / CHAR_BIT] |= 1u << (i % CHAR_BIT);
for (j = i + 1; j < conf->nb_samples; j++) {
struct rte_flow_item_flex_field *ft;
/* Add field to array if its type matches. */
ft = conf->sample_data + j;
ret = mlx5_flex_cover_sample(&cover, ft, fl,
attr, error);
if (ret < 0)
return ret;
if (!ret)
continue;
cover.mapped[j / CHAR_BIT] |= 1u << (j % CHAR_BIT);
}
/* Allocate sample registers to cover array of intervals. */
ret = mlx5_flex_alloc_sample(&cover, parser, item,
fl, attr, error);
if (ret)
return ret;
}
/* Build the item pattern translating data on flow creation. */
item->mapnum = 0;
memset(&item->map, 0, sizeof(item->map));
for (i = 0; i < conf->nb_samples; i++) {
struct rte_flow_item_flex_field *fl = conf->sample_data + i;
ret = mlx5_flex_map_sample(fl, parser, item, error);
if (ret) {
MLX5_ASSERT(false);
return ret;
}
}
if (conf->tunnel == FLEX_TUNNEL_MODE_MULTI) {
/*
* In FLEX_TUNNEL_MODE_MULTI tunnel mode PMD creates 2 sets
* of samples. The first set is for outer and the second set
* for inner flex flow item. Outer and inner samples differ
* only in tunnel_mode.
*/
if (parser->num_samples > MLX5_GRAPH_NODE_SAMPLE_NUM / 2)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"no sample registers for inner");
rte_memcpy(parser->devx_conf.sample + parser->num_samples,
parser->devx_conf.sample,
parser->num_samples *
sizeof(parser->devx_conf.sample[0]));
for (i = 0; i < parser->num_samples; i++) {
struct mlx5_devx_match_sample_attr *sm = i +
parser->devx_conf.sample + parser->num_samples;
sm->flow_match_sample_tunnel_mode =
MLX5_GRAPH_SAMPLE_TUNNEL_INNER;
}
parser->num_samples *= 2;
}
return 0;
}
static int
mlx5_flex_arc_type(enum rte_flow_item_type type, int in)
{
switch (type) {
case RTE_FLOW_ITEM_TYPE_ETH:
return MLX5_GRAPH_ARC_NODE_MAC;
case RTE_FLOW_ITEM_TYPE_IPV4:
return in ? MLX5_GRAPH_ARC_NODE_IP : MLX5_GRAPH_ARC_NODE_IPV4;
case RTE_FLOW_ITEM_TYPE_IPV6:
return in ? MLX5_GRAPH_ARC_NODE_IP : MLX5_GRAPH_ARC_NODE_IPV6;
case RTE_FLOW_ITEM_TYPE_UDP:
return MLX5_GRAPH_ARC_NODE_UDP;
case RTE_FLOW_ITEM_TYPE_TCP:
return MLX5_GRAPH_ARC_NODE_TCP;
case RTE_FLOW_ITEM_TYPE_MPLS:
return MLX5_GRAPH_ARC_NODE_MPLS;
case RTE_FLOW_ITEM_TYPE_GRE:
return MLX5_GRAPH_ARC_NODE_GRE;
case RTE_FLOW_ITEM_TYPE_GENEVE:
return MLX5_GRAPH_ARC_NODE_GENEVE;
case RTE_FLOW_ITEM_TYPE_VXLAN_GPE:
return MLX5_GRAPH_ARC_NODE_VXLAN_GPE;
default:
return -EINVAL;
}
}
static int
mlx5_flex_arc_in_eth(const struct rte_flow_item *item,
struct rte_flow_error *error)
{
const struct rte_flow_item_eth *spec = item->spec;
const struct rte_flow_item_eth *mask = item->mask;
struct rte_flow_item_eth eth = { .hdr.ether_type = RTE_BE16(0xFFFF) };
if (memcmp(mask, &eth, sizeof(struct rte_flow_item_eth))) {
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, item,
"invalid eth item mask");
}
return rte_be_to_cpu_16(spec->hdr.ether_type);
}
static int
mlx5_flex_arc_in_udp(const struct rte_flow_item *item,
struct rte_flow_error *error)
{
const struct rte_flow_item_udp *spec = item->spec;
const struct rte_flow_item_udp *mask = item->mask;
struct rte_flow_item_udp udp = { .hdr.dst_port = RTE_BE16(0xFFFF) };
if (memcmp(mask, &udp, sizeof(struct rte_flow_item_udp))) {
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, item,
"invalid eth item mask");
}
return rte_be_to_cpu_16(spec->hdr.dst_port);
}
static int
mlx5_flex_translate_arc_in(struct mlx5_hca_flex_attr *attr,
const struct rte_flow_item_flex_conf *conf,
struct mlx5_flex_parser_devx *devx,
struct mlx5_flex_item *item,
struct rte_flow_error *error)
{
struct mlx5_devx_graph_node_attr *node = &devx->devx_conf;
uint32_t i;
RTE_SET_USED(item);
if (conf->nb_inputs > attr->max_num_arc_in)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"too many input links");
for (i = 0; i < conf->nb_inputs; i++) {
struct mlx5_devx_graph_arc_attr *arc = node->in + i;
struct rte_flow_item_flex_link *link = conf->input_link + i;
const struct rte_flow_item *rte_item = &link->item;
int arc_type;
int ret;
if (!rte_item->spec || !rte_item->mask || rte_item->last)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid flex item IN arc format");
arc_type = mlx5_flex_arc_type(rte_item->type, true);
if (arc_type < 0 || !(attr->node_in & RTE_BIT32(arc_type)))
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported flex item IN arc type");
arc->arc_parse_graph_node = arc_type;
arc->start_inner_tunnel = 0;
/*
* Configure arc IN condition value. The value location depends
* on protocol. Current FW version supports IP & UDP for IN
* arcs only, and locations for these protocols are defined.
* Add more protocols when available.
*/
switch (rte_item->type) {
case RTE_FLOW_ITEM_TYPE_ETH:
ret = mlx5_flex_arc_in_eth(rte_item, error);
break;
case RTE_FLOW_ITEM_TYPE_UDP:
ret = mlx5_flex_arc_in_udp(rte_item, error);
break;
default:
MLX5_ASSERT(false);
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported flex item IN arc type");
}
if (ret < 0)
return ret;
arc->compare_condition_value = (uint16_t)ret;
}
return 0;
}
static int
mlx5_flex_translate_arc_out(struct mlx5_hca_flex_attr *attr,
const struct rte_flow_item_flex_conf *conf,
struct mlx5_flex_parser_devx *devx,
struct mlx5_flex_item *item,
struct rte_flow_error *error)
{
struct mlx5_devx_graph_node_attr *node = &devx->devx_conf;
bool is_tunnel = conf->tunnel == FLEX_TUNNEL_MODE_TUNNEL;
uint32_t i;
RTE_SET_USED(item);
if (conf->nb_outputs > attr->max_num_arc_out)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"too many output links");
for (i = 0; i < conf->nb_outputs; i++) {
struct mlx5_devx_graph_arc_attr *arc = node->out + i;
struct rte_flow_item_flex_link *link = conf->output_link + i;
const struct rte_flow_item *rte_item = &link->item;
int arc_type;
if (rte_item->spec || rte_item->mask || rte_item->last)
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"flex node: invalid OUT arc format");
arc_type = mlx5_flex_arc_type(rte_item->type, false);
if (arc_type < 0 || !(attr->node_out & RTE_BIT32(arc_type)))
return rte_flow_error_set
(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"unsupported flex item OUT arc type");
arc->arc_parse_graph_node = arc_type;
arc->start_inner_tunnel = !!is_tunnel;
arc->compare_condition_value = link->next;
}
return 0;
}
/* Translate RTE flex item API configuration into flaex parser settings. */
static int
mlx5_flex_translate_conf(struct rte_eth_dev *dev,
const struct rte_flow_item_flex_conf *conf,
struct mlx5_flex_parser_devx *devx,
struct mlx5_flex_item *item,
struct rte_flow_error *error)
{
struct mlx5_priv *priv = dev->data->dev_private;
struct mlx5_hca_flex_attr *attr = &priv->config.hca_attr.flex;
int ret;
ret = mlx5_flex_translate_length(attr, conf, devx, error);
if (ret)
return ret;
ret = mlx5_flex_translate_next(attr, conf, devx, error);
if (ret)
return ret;
ret = mlx5_flex_translate_sample(attr, conf, devx, item, error);
if (ret)
return ret;
ret = mlx5_flex_translate_arc_in(attr, conf, devx, item, error);
if (ret)
return ret;
ret = mlx5_flex_translate_arc_out(attr, conf, devx, item, error);
if (ret)
return ret;
return 0;
}
/**
* Create the flex item with specified configuration over the Ethernet device.
*
* @param dev
* Ethernet device to create flex item on.
* @param[in] conf
* Flex item configuration.
* @param[out] error
* Perform verbose error reporting if not NULL. PMDs initialize this
* structure in case of error only.
*
* @return
* Non-NULL opaque pointer on success, NULL otherwise and rte_errno is set.
*/
struct rte_flow_item_flex_handle *
flow_dv_item_create(struct rte_eth_dev *dev,
const struct rte_flow_item_flex_conf *conf,
struct rte_flow_error *error)
{
struct mlx5_priv *priv = dev->data->dev_private;
struct mlx5_flex_parser_devx devx_config = { .devx_obj = NULL };
struct mlx5_flex_item *flex;
struct mlx5_list_entry *ent;
MLX5_ASSERT(rte_eal_process_type() == RTE_PROC_PRIMARY);
flex = mlx5_flex_alloc(priv);
if (!flex) {
rte_flow_error_set(error, ENOMEM,
RTE_FLOW_ERROR_TYPE_UNSPECIFIED, NULL,
"too many flex items created on the port");
return NULL;
}
if (mlx5_flex_translate_conf(dev, conf, &devx_config, flex, error))
goto error;
ent = mlx5_list_register(priv->sh->flex_parsers_dv, &devx_config);
if (!ent) {
rte_flow_error_set(error, ENOMEM,
RTE_FLOW_ERROR_TYPE_UNSPECIFIED, NULL,
"flex item creation failure");
goto error;
}
flex->devx_fp = container_of(ent, struct mlx5_flex_parser_devx, entry);
/* Mark initialized flex item valid. */
__atomic_add_fetch(&flex->refcnt, 1, __ATOMIC_RELEASE);
return (struct rte_flow_item_flex_handle *)flex;
error:
mlx5_flex_free(priv, flex);
return NULL;
}
/**
* Release the flex item on the specified Ethernet device.
*
* @param dev
* Ethernet device to destroy flex item on.
* @param[in] handle
* Handle of the item existing on the specified device.
* @param[out] error
* Perform verbose error reporting if not NULL. PMDs initialize this
* structure in case of error only.
*
* @return
* 0 on success, a negative errno value otherwise and rte_errno is set.
*/
int
flow_dv_item_release(struct rte_eth_dev *dev,
const struct rte_flow_item_flex_handle *handle,
struct rte_flow_error *error)
{
struct mlx5_priv *priv = dev->data->dev_private;
struct mlx5_flex_item *flex =
(struct mlx5_flex_item *)(uintptr_t)handle;
uint32_t old_refcnt = 1;
int rc;
MLX5_ASSERT(rte_eal_process_type() == RTE_PROC_PRIMARY);
rte_spinlock_lock(&priv->flex_item_sl);
if (mlx5_flex_index(priv, flex) < 0) {
rte_spinlock_unlock(&priv->flex_item_sl);
return rte_flow_error_set(error, EINVAL,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"invalid flex item handle value");
}
if (!__atomic_compare_exchange_n(&flex->refcnt, &old_refcnt, 0, 0,
__ATOMIC_ACQUIRE, __ATOMIC_RELAXED)) {
rte_spinlock_unlock(&priv->flex_item_sl);
return rte_flow_error_set(error, EBUSY,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"flex item has flow references");
}
/* Flex item is marked as invalid, we can leave locked section. */
rte_spinlock_unlock(&priv->flex_item_sl);
MLX5_ASSERT(flex->devx_fp);
rc = mlx5_list_unregister(priv->sh->flex_parsers_dv,
&flex->devx_fp->entry);
flex->devx_fp = NULL;
mlx5_flex_free(priv, flex);
if (rc < 0)
return rte_flow_error_set(error, EBUSY,
RTE_FLOW_ERROR_TYPE_ITEM, NULL,
"flex item release failure");
return 0;
}
/* DevX flex parser list callbacks. */
struct mlx5_list_entry *
mlx5_flex_parser_create_cb(void *list_ctx, void *ctx)
{
struct mlx5_dev_ctx_shared *sh = list_ctx;
struct mlx5_flex_parser_devx *fp, *conf = ctx;
int ret;
fp = mlx5_malloc(MLX5_MEM_ZERO, sizeof(struct mlx5_flex_parser_devx),
0, SOCKET_ID_ANY);
if (!fp)
return NULL;
/* Copy the requested configurations. */
fp->num_samples = conf->num_samples;
memcpy(&fp->devx_conf, &conf->devx_conf, sizeof(fp->devx_conf));
/* Create DevX flex parser. */
fp->devx_obj = mlx5_devx_cmd_create_flex_parser(sh->cdev->ctx,
&fp->devx_conf);
if (!fp->devx_obj)
goto error;
/* Query the firmware assigned sample ids. */
ret = mlx5_devx_cmd_query_parse_samples(fp->devx_obj,
fp->sample_ids,
fp->num_samples);
if (ret)
goto error;
DRV_LOG(DEBUG, "DEVx flex parser %p created, samples num: %u",
(const void *)fp, fp->num_samples);
return &fp->entry;
error:
if (fp->devx_obj)
mlx5_devx_cmd_destroy((void *)(uintptr_t)fp->devx_obj);
if (fp)
mlx5_free(fp);
return NULL;
}
int
mlx5_flex_parser_match_cb(void *list_ctx,
struct mlx5_list_entry *iter, void *ctx)
{
struct mlx5_flex_parser_devx *fp =
container_of(iter, struct mlx5_flex_parser_devx, entry);
struct mlx5_flex_parser_devx *org =
container_of(ctx, struct mlx5_flex_parser_devx, entry);
RTE_SET_USED(list_ctx);
return !iter || !ctx || memcmp(&fp->devx_conf,
&org->devx_conf,
sizeof(fp->devx_conf));
}
void
mlx5_flex_parser_remove_cb(void *list_ctx, struct mlx5_list_entry *entry)
{
struct mlx5_flex_parser_devx *fp =
container_of(entry, struct mlx5_flex_parser_devx, entry);
RTE_SET_USED(list_ctx);
MLX5_ASSERT(fp->devx_obj);
claim_zero(mlx5_devx_cmd_destroy(fp->devx_obj));
DRV_LOG(DEBUG, "DEVx flex parser %p destroyed", (const void *)fp);
mlx5_free(entry);
}
struct mlx5_list_entry *
mlx5_flex_parser_clone_cb(void *list_ctx,
struct mlx5_list_entry *entry, void *ctx)
{
struct mlx5_flex_parser_devx *fp;
RTE_SET_USED(list_ctx);
RTE_SET_USED(entry);
fp = mlx5_malloc(0, sizeof(struct mlx5_flex_parser_devx),
0, SOCKET_ID_ANY);
if (!fp)
return NULL;
memcpy(fp, ctx, sizeof(struct mlx5_flex_parser_devx));
return &fp->entry;
}
void
mlx5_flex_parser_clone_free_cb(void *list_ctx, struct mlx5_list_entry *entry)
{
struct mlx5_flex_parser_devx *fp =
container_of(entry, struct mlx5_flex_parser_devx, entry);
RTE_SET_USED(list_ctx);
mlx5_free(fp);
}