/* SPDX-License-Identifier: BSD-3-Clause * Copyright(c) 2010-2018 Intel Corporation */ #include #include #include #include #include #include #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; }