99a2dd955f
There is no reason for the DPDK libraries to all have 'librte_' prefix on the directory names. This prefix makes the directory names longer and also makes it awkward to add features referring to individual libraries in the build - should the lib names be specified with or without the prefix. Therefore, we can just remove the library prefix and use the library's unique name as the directory name, i.e. 'eal' rather than 'librte_eal' Signed-off-by: Bruce Richardson <bruce.richardson@intel.com>
532 lines
11 KiB
C
532 lines
11 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright(c) 2010-2018 Intel Corporation
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <rte_common.h>
|
|
#include <rte_byteorder.h>
|
|
#include <rte_malloc.h>
|
|
#include <rte_memcpy.h>
|
|
|
|
#include "rte_port_in_action.h"
|
|
|
|
/**
|
|
* RTE_PORT_IN_ACTION_FLTR
|
|
*/
|
|
static int
|
|
fltr_cfg_check(struct rte_port_in_action_fltr_config *cfg)
|
|
{
|
|
if (cfg == NULL)
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct fltr_data {
|
|
uint32_t port_id;
|
|
};
|
|
|
|
static void
|
|
fltr_init(struct fltr_data *data,
|
|
struct rte_port_in_action_fltr_config *cfg)
|
|
{
|
|
data->port_id = cfg->port_id;
|
|
}
|
|
|
|
static int
|
|
fltr_apply(struct fltr_data *data,
|
|
struct rte_port_in_action_fltr_params *p)
|
|
{
|
|
/* Check input arguments */
|
|
if (p == NULL)
|
|
return -1;
|
|
|
|
data->port_id = p->port_id;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* RTE_PORT_IN_ACTION_LB
|
|
*/
|
|
static int
|
|
lb_cfg_check(struct rte_port_in_action_lb_config *cfg)
|
|
{
|
|
if ((cfg == NULL) ||
|
|
(cfg->key_size < RTE_PORT_IN_ACTION_LB_KEY_SIZE_MIN) ||
|
|
(cfg->key_size > RTE_PORT_IN_ACTION_LB_KEY_SIZE_MAX) ||
|
|
(!rte_is_power_of_2(cfg->key_size)) ||
|
|
(cfg->f_hash == NULL))
|
|
return -1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct lb_data {
|
|
uint32_t port_id[RTE_PORT_IN_ACTION_LB_TABLE_SIZE];
|
|
};
|
|
|
|
static void
|
|
lb_init(struct lb_data *data,
|
|
struct rte_port_in_action_lb_config *cfg)
|
|
{
|
|
memcpy(data->port_id, cfg->port_id, sizeof(cfg->port_id));
|
|
}
|
|
|
|
static int
|
|
lb_apply(struct lb_data *data,
|
|
struct rte_port_in_action_lb_params *p)
|
|
{
|
|
/* Check input arguments */
|
|
if (p == NULL)
|
|
return -1;
|
|
|
|
memcpy(data->port_id, p->port_id, sizeof(p->port_id));
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Action profile
|
|
*/
|
|
static int
|
|
action_valid(enum rte_port_in_action_type action)
|
|
{
|
|
switch (action) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
return 1;
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
#define RTE_PORT_IN_ACTION_MAX 64
|
|
|
|
struct ap_config {
|
|
uint64_t action_mask;
|
|
struct rte_port_in_action_fltr_config fltr;
|
|
struct rte_port_in_action_lb_config lb;
|
|
};
|
|
|
|
static size_t
|
|
action_cfg_size(enum rte_port_in_action_type action)
|
|
{
|
|
switch (action) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
return sizeof(struct rte_port_in_action_fltr_config);
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
return sizeof(struct rte_port_in_action_lb_config);
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static void*
|
|
action_cfg_get(struct ap_config *ap_config,
|
|
enum rte_port_in_action_type type)
|
|
{
|
|
switch (type) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
return &ap_config->fltr;
|
|
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
return &ap_config->lb;
|
|
|
|
default:
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
static void
|
|
action_cfg_set(struct ap_config *ap_config,
|
|
enum rte_port_in_action_type type,
|
|
void *action_cfg)
|
|
{
|
|
void *dst = action_cfg_get(ap_config, type);
|
|
|
|
if (dst)
|
|
memcpy(dst, action_cfg, action_cfg_size(type));
|
|
|
|
ap_config->action_mask |= 1LLU << type;
|
|
}
|
|
|
|
struct ap_data {
|
|
size_t offset[RTE_PORT_IN_ACTION_MAX];
|
|
size_t total_size;
|
|
};
|
|
|
|
static size_t
|
|
action_data_size(enum rte_port_in_action_type action,
|
|
struct ap_config *ap_config __rte_unused)
|
|
{
|
|
switch (action) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
return sizeof(struct fltr_data);
|
|
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
return sizeof(struct lb_data);
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
static void
|
|
action_data_offset_set(struct ap_data *ap_data,
|
|
struct ap_config *ap_config)
|
|
{
|
|
uint64_t action_mask = ap_config->action_mask;
|
|
size_t offset;
|
|
uint32_t action;
|
|
|
|
memset(ap_data->offset, 0, sizeof(ap_data->offset));
|
|
|
|
offset = 0;
|
|
for (action = 0; action < RTE_PORT_IN_ACTION_MAX; action++)
|
|
if (action_mask & (1LLU << action)) {
|
|
ap_data->offset[action] = offset;
|
|
offset += action_data_size((enum rte_port_in_action_type)action,
|
|
ap_config);
|
|
}
|
|
|
|
ap_data->total_size = offset;
|
|
}
|
|
|
|
struct rte_port_in_action_profile {
|
|
struct ap_config cfg;
|
|
struct ap_data data;
|
|
int frozen;
|
|
};
|
|
|
|
struct rte_port_in_action_profile *
|
|
rte_port_in_action_profile_create(uint32_t socket_id)
|
|
{
|
|
struct rte_port_in_action_profile *ap;
|
|
|
|
/* Memory allocation */
|
|
ap = rte_zmalloc_socket(NULL,
|
|
sizeof(struct rte_port_in_action_profile),
|
|
RTE_CACHE_LINE_SIZE,
|
|
socket_id);
|
|
if (ap == NULL)
|
|
return NULL;
|
|
|
|
return ap;
|
|
}
|
|
|
|
int
|
|
rte_port_in_action_profile_action_register(struct rte_port_in_action_profile *profile,
|
|
enum rte_port_in_action_type type,
|
|
void *action_config)
|
|
{
|
|
int status;
|
|
|
|
/* Check input arguments */
|
|
if ((profile == NULL) ||
|
|
profile->frozen ||
|
|
(action_valid(type) == 0) ||
|
|
(profile->cfg.action_mask & (1LLU << type)) ||
|
|
((action_cfg_size(type) == 0) && action_config) ||
|
|
(action_cfg_size(type) && (action_config == NULL)))
|
|
return -EINVAL;
|
|
|
|
switch (type) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
status = fltr_cfg_check(action_config);
|
|
break;
|
|
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
status = lb_cfg_check(action_config);
|
|
break;
|
|
|
|
default:
|
|
status = 0;
|
|
break;
|
|
}
|
|
|
|
if (status)
|
|
return status;
|
|
|
|
/* Action enable */
|
|
action_cfg_set(&profile->cfg, type, action_config);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
rte_port_in_action_profile_freeze(struct rte_port_in_action_profile *profile)
|
|
{
|
|
if (profile->frozen)
|
|
return -EBUSY;
|
|
|
|
action_data_offset_set(&profile->data, &profile->cfg);
|
|
profile->frozen = 1;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
rte_port_in_action_profile_free(struct rte_port_in_action_profile *profile)
|
|
{
|
|
if (profile == NULL)
|
|
return 0;
|
|
|
|
free(profile);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Action
|
|
*/
|
|
struct rte_port_in_action {
|
|
struct ap_config cfg;
|
|
struct ap_data data;
|
|
uint8_t memory[0] __rte_cache_aligned;
|
|
};
|
|
|
|
static __rte_always_inline void *
|
|
action_data_get(struct rte_port_in_action *action,
|
|
enum rte_port_in_action_type type)
|
|
{
|
|
size_t offset = action->data.offset[type];
|
|
|
|
return &action->memory[offset];
|
|
}
|
|
|
|
static void
|
|
action_data_init(struct rte_port_in_action *action,
|
|
enum rte_port_in_action_type type)
|
|
{
|
|
void *data = action_data_get(action, type);
|
|
|
|
switch (type) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
fltr_init(data, &action->cfg.fltr);
|
|
return;
|
|
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
lb_init(data, &action->cfg.lb);
|
|
return;
|
|
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
|
|
struct rte_port_in_action *
|
|
rte_port_in_action_create(struct rte_port_in_action_profile *profile,
|
|
uint32_t socket_id)
|
|
{
|
|
struct rte_port_in_action *action;
|
|
size_t size;
|
|
uint32_t i;
|
|
|
|
/* Check input arguments */
|
|
if ((profile == NULL) ||
|
|
(profile->frozen == 0))
|
|
return NULL;
|
|
|
|
/* Memory allocation */
|
|
size = sizeof(struct rte_port_in_action) + profile->data.total_size;
|
|
size = RTE_CACHE_LINE_ROUNDUP(size);
|
|
|
|
action = rte_zmalloc_socket(NULL,
|
|
size,
|
|
RTE_CACHE_LINE_SIZE,
|
|
socket_id);
|
|
if (action == NULL)
|
|
return NULL;
|
|
|
|
/* Initialization */
|
|
memcpy(&action->cfg, &profile->cfg, sizeof(profile->cfg));
|
|
memcpy(&action->data, &profile->data, sizeof(profile->data));
|
|
|
|
for (i = 0; i < RTE_PORT_IN_ACTION_MAX; i++)
|
|
if (action->cfg.action_mask & (1LLU << i))
|
|
action_data_init(action,
|
|
(enum rte_port_in_action_type)i);
|
|
|
|
return action;
|
|
}
|
|
|
|
int
|
|
rte_port_in_action_apply(struct rte_port_in_action *action,
|
|
enum rte_port_in_action_type type,
|
|
void *action_params)
|
|
{
|
|
void *action_data;
|
|
|
|
/* Check input arguments */
|
|
if ((action == NULL) ||
|
|
(action_valid(type) == 0) ||
|
|
((action->cfg.action_mask & (1LLU << type)) == 0) ||
|
|
(action_params == NULL))
|
|
return -EINVAL;
|
|
|
|
/* Data update */
|
|
action_data = action_data_get(action, type);
|
|
|
|
switch (type) {
|
|
case RTE_PORT_IN_ACTION_FLTR:
|
|
return fltr_apply(action_data,
|
|
action_params);
|
|
|
|
case RTE_PORT_IN_ACTION_LB:
|
|
return lb_apply(action_data,
|
|
action_params);
|
|
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
static int
|
|
ah_filter_on_match(struct rte_pipeline *p,
|
|
struct rte_mbuf **pkts,
|
|
uint32_t n_pkts,
|
|
void *arg)
|
|
{
|
|
struct rte_port_in_action *action = arg;
|
|
struct rte_port_in_action_fltr_config *cfg = &action->cfg.fltr;
|
|
uint64_t *key_mask = (uint64_t *) cfg->key_mask;
|
|
uint64_t *key = (uint64_t *) cfg->key;
|
|
uint32_t key_offset = cfg->key_offset;
|
|
struct fltr_data *data = action_data_get(action,
|
|
RTE_PORT_IN_ACTION_FLTR);
|
|
uint32_t i;
|
|
|
|
for (i = 0; i < n_pkts; i++) {
|
|
struct rte_mbuf *pkt = pkts[i];
|
|
uint64_t *pkt_key = RTE_MBUF_METADATA_UINT64_PTR(pkt,
|
|
key_offset);
|
|
|
|
uint64_t xor0 = (pkt_key[0] & key_mask[0]) ^ key[0];
|
|
uint64_t xor1 = (pkt_key[1] & key_mask[1]) ^ key[1];
|
|
uint64_t or = xor0 | xor1;
|
|
|
|
if (or == 0) {
|
|
rte_pipeline_ah_packet_hijack(p, 1LLU << i);
|
|
rte_pipeline_port_out_packet_insert(p,
|
|
data->port_id, pkt);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ah_filter_on_mismatch(struct rte_pipeline *p,
|
|
struct rte_mbuf **pkts,
|
|
uint32_t n_pkts,
|
|
void *arg)
|
|
{
|
|
struct rte_port_in_action *action = arg;
|
|
struct rte_port_in_action_fltr_config *cfg = &action->cfg.fltr;
|
|
uint64_t *key_mask = (uint64_t *) cfg->key_mask;
|
|
uint64_t *key = (uint64_t *) cfg->key;
|
|
uint32_t key_offset = cfg->key_offset;
|
|
struct fltr_data *data = action_data_get(action,
|
|
RTE_PORT_IN_ACTION_FLTR);
|
|
uint32_t i;
|
|
|
|
for (i = 0; i < n_pkts; i++) {
|
|
struct rte_mbuf *pkt = pkts[i];
|
|
uint64_t *pkt_key = RTE_MBUF_METADATA_UINT64_PTR(pkt,
|
|
key_offset);
|
|
|
|
uint64_t xor0 = (pkt_key[0] & key_mask[0]) ^ key[0];
|
|
uint64_t xor1 = (pkt_key[1] & key_mask[1]) ^ key[1];
|
|
uint64_t or = xor0 | xor1;
|
|
|
|
if (or) {
|
|
rte_pipeline_ah_packet_hijack(p, 1LLU << i);
|
|
rte_pipeline_port_out_packet_insert(p,
|
|
data->port_id, pkt);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
ah_lb(struct rte_pipeline *p,
|
|
struct rte_mbuf **pkts,
|
|
uint32_t n_pkts,
|
|
void *arg)
|
|
{
|
|
struct rte_port_in_action *action = arg;
|
|
struct rte_port_in_action_lb_config *cfg = &action->cfg.lb;
|
|
struct lb_data *data = action_data_get(action, RTE_PORT_IN_ACTION_LB);
|
|
uint64_t pkt_mask = RTE_LEN2MASK(n_pkts, uint64_t);
|
|
uint32_t i;
|
|
|
|
rte_pipeline_ah_packet_hijack(p, pkt_mask);
|
|
|
|
for (i = 0; i < n_pkts; i++) {
|
|
struct rte_mbuf *pkt = pkts[i];
|
|
uint8_t *pkt_key = RTE_MBUF_METADATA_UINT8_PTR(pkt,
|
|
cfg->key_offset);
|
|
|
|
uint64_t digest = cfg->f_hash(pkt_key,
|
|
cfg->key_mask,
|
|
cfg->key_size,
|
|
cfg->seed);
|
|
uint64_t pos = digest & (RTE_PORT_IN_ACTION_LB_TABLE_SIZE - 1);
|
|
uint32_t port_id = data->port_id[pos];
|
|
|
|
rte_pipeline_port_out_packet_insert(p, port_id, pkt);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static rte_pipeline_port_in_action_handler
|
|
ah_selector(struct rte_port_in_action *action)
|
|
{
|
|
if (action->cfg.action_mask == 0)
|
|
return NULL;
|
|
|
|
if (action->cfg.action_mask == 1LLU << RTE_PORT_IN_ACTION_FLTR)
|
|
return (action->cfg.fltr.filter_on_match) ?
|
|
ah_filter_on_match : ah_filter_on_mismatch;
|
|
|
|
if (action->cfg.action_mask == 1LLU << RTE_PORT_IN_ACTION_LB)
|
|
return ah_lb;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int
|
|
rte_port_in_action_params_get(struct rte_port_in_action *action,
|
|
struct rte_pipeline_port_in_params *params)
|
|
{
|
|
rte_pipeline_port_in_action_handler f_action;
|
|
|
|
/* Check input arguments */
|
|
if ((action == NULL) ||
|
|
(params == NULL))
|
|
return -EINVAL;
|
|
|
|
f_action = ah_selector(action);
|
|
|
|
/* Fill in params */
|
|
params->f_action = f_action;
|
|
params->arg_ah = (f_action) ? action : NULL;
|
|
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
rte_port_in_action_free(struct rte_port_in_action *action)
|
|
{
|
|
if (action == NULL)
|
|
return 0;
|
|
|
|
rte_free(action);
|
|
|
|
return 0;
|
|
}
|