* [RFC PATCH 0/3] add feature arc in rte_graph @ 2024-09-07 7:31 Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 1/3] graph: add feature arc support Nitin Saxena ` (4 more replies) 0 siblings, 5 replies; 55+ messages in thread From: Nitin Saxena @ 2024-09-07 7:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on wheter feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Nitin Saxena (3): graph: add feature arc support graph: add feature arc option in graph create graph: add IPv4 output feature arc lib/graph/graph.c | 1 + lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ lib/graph/graph_populate.c | 7 +- lib/graph/graph_private.h | 3 + lib/graph/meson.build | 2 + lib/graph/node.c | 2 + lib/graph/rte_graph.h | 3 + lib/graph/rte_graph_feature_arc.h | 373 +++++++++ lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ lib/graph/version.map | 17 + lib/node/ip4_rewrite.c | 476 ++++++++--- lib/node/ip4_rewrite_priv.h | 9 +- lib/node/node_private.h | 19 +- lib/node/rte_node_ip4_api.h | 3 + 14 files changed, 2325 insertions(+), 97 deletions(-) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH 1/3] graph: add feature arc support 2024-09-07 7:31 [RFC PATCH 0/3] add feature arc in rte_graph Nitin Saxena @ 2024-09-07 7:31 ` Nitin Saxena 2024-09-11 4:41 ` Kiran Kumar Kokkilagadda 2024-09-07 7:31 ` [RFC PATCH 2/3] graph: add feature arc option in graph create Nitin Saxena ` (3 subsequent siblings) 4 siblings, 1 reply; 55+ messages in thread From: Nitin Saxena @ 2024-09-07 7:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena add feature arc to allow dynamic steering of packets across graph nodes based on protocol features enabled on incoming or outgoing interface Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ lib/graph/meson.build | 2 + lib/graph/rte_graph_feature_arc.h | 373 +++++++++ lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ lib/graph/version.map | 17 + 5 files changed, 1899 insertions(+) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c new file mode 100644 index 0000000000..3b05bac137 --- /dev/null +++ b/lib/graph/graph_feature_arc.c @@ -0,0 +1,959 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "graph_private.h" +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> + +#define __RTE_GRAPH_FEATURE_ARC_MAX 32 + +#define ARC_PASSIVE_LIST(arc) (arc->active_feature_list ^ 0x1) + +#define rte_graph_uint_cast(x) ((unsigned int)x) +#define feat_dbg graph_err + +rte_graph_feature_arc_main_t *__feature_arc_main; + +/* Make sure fast path cache line is compact */ +_Static_assert((offsetof(struct rte_graph_feature_arc, slow_path_variables) + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) + <= RTE_CACHE_LINE_SIZE); + + +static int +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + + if (!feat_name) + return -1; + + if (slot) + *slot = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, RTE_GRAPH_NAMESIZE)) { + if (ffinfo) + *ffinfo = finfo; + return 0; + } + if (slot) + (*slot)++; + } + return -1; +} + +static int +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t feature_index, + struct rte_graph_feature_node_list **ppfinfo) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + if (!ppfinfo) + return -1; + + *ppfinfo = NULL; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + if (index == feature_index) { + if (finfo->node_index == feature_index) + return -1; + *ppfinfo = finfo; + } + index++; + } + if (feature_index && (index >= feature_index)) + return -1; + + return 0; +} + +static void +prepare_feature_arc(struct rte_graph_feature_arc *arc) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + finfo->node_index = index; + index++; + } +} + +static int +feature_arc_lookup(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __feature_arc_main; + uint32_t iter; + + if (!__feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (arc == (rte_graph_feature_arc_get(dm->feature_arcs[iter]))) + return 0; + } + return -1; +} + +static int +get_existing_edge(const char *arc_name, struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t count, i; + + RTE_SET_USED(arc_name); + + count = rte_node_edge_get(parent_node->id, NULL); + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + feat_dbg("%s: Edge exists [%s[%u]: \"%s\"]", arc_name, + parent_node->name, i, child_node->name); + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +static int +connect_graph_nodes(struct rte_node_register *parent_node, struct rte_node_register *child_node, + rte_edge_t *_edge, char *arc_name) +{ + const char *next_node = NULL; + rte_edge_t edge; + + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { + feat_dbg("%s: add_feature: Edge reused [%s[%u]: \"%s\"]", arc_name, + parent_node->name, edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; + } + + /* Node to be added */ + next_node = child_node->name; + + edge = rte_node_edge_update(parent_node->id, RTE_EDGE_ID_INVALID, &next_node, 1); + + if (edge == RTE_EDGE_ID_INVALID) { + graph_err("edge invalid"); + return -1; + } + edge = rte_node_edge_count(parent_node->id) - 1; + + feat_dbg("%s: add_feature: edge added [%s[%u]: \"%s\"]", arc_name, parent_node->name, edge, + child_node->name); + + if (_edge) + *_edge = edge; + + return 0; +} + +static int +feature_arc_init(rte_graph_feature_arc_main_t **pfl, uint32_t max_feature_arcs) +{ + rte_graph_feature_arc_main_t *pm = NULL; + uint32_t i; + size_t sz; + + if (!pfl) + return -1; + + sz = sizeof(rte_graph_feature_arc_main_t) + + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); + + pm = malloc(sz); + if (!pm) + return -1; + + memset(pm, 0, sz); + + for (i = 0; i < max_feature_arcs; i++) + pm->feature_arcs[i] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + pm->max_feature_arcs = max_feature_arcs; + + *pfl = pm; + + return 0; +} + +int +rte_graph_feature_arc_init(int max_feature_arcs) +{ + if (!max_feature_arcs) + return -1; + + if (__feature_arc_main) + return -1; + + return feature_arc_init(&__feature_arc_main, max_feature_arcs); +} + +static void +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t list_index) +{ + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + uint32_t i, j; + + list = arc->feature_list[list_index]; + feat = arc->features[list_index]; + + /*Initialize variables*/ + memset(feat, 0, arc->feature_size); + memset(list, 0, arc->feature_list_size); + + /* Initialize feature and feature_data */ + for (i = 0; i < arc->max_features; i++) { + feat = __rte_graph_feature_get(arc, i, list_index); + feat->this_feature_index = i; + + for (j = 0; j < arc->max_indexes; j++) { + fdata = rte_graph_feature_data_get(arc, feat, j); + fdata->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + fdata->next_edge = UINT16_MAX; + fdata->user_data = UINT32_MAX; + } + } + + for (i = 0; i < arc->max_indexes; i++) + list->first_enabled_feature_by_index[i] = RTE_GRAPH_FEATURE_INVALID; +} + +static int +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char *flist_name, + rte_graph_feature_list_t **pplist, + struct rte_graph_feature **ppfeature, uint32_t list_index) +{ + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + size_t list_size, feat_size, fdata_size; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + + list_size = sizeof(list->first_enabled_feature_by_index[0]) * arc->max_indexes; + + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); + if (!list) + return -ENOMEM; + + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); + + /* Let one feature capture complete cache lines */ + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + fdata_size, + RTE_CACHE_LINE_SIZE); + + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, "feat"); + + feat = rte_malloc(fname, feat_size * arc->max_features, RTE_CACHE_LINE_SIZE); + if (!feat) { + rte_free(list); + return -ENOMEM; + } + arc->feature_size = feat_size; + arc->feature_data_size = fdata_size; + arc->feature_list_size = list_size; + + /* Initialize list */ + list->indexed_by_features = feat; + *pplist = list; + *ppfeature = feat; + + feature_arc_list_reset(arc, list_index); + + return 0; +} + +static void +feature_arc_list_destroy(rte_graph_feature_list_t *list) +{ + rte_free(list->indexed_by_features); + rte_free(list); +} + +int +rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, rte_graph_feature_arc_t *_arc) +{ + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + rte_graph_feature_arc_main_t *dfm = NULL; + struct rte_graph_feature_arc *arc = NULL; + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature *df = NULL; + uint32_t iter, j, arc_index; + size_t sz; + + if (!_arc) + return -1; + + if (max_features < 2) + return -1; + + if (!start_node) + return -1; + + if (!feature_arc_name) + return -1; + + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) { + graph_err("Invalid max features: %u", max_features); + return -1; + } + + /* + * Application hasn't called rte_graph_feature_arc_init(). Initialize with + * default values + */ + if (!__feature_arc_main) { + if (rte_graph_feature_arc_init((int)__RTE_GRAPH_FEATURE_ARC_MAX) < 0) { + graph_err("rte_graph_feature_arc_init() failed"); + return -1; + } + } + + dfm = __feature_arc_main; + + /* threshold check */ + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) { + graph_err("max threshold for num_feature_arcs: %d reached", + dfm->max_feature_arcs - 1); + return -1; + } + /* Find the free slot for feature arc */ + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { + if (dfm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + break; + } + arc_index = iter; + + if (arc_index >= dfm->max_feature_arcs) { + graph_err("No free slot found for num_feature_arc"); + return -1; + } + + /* This should not happen */ + RTE_VERIFY(dfm->feature_arcs[arc_index] == RTE_GRAPH_FEATURE_ARC_INITIALIZER); + + /* size of feature arc + feature_bit_mask_by_index */ + sz = sizeof(*arc) + (sizeof(uint64_t) * max_indexes); + + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); + + if (!arc) { + graph_err("malloc failed for feature_arc_create()"); + return -1; + } + + memset(arc, 0, sz); + + /* Initialize rte_graph port group fixed variables */ + STAILQ_INIT(&arc->all_features); + strncpy(arc->feature_arc_name, feature_arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + arc->feature_arc_main = (void *)dfm; + arc->start_node = start_node; + arc->max_features = max_features; + arc->max_indexes = max_indexes; + + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc->features[0], 0) < 0) { + rte_free(arc); + graph_err("feature_arc_list_init(0) failed"); + return -1; + } + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc->features[1], 1) < 0) { + feature_arc_list_destroy(arc->feature_list[0]); + graph_err("feature_arc_list_init(1) failed"); + return -1; + } + + for (iter = 0; iter < arc->max_features; iter++) { + df = rte_graph_feature_get(arc, iter); + for (j = 0; j < arc->max_indexes; j++) { + gfd = rte_graph_feature_data_get(arc, df, j); + gfd->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + } + } + arc->feature_arc_index = arc_index; + dfm->feature_arcs[arc->feature_arc_index] = (rte_graph_feature_arc_t)arc; + dfm->num_feature_arcs++; + + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + + return 0; +} + +int +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *after_feature, const char *before_feature) +{ + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo = NULL; + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + uint32_t slot, add_flag; + rte_edge_t edge = -1; + + RTE_VERIFY(arc->feature_arc_main == __feature_arc_main); + + if (feature_arc_lookup(_arc)) { + graph_err("invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (arc->runtime_enabled_features) { + graph_err("adding features after enabling any one of them is not supported"); + return -1; + } + + if ((after_feature != NULL) && (before_feature != NULL) && + (after_feature == before_feature)) { + graph_err("after_feature and before_feature are same '%s:%s]", after_feature, + before_feature); + return -1; + } + + if (!feature_node) { + graph_err("feature_node: %p invalid", feature_node); + return -1; + } + + arc = rte_graph_feature_arc_get(_arc); + + if (feature_node->id == RTE_NODE_ID_INVALID) { + graph_err("Invalid node: %s", feature_node->name); + return -1; + } + + if (!feature_lookup(arc, feature_node->name, &finfo, &slot)) { + graph_err("%s feature already added", feature_node->name); + return -1; + } + + if (slot >= RTE_GRAPH_FEATURE_MAX_PER_ARC) { + graph_err("Max slot %u reached for feature addition", slot); + return -1; + } + + if (strstr(feature_node->name, arc->start_node->name)) { + graph_err("Feature %s cannot point to itself: %s", feature_node->name, + arc->start_node->name); + return -1; + } + + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc->feature_arc_name)) { + graph_err("unable to connect %s -> %s", arc->start_node->name, feature_node->name); + return -1; + } + + finfo = malloc(sizeof(*finfo)); + if (!finfo) + return -1; + + memset(finfo, 0, sizeof(*finfo)); + + finfo->feature_arc = (void *)arc; + finfo->feature_node = feature_node; + finfo->edge_to_this_feature = edge; + + /* Check for before and after constraints */ + if (before_feature) { + /* before_feature sanity */ + if (feature_lookup(arc, before_feature, &before_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid before feature name: %s", before_feature); + + if (!before_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "before_feature %s does not exist", before_feature); + + /* + * Starting from 0 to before_feature, continue connecting edges + */ + add_flag = 1; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + /* + * As soon as we see before_feature. stop adding edges + */ + if (!strncmp(temp->feature_node->name, before_feature, + RTE_GRAPH_NAMESIZE)) + if (!connect_graph_nodes(finfo->feature_node, temp->feature_node, + &edge, arc->feature_arc_name)) + add_flag = 0; + + if (add_flag) + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + } + } + + if (after_feature) { + if (feature_lookup(arc, after_feature, &after_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid after feature_name %s", after_feature); + + if (!after_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "after_feature %s does not exist", after_feature); + + /* Starting from after_feature to end continue connecting edges */ + add_flag = 0; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + /* We have already seen after_feature now */ + if (add_flag) + /* Add all features as next node to current feature*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, NULL, + arc->feature_arc_name); + + /* as soon as we see after_feature. start adding edges + * from next iteration + */ + if (!strncmp(temp->feature_node->name, after_feature, RTE_GRAPH_NAMESIZE)) + /* connect after_feature to this feature */ + if (!connect_graph_nodes(temp->feature_node, finfo->feature_node, + &edge, arc->feature_arc_name)) + add_flag = 1; + } + + /* add feature next to after_feature */ + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, next_feature); + } else { + if (before_finfo) { + after_finfo = NULL; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (before_finfo == temp) { + if (after_finfo) + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, + finfo, next_feature); + else + STAILQ_INSERT_HEAD(&arc->all_features, finfo, + next_feature); + + return 0; + } + after_finfo = temp; + } + } else { + STAILQ_INSERT_TAIL(&arc->all_features, finfo, next_feature); + } + } + + return 0; + +finfo_free: + free(finfo); + + return -1; +} + +int +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot; + + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { + *feat = (rte_graph_feature_t) slot; + return 0; + } + + return -1; +} + +int +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int is_enable_disable) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature *gf = NULL; + uint32_t slot; + + /* validate _arc */ + if (arc->feature_arc_main != __feature_arc_main) { + graph_err("invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -EINVAL; + } + + /* validate index */ + if (index >= arc->max_indexes) { + graph_err("%s: Invalid provided index: %u >= %u configured", arc->feature_arc_name, + index, arc->max_indexes); + return -1; + } + + /* validate feature_name is already added or not */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) { + graph_err("%s: No feature %s added", arc->feature_arc_name, feature_name); + return -EINVAL; + } + + if (!finfo) { + graph_err("%s: No feature: %s found", arc->feature_arc_name, feature_name); + return -EINVAL; + } + + /* slot should be in valid range */ + if (slot >= arc->max_features) { + graph_err("%s/%s: Invalid free slot %u(max=%u) for feature", arc->feature_arc_name, + feature_name, slot, arc->max_features); + return -EINVAL; + } + + /* slot should be in range of 0 - 63 */ + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { + graph_err("%s/%s: Invalid slot: %u", arc->feature_arc_name, + feature_name, slot); + return -EINVAL; + } + + if (finfo->node_index != slot) { + graph_err("%s/%s: feature lookup slot mismatch with finfo index: %u and lookup slot: %u", + arc->feature_arc_name, feature_name, finfo->node_index, slot); + return -1; + } + + /* Get feature from active list */ + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(arc)); + if (gf->this_feature_index != slot) { + graph_err("%s: %s received feature_index: %u does not match with saved feature_index: %u", + arc->feature_arc_name, feature_name, slot, gf->this_feature_index); + return -1; + } + + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & + RTE_BIT64(slot))) { + graph_err("%s: %s already enabled on index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + if (!is_enable_disable && !arc->runtime_enabled_features) { + graph_err("%s: No feature enabled to disable", arc->feature_arc_name); + return -1; + } + + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot))) { + graph_err("%s: %s not enabled in bitmask for index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + return 0; +} + +static void +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t dest_list_index, + uint16_t src_list_index) +{ + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; + struct rte_graph_feature *sgf = NULL, *dgf = NULL; + uint32_t i, j; + + for (i = 0; i < arc->max_features; i++) { + sgf = __rte_graph_feature_get(arc, i, src_list_index); + dgf = __rte_graph_feature_get(arc, i, dest_list_index); + for (j = 0; j < arc->max_indexes; j++) { + sgfd = rte_graph_feature_data_get(arc, sgf, j); + dgfd = rte_graph_feature_data_get(arc, dgf, j); + dgfd->user_data = sgfd->user_data; + } + } +} + +static void +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t list_index) +{ + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; + rte_graph_feature_list_t *flist = NULL; + uint32_t fi, di, prev_fi; + uint64_t bitmask; + rte_edge_t edge; + + flist = arc->feature_list[list_index]; + + for (di = 0; di < arc->max_indexes; di++) { + bitmask = arc->feature_bit_mask_by_index[di]; + prev_fi = RTE_GRAPH_FEATURE_INVALID; + /* for each feature set for index, set fast path data */ + while (rte_bsf64_safe(bitmask, &fi)) { + gf = __rte_graph_feature_get(arc, fi, list_index); + gfd = rte_graph_feature_data_get(arc, gf, di); + feature_arc_node_info_lookup(arc, fi, &finfo); + + /* If previous feature_index was valid in last loop */ + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { + prev_gf = __rte_graph_feature_get(arc, prev_fi, list_index); + prev_gfd = rte_graph_feature_data_get(arc, prev_gf, di); + /* + * Get edge of previous feature node connecting to this feature node + */ + feature_arc_node_info_lookup(arc, prev_fi, &prev_finfo); + if (!get_existing_edge(arc->feature_arc_name, + prev_finfo->feature_node, + finfo->feature_node, &edge)) { + feat_dbg("[%s/%s(%2u)/idx:%2u]: %s[%u] = %s", + arc->feature_arc_name, + prev_finfo->feature_node->name, prev_fi, di, + prev_finfo->feature_node->name, + edge, finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* + * Fill current feature as next enabled + * feature to previous one + */ + prev_gfd->next_enabled_feature = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* On first feature edge of the node to be added */ + if (fi == rte_bsf64(arc->feature_bit_mask_by_index[di])) { + if (!get_existing_edge(arc->feature_arc_name, arc->start_node, + finfo->feature_node, + &edge)) { + feat_dbg("[%s/%s/%2u/idx:%2u]: 1st feat %s[%u] = %s", + arc->feature_arc_name, + arc->start_node->name, fi, di, + arc->start_node->name, edge, + finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* Set first feature set array for index*/ + flist->first_enabled_feature_by_index[di] = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* Clear current feature index */ + bitmask &= ~RTE_BIT64(fi); + } + } +} + +int +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const + char *feature_name, int32_t user_data) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_rt_list_t passive_list; + struct rte_graph_feature *gf = NULL; + uint64_t fp_bitmask; + uint32_t slot; + + if (rte_graph_feature_validate(_arc, index, feature_name, 1)) + return -1; + + /** This should not fail as validate() has passed */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) + RTE_VERIFY(0); + + if (!arc->runtime_enabled_features) + prepare_feature_arc(arc); + + passive_list = ARC_PASSIVE_LIST(arc); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + feat_dbg("%s/%s: Enabling feature on list: %u for index: %u at feature slot %u", + arc->feature_arc_name, feature_name, passive_list, index, slot); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); + + /* Set current user-data */ + gfd->user_data = user_data; + + /* Set bitmask in control path bitmask */ + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + refill_feature_fastpath_data(arc, passive_list); + + /* Set fast path enable bitmask */ + fp_bitmask = __atomic_load_n(&arc->feature_enable_bitmask[passive_list], __ATOMIC_RELAXED); + fp_bitmask |= RTE_BIT64(slot); + __atomic_store(&arc->feature_enable_bitmask[passive_list], &fp_bitmask, __ATOMIC_RELAXED); + + /* Slow path updates */ + arc->runtime_enabled_features++; + + /* Increase feature node info reference count */ + finfo->ref_count++; + + /* Store release semantics for active_list update */ + __atomic_store(&arc->active_feature_list, &passive_list, __ATOMIC_RELEASE); + + return 0; +} + +int +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + rte_graph_feature_rt_list_t passive_list; + struct rte_graph_feature *gf = NULL; + uint32_t slot; + + if (rte_graph_feature_validate(_arc, index, feature_name, 0)) + return -1; + + if (feature_lookup(arc, feature_name, &finfo, &slot)) + return -1; + + passive_list = ARC_PASSIVE_LIST(arc); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + feat_dbg("%s/%s: Disabling feature for index: %u at feature slot %u", arc->feature_arc_name, + feature_name, index, slot); + + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + + /* Set fast path enable bitmask */ + arc->feature_enable_bitmask[passive_list] &= ~(RTE_BIT64(slot)); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); + + /* Reset current user-data */ + gfd->user_data = ~0; + + refill_feature_fastpath_data(arc, passive_list); + + finfo->ref_count--; + arc->runtime_enabled_features--; + + /* Store release semantics for active_list update */ + __atomic_store(&arc->active_feature_list, &passive_list, __ATOMIC_RELEASE); + + return 0; +} + +int +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __feature_arc_main; + struct rte_graph_feature_node_list *node_info = NULL; + + while (!STAILQ_EMPTY(&arc->all_features)) { + node_info = STAILQ_FIRST(&arc->all_features); + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); + free(node_info); + } + feature_arc_list_destroy(arc->feature_list[0]); + feature_arc_list_destroy(arc->feature_list[1]); + rte_free(arc->features[0]); + rte_free(arc->features[1]); + + dm->feature_arcs[arc->feature_arc_index] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + rte_free(arc); + return 0; +} + +int +rte_graph_feature_arc_cleanup(void) +{ + rte_graph_feature_arc_main_t *dm = __feature_arc_main; + uint32_t iter; + + if (!__feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm->feature_arcs[iter]); + } + free(dm); + + __feature_arc_main = NULL; + + return 0; +} + +int +rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc) +{ + rte_graph_feature_arc_main_t *dm = __feature_arc_main; + struct rte_graph_feature_arc *arc = NULL; + uint32_t iter; + + if (!__feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); + + if (strstr(arc_name, arc->feature_arc_name)) { + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + return 0; + } + } + + return -1; +} + +int +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + return arc->runtime_enabled_features; +} + + diff --git a/lib/graph/meson.build b/lib/graph/meson.build index 0cb15442ab..d916176fb7 100644 --- a/lib/graph/meson.build +++ b/lib/graph/meson.build @@ -14,11 +14,13 @@ sources = files( 'graph_debug.c', 'graph_stats.c', 'graph_populate.c', + 'graph_feature_arc.c', 'graph_pcap.c', 'rte_graph_worker.c', 'rte_graph_model_mcore_dispatch.c', ) headers = files('rte_graph.h', 'rte_graph_worker.h') +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') indirect_headers += files( 'rte_graph_model_mcore_dispatch.h', 'rte_graph_model_rtc.h', diff --git a/lib/graph/rte_graph_feature_arc.h b/lib/graph/rte_graph_feature_arc.h new file mode 100644 index 0000000000..e3bf4eb73d --- /dev/null +++ b/lib/graph/rte_graph_feature_arc.h @@ -0,0 +1,373 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ +#define _RTE_GRAPH_FEATURE_ARC_H_ + +#include <assert.h> +#include <errno.h> +#include <signal.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <rte_common.h> +#include <rte_compat.h> +#include <rte_debug.h> +#include <rte_graph.h> +#include <rte_graph_worker.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @file + * + * rte_graph_feature_arc.h + * + * Define APIs and structures/variables with respect to feature arc + * + * - Feature arc(s) + * - Feature(s) + * + * A feature arc represents an ordered list of features/protocol-nodes at a + * given networking layer. Feature arc provides a high level abstraction to + * connect various *rte_graph* nodes, designated as *feature nodes*, and + * allowing steering of packets across these feature nodes fast path processing + * in a generic manner. In a typical network stack, often a protocol or feature + * must be first enabled on a given interface, before any packet is steered + * towards it for feature processing. For eg: incoming IPv4 packets are sent to + * routing sub-system only after a valid IPv4 address is assigned to the + * received interface. In other words, often packets needs to be steered across + * features not based on the packet content but based on whether a feature is + * enable or disable on a given incoming/outgoing interface. Feature arc + * provides mechanism to enable/disable feature(s) on each interface at runtime + * and allow seamless packet steering across runtime enabled feature nodes in + * fast path. + * + * Feature arc also provides a way to steer packets from standard nodes to + * custom/user-defined *feature nodes* without any change in standard node's + * fast path functions + * + * On a given interface multiple feature(s) might be enabled in a particular + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" + * features may be enabled on "eth0" interface in "L3-output" feature arc. + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" + * interface in same "L3-output" feature arc. + * + * When multiple features are present in a given feature arc, its imperative + * to allow each feature processing in a particular sequential order. For + * instance, in "L3-input" feature arc it may be required to run "IPsec + * input" feature first, for packet decryption, before "ip-lookup". So a + * sequential order must be maintained among features present in a feature arc. + * + * Features are enabled/disabled multiple times at runtime to some or all + * available interfaces present in the system. Features can be enabled/disabled + * even after @b rte_graph_create() is called. Enable/disabling features on one + * interface is independent of other interface. + * + * A given feature might consume packet (if it's configured to consume) or may + * forward it to next enabled feature. For instance, "IPsec input" feature may + * consume/drop all packets with "Protect" policy action while all packets with + * policy action as "Bypass" may be forwarded to next enabled feature (with in + * same feature arc) + * + * This library facilitates rte graph based applications to steer packets in + * fast path to different feature nodes with-in a feature arc and support all + * functionalities described above + * + * In order to use feature-arc APIs, applications needs to do following in + * control path: + * - Initialize feature arc library via rte_graph_feature_arc_init() + * - Create feature arc via rte_graph_feature_arc_create() + * - *Before calling rte_graph_create()*, features must be added to feature-arc + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding + * features in a sequential order with "runs_after" and "runs_before" + * constraints. + * - Post rte_graph_create(), features can be enabled/disabled at runtime on + * any interface via rte_graph_feature_enable()/rte_graph_feature_disable() + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() + * + * In fast path, APIs are provided to steer packets towards feature path from + * - start_node (provided as an argument to rte_graph_feature_arc_create()) + * - feature nodes (which are added via rte_graph_feature_add()) + * + * For typical steering of packets across feature nodes, application required + * to know "rte_edges" which are saved in feature data object. Feature data + * object is unique for every interface per feature with in a feature arc. + * + * When steering packets from start_node to feature node: + * - rte_graph_feature_arc_first_feature_get() provides first enabled feature. + * - Next rte_edge from start_node to first enabled feature can be obtained via + * rte_graph_feature_arc_feature_set() + * + * rte_mbuf can carry [current feature, index] from start_node of an arc to other + * feature nodes + * + * In feature node, application can get 32-bit user_data + * via_rte_graph_feature_user_data_get() which is provided in + * rte_graph_feature_enable(). User data can hold feature specific cookie like + * IPsec policy database index (if more than one are supported) + * + * If feature node is not consuming packet, next enabled feature and next + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() + * + * It is application responsibility to ensure that at-least *last feature*(or sink + * feature) must be enabled from where packet can exit feature-arc path, if + * *NO* intermediate feature is consuming the packet and it has reached till + * the end of feature arc path + * + * Synchronization among cores + * --------------------------- + * Subsequent calls to rte_graph_feature_enable() is allowed while worker cores + * are processing in rte_graph_walk() loop. However, for + * rte_graph_feature_disable() application must use RCU based synchronization + */ + +/**< Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER ((rte_graph_feature_arc_t)UINT64_MAX) + +/** Max number of features supported in a given feature arc */ +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 + +/** Length of feature arc name */ +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE + +/** @internal */ +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) + +/**< Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_INVALID rte_graph_feature_cast(UINT8_MAX) + +/** rte_graph feature arc object */ +typedef uint64_t rte_graph_feature_arc_t; + +/** rte_graph feature object */ +typedef uint8_t rte_graph_feature_t; + +/** runtime active feature list index with in feature arc*/ +typedef uint8_t rte_graph_feature_rt_list_t; + +/** per feature arc monotonically increasing counter to synchronize fast path APIs */ +typedef uint16_t rte_graph_feature_counter_t; + +/** + * Initialize feature arc subsystem + * + * @param max_feature_arcs + * Maximum number of feature arcs required to be supported + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_init(int max_feature_arcs); + +/** + * Create a feature arc + * + * @param feature_arc_name + * Feature arc name with max length of @ref RTE_GRAPH_FEATURE_ARC_NAMELEN + * @param max_features + * Maximum number of features to be supported in this feature arc + * @param max_indexes + * Maximum number of interfaces/ports/indexes to be supported + * @param start_node + * Base node where this feature arc's features are checked in fast path + * @param[out] _arc + * Feature arc object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, + rte_graph_feature_arc_t *_arc); + +/** + * Get feature arc object with name + * + * @param arc_name + * Feature arc name provided to successful @ref rte_graph_feature_arc_create + * @param[out] _arc + * Feature arc object returned + * + * @return + * 0: Success + * <0: Failure. + */ +__rte_experimental +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc); + +/** + * Add a feature to already created feature arc. For instance + * + * 1. Add first feature node: "ipv4-input" to input arc + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); + * + * 2. Add "ipsec-input" feature node after "ipv4-input" node + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", NULL); + * + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" node + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", NULL, "ipv4-input"); + * + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4-input", "ipsec-input"); + * + * @param _arc + * Feature arc handle returned from @ref rte_graph_feature_arc_create() + * @param feature_node + * Graph node representing feature. On success, feature_node is next_node of + * feature_arc->start_node + * @param runs_after + * Add this feature_node after already added "runs_after". Creates + * start_node -> runs_after -> this_feature sequence + * @param runs_before + * Add this feature_node before already added "runs_before". Creates + * start_node -> this_feature -> runs_before sequence + * + * <I> Must be called before rte_graph_create() </I> + * <I> rte_graph_feature_add() is not allowed after call to + * rte_graph_feature_enable() so all features must be added before they can be + * enabled </I> + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *runs_after, const char *runs_before); + +/** + * Enable feature within a feature arc + * + * Must be called after @b rte_graph_create(). + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param user_data + * Application specific data which is retrieved in fast path + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int32_t user_data); + +/** + * Validate whether subsequent enable/disable feature would succeed or not. + * API is thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param is_enable_disable + * If 1, validate whether subsequent @ref rte_graph_feature_enable would pass or not + * If 0, validate whether subsequent @ref rte_graph_feature_disable would pass or not + * + * @return + * 0: Subsequent enable/disable API would pass + * <0: Subsequent enable/disable API would not pass + */ +__rte_experimental +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, int is_enable_disable); + +/** + * Disable already enabled feature within a feature arc + * + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name); + +/** + * Get rte_graph_feature_t object from feature name + * + * @param arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param feature_name + * Feature name provided to @ref rte_graph_feature_add + * @param[out] feature + * Feature object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feature); + +/** + * Delete feature_arc object + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); + +/** + * Cleanup all feature arcs + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_cleanup(void); + +/** + * Slow path API to know how many features are currently enabled within a featur-arc + * + * @param _arc + * Feature arc object + * + * @return: Number of enabled features + */ +__rte_experimental +int rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc); +#ifdef __cplusplus +} +#endif + +#endif diff --git a/lib/graph/rte_graph_feature_arc_worker.h b/lib/graph/rte_graph_feature_arc_worker.h new file mode 100644 index 0000000000..6019d74853 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc_worker.h @@ -0,0 +1,548 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ + +#include <stddef.h> +#include <rte_graph_feature_arc.h> +#include <rte_bitops.h> + +/** + * @file + * + * rte_graph_feature_arc_worker.h + * + * Defines fast path structure + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/** @internal + * + * Slow path feature node info list + */ +struct rte_graph_feature_node_list { + /** Next feature */ + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; + + /** node representing feature */ + struct rte_node_register *feature_node; + + /** How many indexes/interfaces using this feature */ + int32_t ref_count; + + /* node_index in list (after feature_enable())*/ + uint32_t node_index; + + /** Back pointer to feature arc */ + void *feature_arc; + + /** rte_edge_t to this feature node from feature_arc->start_node */ + rte_edge_t edge_to_this_feature; +}; + +/** + * Fast path holding rte_edge_t and next enabled feature for an feature + */ +typedef struct __rte_packed rte_graph_feature_data { + /* next node to which current mbuf should go*/ + rte_edge_t next_edge; + + /* next enabled feature on this arc for current index */ + union { + uint16_t reserved; + struct { + rte_graph_feature_t next_enabled_feature; + }; + }; + + /* user_data */ + int32_t user_data; +} rte_graph_feature_data_t; + +/** + * Fast path feature structure. Holds re_graph_feature_data_t per index + */ +struct __rte_cache_aligned rte_graph_feature { + uint16_t this_feature_index; + + /* Array of size arc->feature_data_size + * [data-index-0][data-index-1]... + * Each index of size: sizeof(rte_graph_feature_data_t) + */ + uint8_t feature_data_by_index[]; +}; + +/** + * fast path cache aligned feature list holding all features + * There are two feature lists: active, passive + * + * Fast APIs works on active list while control plane updates passive list + * A atomic update to arc->active_feature_list is done to switch between active + * and passive + */ +typedef struct __rte_cache_aligned rte_graph_feature_list { + /** + * fast path array holding per_feature data. + * Duplicate entry as feature-arc also hold this pointer + * arc->features[] + * + *<-------------feature-0 ---------><CEIL><---------feature-1 -------------->... + *[index-0][index-1]...[max_index-1] [index-0][index-1] ...[max_index-1]... + */ + struct rte_graph_feature *indexed_by_features; + /* + * fast path array holding first enabled feature per index + * (Required in start_node. In non start_node, mbuf can hold next enabled + * feature) + */ + rte_graph_feature_t first_enabled_feature_by_index[]; +} rte_graph_feature_list_t; + +/** + * rte_graph feature arc object + * + * A feature-arc can only hold RTE_GRAPH_FEATURE_MAX_PER_ARC features but no + * limit to interface index + * + * Representing a feature arc holding all features which are enabled/disabled + * on any interfaces + */ +struct __rte_cache_aligned rte_graph_feature_arc { + /* First 64B is fast path variables */ + RTE_MARKER fast_path_variables; + + /** runtime active feature list */ + rte_graph_feature_rt_list_t active_feature_list; + + /* Actual Size of feature_list0 */ + uint16_t feature_list_size; + + /** + * Size each feature in fastpath. + * sizeof(arc->active_list->indexed_by_feature[0]) + */ + uint16_t feature_size; + + /* Size of arc->max_index * sizeof(rte_graph_feature_data_t) */ + uint16_t feature_data_size; + + /** + * Fast path bitmask indicating if a feature is enabled or not Number + * of bits: RTE_GRAPH_FEATURE_MAX_PER_ARC + */ + uint64_t feature_enable_bitmask[2]; + rte_graph_feature_list_t *feature_list[2]; + struct rte_graph_feature *features[2]; + + /** index in feature_arc_main */ + uint16_t feature_arc_index; + + uint16_t reserved[3]; + + /** Slow path variables follows*/ + RTE_MARKER slow_path_variables; + + /** feature arc name */ + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** All feature lists */ + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; + + uint32_t runtime_enabled_features; + + /** Back pointer to feature_arc_main */ + void *feature_arc_main; + + /* start_node */ + struct rte_node_register *start_node; + + /* maximum number of features supported by this arc */ + uint32_t max_features; + + /* maximum number of index supported by this arc */ + uint32_t max_indexes; + + /* Slow path bit mask per feature per index */ + uint64_t feature_bit_mask_by_index[]; +}; + +/** Feature arc main */ +typedef struct feature_arc_main { + /** number of feature arcs created by application */ + uint32_t num_feature_arcs; + + /** max features arcs allowed */ + uint32_t max_feature_arcs; + + /** feature arcs */ + rte_graph_feature_arc_t feature_arcs[]; +} rte_graph_feature_arc_main_t; + +/** @internal Get feature arc pointer from object */ +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc *)arc) + +extern rte_graph_feature_arc_main_t *__feature_arc_main; + +/** + * API to know if feature is valid or not + */ + +static __rte_always_inline int +rte_graph_feature_is_valid(rte_graph_feature_t feature) +{ + return (feature != RTE_GRAPH_FEATURE_INVALID); +} + +/** + * Get rte_graph_feature object with no checks + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * @param feature_list + * active feature list retrieved from rte_graph_feature_arc_has_any_feature() + * or rte_graph_feature_arc_has_feature() + * + * @return + * Internal feature object. + */ +static __rte_always_inline struct rte_graph_feature * +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature, + const rte_graph_feature_rt_list_t feature_list) +{ + return ((struct rte_graph_feature *)((uint8_t *)(arc->features[feature_list] + + (feature * arc->feature_size)))); +} + +/** + * Get rte_graph_feature object for a given interface/index from feature arc + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * + * @return + * Internal feature object. + */ +static __rte_always_inline struct rte_graph_feature * +rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature) +{ + RTE_VERIFY(feature < arc->max_features); + + if (likely(rte_graph_feature_is_valid(feature))) + return __rte_graph_feature_get(arc, feature, arc->active_feature_list); + + return NULL; +} + +static __rte_always_inline rte_graph_feature_data_t * +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + RTE_SET_USED(arc); + return ((rte_graph_feature_data_t *)(feature->feature_data_by_index + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Get rte_graph feature data object for a index in feature + * + * @param arc + * feature arc + * @param feature + * Pointer to feature object + * @param index + * Index of feature maintained in slow path linked list + * + * @return + * Valid feature data + */ +static __rte_always_inline rte_graph_feature_data_t * +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + if (likely(index < arc->max_indexes)) + return __rte_graph_feature_data_get(arc, feature, index); + + RTE_VERIFY(0); +} + +/** + * Fast path API to check if any feature enabled on a feature arc + * Typically from arc->start_node process function + * + * @param arc + * Feature arc object + * @param[out] plist + * Pointer to runtime active feature list which needs to be provided to other + * fast path APIs + * + * @return + * 0: If no feature enabled + * Non-Zero: Bitmask of features enabled. plist is valid + * + */ +static __rte_always_inline uint64_t +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_rt_list_t *plist) +{ + *plist = __atomic_load_n(&arc->active_feature_list, __ATOMIC_RELAXED); + + return (__atomic_load_n(arc->feature_enable_bitmask + (uint8_t)*plist, + __ATOMIC_RELAXED)); +} + +/** + * Fast path API to check if provided feature is enabled on any interface/index + * or not + * + * @param arc + * Feature arc object + * @param feature + * Input rte_graph_feature_t that needs to be checked + * @param[out] plist + * Returns active list to caller which needs to be provided to other fast path + * APIs + * + * @return + * 1: If feature is enabled in arc + * 0: If feature is not enabled in arc + */ +static __rte_always_inline int +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_t feature, + rte_graph_feature_rt_list_t *plist) +{ + uint64_t bitmask = RTE_BIT64(feature); + + *plist = __atomic_load_n(&arc->active_feature_list, __ATOMIC_RELAXED); + + return (bitmask & __atomic_load_n(arc->feature_enable_bitmask + (uint8_t)*plist, + __ATOMIC_RELAXED)); +} + +/** + * Prefetch feature arc fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + */ +static __rte_always_inline void +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) +{ + rte_prefetch0((void *)&arc->fast_path_variables); +} + +/** + * Prefetch feature related fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object + */ +static __rte_always_inline void +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature) +{ + /* feature cache line */ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, list)); +} + +/** + * Prefetch feature data upfront. Perform sanity + * + * @param _arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object returned from @ref + * rte_graph_feature_arc_first_feature_get() + * @param index + * Interface/index + */ +static __rte_always_inline void +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, uint32_t index) +{ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)((uint8_t *)arc->features[list] + + offsetof(struct rte_graph_feature, feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Fast path API to get first enabled feature on interface index + * Typically required in arc->start_node so that from returned feature, + * feature-data can be retrieved to steer packets + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. + * + * @return + * 0. Success. feature field is valid + * 1. Failure. feature field is invalid + * + */ +static __rte_always_inline int +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + + *feature = feature_list->first_enabled_feature_by_index[index]; + + return rte_graph_feature_is_valid(*feature); +} + +/** + * Fast path API to get next enabled feature on interface index with provided + * input feature + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * @param index + * Interface Index + * @param[in][out] feature + * Pointer to rte_graph_feature_t. Input feature set to next enabled feature + * after success return + * @param[out] next_edge + * Edge from current feature to next feature. Valid only if next feature is valid + * + * @return + * 0. Success. next enabled feature is valid. + * 1. Failure. next enabled feature is invalid + */ +static __rte_always_inline int +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature, + rte_edge_t *next_edge) +{ + rte_graph_feature_data_t *feature_data = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(*feature))) { + f = __rte_graph_feature_get(arc, *feature, list); + feature_data = rte_graph_feature_data_get(arc, f, index); + *feature = feature_data->next_enabled_feature; + *next_edge = feature_data->next_edge; + return (*feature == RTE_GRAPH_FEATURE_INVALID); + } + + return 1; +} + +/** + * Set fields with respect to first enabled feature in an arc and return edge + * Typically returned feature and interface index must be saved in rte_mbuf + * structure to pass this information to next feature node + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param index + * Index (of interface) + * @param[out] gf + * Pointer to rte_graph_feature_t. Valid if API returns Success + * @param[out] edge + * Edge to steer packet from arc->start_node to first enabled feature. Valid + * only if API returns Success + * + * @return + * 0: If valid feature is set by API + * 1: If valid feature is NOT set by API + */ +static __rte_always_inline rte_graph_feature_t +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *gf, + rte_edge_t *edge) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + struct rte_graph_feature_data *feature_data = NULL; + struct rte_graph_feature *feature = NULL; + rte_graph_feature_t f; + + /* reset */ + *gf = RTE_GRAPH_FEATURE_INVALID; + f = feature_list->first_enabled_feature_by_index[index]; + + if (unlikely(rte_graph_feature_is_valid(f))) { + feature = __rte_graph_feature_get(arc, f, list); + feature_data = rte_graph_feature_data_get(arc, feature, index); + *gf = f; + *edge = feature_data->next_edge; + return 0; + } + + return 1; +} + +/** + * Get user data corresponding to current feature set by application in + * rte_graph_feature_enable() + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Feature index + * @param index + * Interface index + * + * @return + * UINT32_MAX: Failure + * Valid user data: Success + */ +static __rte_always_inline uint32_t +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, + uint32_t index) +{ + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(feature))) { + f = __rte_graph_feature_get(arc, feature, list); + fdata = rte_graph_feature_data_get(arc, f, index); + return fdata->user_data; + } + + return UINT32_MAX; +} +#ifdef __cplusplus +} +#endif +#endif diff --git a/lib/graph/version.map b/lib/graph/version.map index 2c83425ddc..82b2469fba 100644 --- a/lib/graph/version.map +++ b/lib/graph/version.map @@ -52,3 +52,20 @@ DPDK_25 { local: *; }; + +EXPERIMENTAL { + global: + + # added in 24.11 + rte_graph_feature_arc_init; + rte_graph_feature_arc_create; + rte_graph_feature_arc_lookup_by_name; + rte_graph_feature_add; + rte_graph_feature_enable; + rte_graph_feature_validate; + rte_graph_feature_disable; + rte_graph_feature_lookup; + rte_graph_feature_arc_destroy; + rte_graph_feature_arc_cleanup; + rte_graph_feature_arc_num_enabled_features; +}; -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* RE: [RFC PATCH 1/3] graph: add feature arc support 2024-09-07 7:31 ` [RFC PATCH 1/3] graph: add feature arc support Nitin Saxena @ 2024-09-11 4:41 ` Kiran Kumar Kokkilagadda 2024-10-10 4:42 ` Nitin Saxena 0 siblings, 1 reply; 55+ messages in thread From: Kiran Kumar Kokkilagadda @ 2024-09-11 4:41 UTC (permalink / raw) To: Nitin Saxena, Jerin Jacob, Nithin Kumar Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena > -----Original Message----- > From: Nitin Saxena <nsaxena@marvell.com> > Sent: Saturday, September 7, 2024 1:01 PM > To: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda > <kirankumark@marvell.com>; Nithin Kumar Dabilpuram > <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com> > Cc: dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com> > Subject: [RFC PATCH 1/3] graph: add feature arc support > > add feature arc to allow dynamic steering of packets across graph nodes > based on protocol features enabled on incoming or outgoing interface > > Signed-off-by: Nitin Saxena <nsaxena@marvell.com> > --- > lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ > lib/graph/meson.build | 2 + > lib/graph/rte_graph_feature_arc.h | 373 +++++++++ > lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ > lib/graph/version.map | 17 + > 5 files changed, 1899 insertions(+) > create mode 100644 lib/graph/graph_feature_arc.c > create mode 100644 lib/graph/rte_graph_feature_arc.h > create mode 100644 lib/graph/rte_graph_feature_arc_worker.h > > diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c > new file mode 100644 > index 0000000000..3b05bac137 > --- /dev/null > +++ b/lib/graph/graph_feature_arc.c > @@ -0,0 +1,959 @@ > +/* SPDX-License-Identifier: BSD-3-Clause > + * Copyright(C) 2024 Marvell International Ltd. > + */ > + > +#include "graph_private.h" > +#include <rte_graph_feature_arc_worker.h> > +#include <rte_malloc.h> > + > +#define __RTE_GRAPH_FEATURE_ARC_MAX 32 > + > +#define ARC_PASSIVE_LIST(arc) (arc->active_feature_list ^ 0x1) > + > +#define rte_graph_uint_cast(x) ((unsigned int)x) > +#define feat_dbg graph_err > + > +rte_graph_feature_arc_main_t *__feature_arc_main; > + > +/* Make sure fast path cache line is compact */ > +_Static_assert((offsetof(struct rte_graph_feature_arc, slow_path_variables) > + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) > + <= RTE_CACHE_LINE_SIZE); > + > + > +static int > +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, > + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) > +{ > + struct rte_graph_feature_node_list *finfo = NULL; > + const char *name; > + > + if (!feat_name) > + return -1; > + > + if (slot) > + *slot = 0; > + > + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { > + RTE_VERIFY(finfo->feature_arc == arc); > + name = rte_node_id_to_name(finfo->feature_node->id); > + if (!strncmp(name, feat_name, RTE_GRAPH_NAMESIZE)) { > + if (ffinfo) > + *ffinfo = finfo; > + return 0; > + } > + if (slot) > + (*slot)++; > + } > + return -1; > +} > + > +static int > +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t > feature_index, > + struct rte_graph_feature_node_list **ppfinfo) > +{ > + struct rte_graph_feature_node_list *finfo = NULL; > + uint32_t index = 0; > + > + if (!ppfinfo) > + return -1; > + > + *ppfinfo = NULL; > + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { > + if (index == feature_index) { > + if (finfo->node_index == feature_index) > + return -1; > + *ppfinfo = finfo; > + } > + index++; > + } > + if (feature_index && (index >= feature_index)) > + return -1; > + > + return 0; > +} > + > +static void > +prepare_feature_arc(struct rte_graph_feature_arc *arc) > +{ > + struct rte_graph_feature_node_list *finfo = NULL; > + uint32_t index = 0; > + > + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { > + finfo->node_index = index; > + index++; > + } > +} > + > +static int > +feature_arc_lookup(rte_graph_feature_arc_t _arc) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > + uint32_t iter; > + > + if (!__feature_arc_main) > + return -1; > + > + for (iter = 0; iter < dm->max_feature_arcs; iter++) { > + if (dm->feature_arcs[iter] == > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > + continue; > + > + if (arc == (rte_graph_feature_arc_get(dm- > >feature_arcs[iter]))) > + return 0; > + } > + return -1; > +} > + > +static int > +get_existing_edge(const char *arc_name, struct rte_node_register > *parent_node, > + struct rte_node_register *child_node, rte_edge_t *_edge) > +{ > + char **next_edges = NULL; > + uint32_t count, i; > + > + RTE_SET_USED(arc_name); > + > + count = rte_node_edge_get(parent_node->id, NULL); > + next_edges = malloc(count); > + > + if (!next_edges) > + return -1; > + > + count = rte_node_edge_get(parent_node->id, next_edges); > + for (i = 0; i < count; i++) { > + if (strstr(child_node->name, next_edges[i])) { > + feat_dbg("%s: Edge exists [%s[%u]: \"%s\"]", > arc_name, > + parent_node->name, i, child_node->name); > + if (_edge) > + *_edge = (rte_edge_t)i; > + > + free(next_edges); > + return 0; > + } > + } > + free(next_edges); > + > + return -1; > +} > + > +static int > +connect_graph_nodes(struct rte_node_register *parent_node, struct > rte_node_register *child_node, > + rte_edge_t *_edge, char *arc_name) > +{ > + const char *next_node = NULL; > + rte_edge_t edge; > + > + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { > + feat_dbg("%s: add_feature: Edge reused [%s[%u]: \"%s\"]", > arc_name, > + parent_node->name, edge, child_node->name); > + > + if (_edge) > + *_edge = edge; > + > + return 0; > + } > + > + /* Node to be added */ > + next_node = child_node->name; > + > + edge = rte_node_edge_update(parent_node->id, > RTE_EDGE_ID_INVALID, &next_node, 1); > + > + if (edge == RTE_EDGE_ID_INVALID) { > + graph_err("edge invalid"); > + return -1; > + } > + edge = rte_node_edge_count(parent_node->id) - 1; > + > + feat_dbg("%s: add_feature: edge added [%s[%u]: \"%s\"]", arc_name, > parent_node->name, edge, > + child_node->name); > + > + if (_edge) > + *_edge = edge; > + > + return 0; > +} > + > +static int > +feature_arc_init(rte_graph_feature_arc_main_t **pfl, uint32_t > max_feature_arcs) > +{ > + rte_graph_feature_arc_main_t *pm = NULL; > + uint32_t i; > + size_t sz; > + > + if (!pfl) > + return -1; > + > + sz = sizeof(rte_graph_feature_arc_main_t) + > + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); > + > + pm = malloc(sz); > + if (!pm) > + return -1; > + > + memset(pm, 0, sz); > + > + for (i = 0; i < max_feature_arcs; i++) > + pm->feature_arcs[i] = > RTE_GRAPH_FEATURE_ARC_INITIALIZER; > + > + pm->max_feature_arcs = max_feature_arcs; > + > + *pfl = pm; > + > + return 0; > +} > + > +int > +rte_graph_feature_arc_init(int max_feature_arcs) > +{ > + if (!max_feature_arcs) > + return -1; > + > + if (__feature_arc_main) > + return -1; > + > + return feature_arc_init(&__feature_arc_main, max_feature_arcs); > +} > + > +static void > +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t list_index) > +{ > + rte_graph_feature_data_t *fdata = NULL; > + rte_graph_feature_list_t *list = NULL; > + struct rte_graph_feature *feat = NULL; > + uint32_t i, j; > + > + list = arc->feature_list[list_index]; > + feat = arc->features[list_index]; > + > + /*Initialize variables*/ > + memset(feat, 0, arc->feature_size); > + memset(list, 0, arc->feature_list_size); > + > + /* Initialize feature and feature_data */ > + for (i = 0; i < arc->max_features; i++) { > + feat = __rte_graph_feature_get(arc, i, list_index); > + feat->this_feature_index = i; > + > + for (j = 0; j < arc->max_indexes; j++) { > + fdata = rte_graph_feature_data_get(arc, feat, j); > + fdata->next_enabled_feature = > RTE_GRAPH_FEATURE_INVALID; > + fdata->next_edge = UINT16_MAX; > + fdata->user_data = UINT32_MAX; > + } > + } > + > + for (i = 0; i < arc->max_indexes; i++) > + list->first_enabled_feature_by_index[i] = > RTE_GRAPH_FEATURE_INVALID; > +} > + > +static int > +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char > *flist_name, > + rte_graph_feature_list_t **pplist, > + struct rte_graph_feature **ppfeature, uint32_t > list_index) > +{ > + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; > + size_t list_size, feat_size, fdata_size; > + rte_graph_feature_list_t *list = NULL; > + struct rte_graph_feature *feat = NULL; > + > + list_size = sizeof(list->first_enabled_feature_by_index[0]) * arc- > >max_indexes; > + > + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); > + if (!list) > + return -ENOMEM; > + > + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); > + > + /* Let one feature capture complete cache lines */ > + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + > fdata_size, > + RTE_CACHE_LINE_SIZE); > + > + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, > "feat"); > + > + feat = rte_malloc(fname, feat_size * arc->max_features, > RTE_CACHE_LINE_SIZE); > + if (!feat) { > + rte_free(list); > + return -ENOMEM; > + } > + arc->feature_size = feat_size; > + arc->feature_data_size = fdata_size; > + arc->feature_list_size = list_size; > + > + /* Initialize list */ > + list->indexed_by_features = feat; > + *pplist = list; > + *ppfeature = feat; > + > + feature_arc_list_reset(arc, list_index); > + > + return 0; > +} > + > +static void > +feature_arc_list_destroy(rte_graph_feature_list_t *list) > +{ > + rte_free(list->indexed_by_features); Do you need to free individual rte_graph_feature here, that is allocated in arc_list_init? > + rte_free(list); > +} > + > +int > +rte_graph_feature_arc_create(const char *feature_arc_name, int > max_features, int max_indexes, > + struct rte_node_register *start_node, > rte_graph_feature_arc_t *_arc) > +{ > + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; > + rte_graph_feature_arc_main_t *dfm = NULL; > + struct rte_graph_feature_arc *arc = NULL; > + struct rte_graph_feature_data *gfd = NULL; > + struct rte_graph_feature *df = NULL; > + uint32_t iter, j, arc_index; > + size_t sz; > + > + if (!_arc) > + return -1; > + > + if (max_features < 2) > + return -1; > + > + if (!start_node) > + return -1; > + > + if (!feature_arc_name) > + return -1; > + > + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) { > + graph_err("Invalid max features: %u", max_features); > + return -1; > + } > + > + /* > + * Application hasn't called rte_graph_feature_arc_init(). Initialize with > + * default values > + */ > + if (!__feature_arc_main) { > + if > (rte_graph_feature_arc_init((int)__RTE_GRAPH_FEATURE_ARC_MAX) < 0) { > + graph_err("rte_graph_feature_arc_init() failed"); > + return -1; > + } > + } > + > + dfm = __feature_arc_main; > + > + /* threshold check */ > + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) { > + graph_err("max threshold for num_feature_arcs: %d > reached", > + dfm->max_feature_arcs - 1); > + return -1; > + } > + /* Find the free slot for feature arc */ > + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { > + if (dfm->feature_arcs[iter] == > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > + break; > + } > + arc_index = iter; > + > + if (arc_index >= dfm->max_feature_arcs) { > + graph_err("No free slot found for num_feature_arc"); > + return -1; > + } > + > + /* This should not happen */ > + RTE_VERIFY(dfm->feature_arcs[arc_index] == > RTE_GRAPH_FEATURE_ARC_INITIALIZER); > + > + /* size of feature arc + feature_bit_mask_by_index */ > + sz = sizeof(*arc) + (sizeof(uint64_t) * max_indexes); > + > + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); > + > + if (!arc) { > + graph_err("malloc failed for feature_arc_create()"); > + return -1; > + } > + > + memset(arc, 0, sz); > + > + /* Initialize rte_graph port group fixed variables */ > + STAILQ_INIT(&arc->all_features); > + strncpy(arc->feature_arc_name, feature_arc_name, > RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); > + arc->feature_arc_main = (void *)dfm; > + arc->start_node = start_node; > + arc->max_features = max_features; > + arc->max_indexes = max_indexes; > + > + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); > + > + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc- > >features[0], 0) < 0) { > + rte_free(arc); > + graph_err("feature_arc_list_init(0) failed"); > + return -1; > + } > + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); > + > + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc- > >features[1], 1) < 0) { > + feature_arc_list_destroy(arc->feature_list[0]); > + graph_err("feature_arc_list_init(1) failed"); > + return -1; > + } > + > + for (iter = 0; iter < arc->max_features; iter++) { > + df = rte_graph_feature_get(arc, iter); > + for (j = 0; j < arc->max_indexes; j++) { > + gfd = rte_graph_feature_data_get(arc, df, j); > + gfd->next_enabled_feature = > RTE_GRAPH_FEATURE_INVALID; > + } > + } > + arc->feature_arc_index = arc_index; > + dfm->feature_arcs[arc->feature_arc_index] = > (rte_graph_feature_arc_t)arc; > + dfm->num_feature_arcs++; > + > + if (_arc) > + *_arc = (rte_graph_feature_arc_t)arc; > + > + return 0; > +} > + > +int > +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct > rte_node_register *feature_node, > + const char *after_feature, const char *before_feature) > +{ > + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo > = NULL; > + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + uint32_t slot, add_flag; > + rte_edge_t edge = -1; > + > + RTE_VERIFY(arc->feature_arc_main == __feature_arc_main); > + > + if (feature_arc_lookup(_arc)) { > + graph_err("invalid feature arc: 0x%016" PRIx64, > (uint64_t)_arc); > + return -1; > + } > + > + if (arc->runtime_enabled_features) { > + graph_err("adding features after enabling any one of them is > not supported"); > + return -1; > + } > + > + if ((after_feature != NULL) && (before_feature != NULL) && > + (after_feature == before_feature)) { > + graph_err("after_feature and before_feature are same > '%s:%s]", after_feature, > + before_feature); > + return -1; > + } > + > + if (!feature_node) { > + graph_err("feature_node: %p invalid", feature_node); > + return -1; > + } > + > + arc = rte_graph_feature_arc_get(_arc); > + > + if (feature_node->id == RTE_NODE_ID_INVALID) { > + graph_err("Invalid node: %s", feature_node->name); > + return -1; > + } > + > + if (!feature_lookup(arc, feature_node->name, &finfo, &slot)) { > + graph_err("%s feature already added", feature_node->name); > + return -1; > + } > + > + if (slot >= RTE_GRAPH_FEATURE_MAX_PER_ARC) { > + graph_err("Max slot %u reached for feature addition", slot); > + return -1; > + } > + > + if (strstr(feature_node->name, arc->start_node->name)) { > + graph_err("Feature %s cannot point to itself: %s", > feature_node->name, > + arc->start_node->name); > + return -1; > + } > + > + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc- > >feature_arc_name)) { > + graph_err("unable to connect %s -> %s", arc->start_node- > >name, feature_node->name); > + return -1; > + } > + > + finfo = malloc(sizeof(*finfo)); > + if (!finfo) > + return -1; > + > + memset(finfo, 0, sizeof(*finfo)); > + > + finfo->feature_arc = (void *)arc; > + finfo->feature_node = feature_node; > + finfo->edge_to_this_feature = edge; > + > + /* Check for before and after constraints */ > + if (before_feature) { > + /* before_feature sanity */ > + if (feature_lookup(arc, before_feature, &before_finfo, NULL)) > + SET_ERR_JMP(EINVAL, finfo_free, > + "Invalid before feature name: %s", > before_feature); > + > + if (!before_finfo) > + SET_ERR_JMP(EINVAL, finfo_free, > + "before_feature %s does not exist", > before_feature); > + > + /* > + * Starting from 0 to before_feature, continue connecting > edges > + */ > + add_flag = 1; > + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { > + /* > + * As soon as we see before_feature. stop adding > edges > + */ > + if (!strncmp(temp->feature_node->name, > before_feature, > + RTE_GRAPH_NAMESIZE)) > + if (!connect_graph_nodes(finfo- > >feature_node, temp->feature_node, > + &edge, arc- > >feature_arc_name)) > + add_flag = 0; > + > + if (add_flag) > + connect_graph_nodes(temp->feature_node, > finfo->feature_node, NULL, > + arc->feature_arc_name); > + } > + } > + > + if (after_feature) { > + if (feature_lookup(arc, after_feature, &after_finfo, NULL)) > + SET_ERR_JMP(EINVAL, finfo_free, > + "Invalid after feature_name %s", > after_feature); > + > + if (!after_finfo) > + SET_ERR_JMP(EINVAL, finfo_free, > + "after_feature %s does not exist", > after_feature); > + > + /* Starting from after_feature to end continue connecting > edges */ > + add_flag = 0; > + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { > + /* We have already seen after_feature now */ > + if (add_flag) > + /* Add all features as next node to current > feature*/ > + connect_graph_nodes(finfo->feature_node, > temp->feature_node, NULL, > + arc->feature_arc_name); > + > + /* as soon as we see after_feature. start adding edges > + * from next iteration > + */ > + if (!strncmp(temp->feature_node->name, > after_feature, RTE_GRAPH_NAMESIZE)) > + /* connect after_feature to this feature */ > + if (!connect_graph_nodes(temp- > >feature_node, finfo->feature_node, > + &edge, arc- > >feature_arc_name)) > + add_flag = 1; > + } > + > + /* add feature next to after_feature */ > + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, > next_feature); > + } else { > + if (before_finfo) { > + after_finfo = NULL; > + STAILQ_FOREACH(temp, &arc->all_features, > next_feature) { > + if (before_finfo == temp) { > + if (after_finfo) > + STAILQ_INSERT_AFTER(&arc- > >all_features, after_finfo, > + finfo, > next_feature); > + else > + STAILQ_INSERT_HEAD(&arc- > >all_features, finfo, > + > next_feature); > + > + return 0; > + } > + after_finfo = temp; > + } > + } else { > + STAILQ_INSERT_TAIL(&arc->all_features, finfo, > next_feature); > + } > + } > + > + return 0; > + > +finfo_free: > + free(finfo); > + > + return -1; > +} > + > +int > +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char > *feature_name, > + rte_graph_feature_t *feat) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + struct rte_graph_feature_node_list *finfo = NULL; > + uint32_t slot; > + > + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { > + *feat = (rte_graph_feature_t) slot; > + return 0; > + } > + > + return -1; > +} > + > +int > +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, > const char *feature_name, > + int is_enable_disable) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + struct rte_graph_feature_node_list *finfo = NULL; > + struct rte_graph_feature *gf = NULL; > + uint32_t slot; > + > + /* validate _arc */ > + if (arc->feature_arc_main != __feature_arc_main) { > + graph_err("invalid feature arc: 0x%016" PRIx64, > (uint64_t)_arc); > + return -EINVAL; > + } > + > + /* validate index */ > + if (index >= arc->max_indexes) { > + graph_err("%s: Invalid provided index: %u >= %u configured", > arc->feature_arc_name, > + index, arc->max_indexes); > + return -1; > + } > + > + /* validate feature_name is already added or not */ > + if (feature_lookup(arc, feature_name, &finfo, &slot)) { > + graph_err("%s: No feature %s added", arc- > >feature_arc_name, feature_name); > + return -EINVAL; > + } > + > + if (!finfo) { > + graph_err("%s: No feature: %s found", arc- > >feature_arc_name, feature_name); > + return -EINVAL; > + } > + > + /* slot should be in valid range */ > + if (slot >= arc->max_features) { > + graph_err("%s/%s: Invalid free slot %u(max=%u) for feature", > arc->feature_arc_name, > + feature_name, slot, arc->max_features); > + return -EINVAL; > + } > + > + /* slot should be in range of 0 - 63 */ > + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { > + graph_err("%s/%s: Invalid slot: %u", arc->feature_arc_name, > + feature_name, slot); > + return -EINVAL; > + } > + > + if (finfo->node_index != slot) { > + graph_err("%s/%s: feature lookup slot mismatch with finfo > index: %u and lookup slot: %u", > + arc->feature_arc_name, feature_name, finfo- > >node_index, slot); > + return -1; > + } > + > + /* Get feature from active list */ > + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(arc)); > + if (gf->this_feature_index != slot) { > + graph_err("%s: %s received feature_index: %u does not match > with saved feature_index: %u", > + arc->feature_arc_name, feature_name, slot, gf- > >this_feature_index); > + return -1; > + } > + > + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & > + RTE_BIT64(slot))) { > + graph_err("%s: %s already enabled on index: %u", > + arc->feature_arc_name, feature_name, index); > + return -1; > + } > + > + if (!is_enable_disable && !arc->runtime_enabled_features) { > + graph_err("%s: No feature enabled to disable", arc- > >feature_arc_name); > + return -1; > + } > + > + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & > RTE_BIT64(slot))) { > + graph_err("%s: %s not enabled in bitmask for index: %u", > + arc->feature_arc_name, feature_name, index); > + return -1; > + } > + > + return 0; > +} > + > +static void > +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t > dest_list_index, > + uint16_t src_list_index) > +{ > + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; > + struct rte_graph_feature *sgf = NULL, *dgf = NULL; > + uint32_t i, j; > + > + for (i = 0; i < arc->max_features; i++) { > + sgf = __rte_graph_feature_get(arc, i, src_list_index); > + dgf = __rte_graph_feature_get(arc, i, dest_list_index); > + for (j = 0; j < arc->max_indexes; j++) { > + sgfd = rte_graph_feature_data_get(arc, sgf, j); > + dgfd = rte_graph_feature_data_get(arc, dgf, j); > + dgfd->user_data = sgfd->user_data; > + } > + } > +} > + > +static void > +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t > list_index) > +{ > + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; > + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; > + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; > + rte_graph_feature_list_t *flist = NULL; > + uint32_t fi, di, prev_fi; > + uint64_t bitmask; > + rte_edge_t edge; > + > + flist = arc->feature_list[list_index]; > + > + for (di = 0; di < arc->max_indexes; di++) { > + bitmask = arc->feature_bit_mask_by_index[di]; > + prev_fi = RTE_GRAPH_FEATURE_INVALID; > + /* for each feature set for index, set fast path data */ > + while (rte_bsf64_safe(bitmask, &fi)) { > + gf = __rte_graph_feature_get(arc, fi, list_index); > + gfd = rte_graph_feature_data_get(arc, gf, di); > + feature_arc_node_info_lookup(arc, fi, &finfo); > + > + /* If previous feature_index was valid in last loop */ > + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { > + prev_gf = __rte_graph_feature_get(arc, > prev_fi, list_index); > + prev_gfd = rte_graph_feature_data_get(arc, > prev_gf, di); > + /* > + * Get edge of previous feature node > connecting to this feature node > + */ > + feature_arc_node_info_lookup(arc, prev_fi, > &prev_finfo); > + if (!get_existing_edge(arc->feature_arc_name, > + prev_finfo->feature_node, > + finfo->feature_node, > &edge)) { > + feat_dbg("[%s/%s(%2u)/idx:%2u]: > %s[%u] = %s", > + arc->feature_arc_name, > + prev_finfo->feature_node- > >name, prev_fi, di, > + prev_finfo->feature_node- > >name, > + edge, finfo->feature_node- > >name); > + /* Copy feature index for next > iteration*/ > + gfd->next_edge = edge; > + prev_fi = fi; > + /* > + * Fill current feature as next enabled > + * feature to previous one > + */ > + prev_gfd->next_enabled_feature = fi; > + } else { > + /* Should not fail */ > + RTE_VERIFY(0); > + } > + } > + /* On first feature edge of the node to be added */ > + if (fi == rte_bsf64(arc- > >feature_bit_mask_by_index[di])) { > + if (!get_existing_edge(arc->feature_arc_name, > arc->start_node, > + finfo->feature_node, > + &edge)) { > + feat_dbg("[%s/%s/%2u/idx:%2u]: 1st > feat %s[%u] = %s", > + arc->feature_arc_name, > + arc->start_node->name, fi, di, > + arc->start_node->name, > edge, > + finfo->feature_node->name); > + /* Copy feature index for next > iteration*/ > + gfd->next_edge = edge; > + prev_fi = fi; > + /* Set first feature set array for > index*/ > + flist- > >first_enabled_feature_by_index[di] = fi; > + } else { > + /* Should not fail */ > + RTE_VERIFY(0); > + } > + } > + /* Clear current feature index */ > + bitmask &= ~RTE_BIT64(fi); > + } > + } > +} > + > +int > +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, > const > + char *feature_name, int32_t user_data) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + struct rte_graph_feature_node_list *finfo = NULL; > + struct rte_graph_feature_data *gfd = NULL; > + rte_graph_feature_rt_list_t passive_list; > + struct rte_graph_feature *gf = NULL; > + uint64_t fp_bitmask; > + uint32_t slot; > + > + if (rte_graph_feature_validate(_arc, index, feature_name, 1)) > + return -1; > + > + /** This should not fail as validate() has passed */ > + if (feature_lookup(arc, feature_name, &finfo, &slot)) > + RTE_VERIFY(0); > + > + if (!arc->runtime_enabled_features) > + prepare_feature_arc(arc); > + > + passive_list = ARC_PASSIVE_LIST(arc); > + > + gf = __rte_graph_feature_get(arc, slot, passive_list); > + gfd = rte_graph_feature_data_get(arc, gf, index); > + > + feat_dbg("%s/%s: Enabling feature on list: %u for index: %u at feature > slot %u", > + arc->feature_arc_name, feature_name, passive_list, index, > slot); > + > + /* Reset feature list */ > + feature_arc_list_reset(arc, passive_list); > + > + /* Copy user-data */ > + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); > + > + /* Set current user-data */ > + gfd->user_data = user_data; > + > + /* Set bitmask in control path bitmask */ > + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc- > >feature_bit_mask_by_index[index]); > + refill_feature_fastpath_data(arc, passive_list); > + > + /* Set fast path enable bitmask */ > + fp_bitmask = __atomic_load_n(&arc- > >feature_enable_bitmask[passive_list], __ATOMIC_RELAXED); > + fp_bitmask |= RTE_BIT64(slot); > + __atomic_store(&arc->feature_enable_bitmask[passive_list], > &fp_bitmask, __ATOMIC_RELAXED); > + > + /* Slow path updates */ > + arc->runtime_enabled_features++; > + > + /* Increase feature node info reference count */ > + finfo->ref_count++; > + > + /* Store release semantics for active_list update */ > + __atomic_store(&arc->active_feature_list, &passive_list, > __ATOMIC_RELEASE); > + > + return 0; > +} > + > +int > +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, > const char *feature_name) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + struct rte_graph_feature_data *gfd = NULL; > + struct rte_graph_feature_node_list *finfo = NULL; > + rte_graph_feature_rt_list_t passive_list; > + struct rte_graph_feature *gf = NULL; > + uint32_t slot; > + > + if (rte_graph_feature_validate(_arc, index, feature_name, 0)) > + return -1; > + > + if (feature_lookup(arc, feature_name, &finfo, &slot)) > + return -1; > + > + passive_list = ARC_PASSIVE_LIST(arc); > + > + gf = __rte_graph_feature_get(arc, slot, passive_list); > + gfd = rte_graph_feature_data_get(arc, gf, index); > + > + feat_dbg("%s/%s: Disabling feature for index: %u at feature slot %u", > arc->feature_arc_name, > + feature_name, index, slot); > + > + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc- > >feature_bit_mask_by_index[index]); > + > + /* Set fast path enable bitmask */ > + arc->feature_enable_bitmask[passive_list] &= ~(RTE_BIT64(slot)); > + > + /* Reset feature list */ > + feature_arc_list_reset(arc, passive_list); > + > + /* Copy user-data */ > + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); > + > + /* Reset current user-data */ > + gfd->user_data = ~0; > + > + refill_feature_fastpath_data(arc, passive_list); > + > + finfo->ref_count--; > + arc->runtime_enabled_features--; > + > + /* Store release semantics for active_list update */ > + __atomic_store(&arc->active_feature_list, &passive_list, > __ATOMIC_RELEASE); > + > + return 0; > +} > + > +int > +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > + struct rte_graph_feature_node_list *node_info = NULL; > + > + while (!STAILQ_EMPTY(&arc->all_features)) { > + node_info = STAILQ_FIRST(&arc->all_features); > + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); > + free(node_info); > + } > + feature_arc_list_destroy(arc->feature_list[0]); > + feature_arc_list_destroy(arc->feature_list[1]); > + rte_free(arc->features[0]); > + rte_free(arc->features[1]); > + > + dm->feature_arcs[arc->feature_arc_index] = > RTE_GRAPH_FEATURE_ARC_INITIALIZER; > + > + rte_free(arc); > + return 0; > +} > + > +int > +rte_graph_feature_arc_cleanup(void) > +{ > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > + uint32_t iter; > + > + if (!__feature_arc_main) > + return -1; > + > + for (iter = 0; iter < dm->max_feature_arcs; iter++) { > + if (dm->feature_arcs[iter] == > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > + continue; > + > + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm- > >feature_arcs[iter]); > + } > + free(dm); > + > + __feature_arc_main = NULL; > + > + return 0; > +} > + > +int > +rte_graph_feature_arc_lookup_by_name(const char *arc_name, > rte_graph_feature_arc_t *_arc) > +{ > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > + struct rte_graph_feature_arc *arc = NULL; > + uint32_t iter; > + > + if (!__feature_arc_main) > + return -1; > + > + for (iter = 0; iter < dm->max_feature_arcs; iter++) { > + if (dm->feature_arcs[iter] == > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > + continue; > + > + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); > + > + if (strstr(arc_name, arc->feature_arc_name)) { > + if (_arc) > + *_arc = (rte_graph_feature_arc_t)arc; > + return 0; > + } > + } > + > + return -1; > +} > + > +int > +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t > _arc) > +{ > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > + > + return arc->runtime_enabled_features; > +} > + > + > diff --git a/lib/graph/meson.build b/lib/graph/meson.build > index 0cb15442ab..d916176fb7 100644 > --- a/lib/graph/meson.build > +++ b/lib/graph/meson.build > @@ -14,11 +14,13 @@ sources = files( > 'graph_debug.c', > 'graph_stats.c', > 'graph_populate.c', > + 'graph_feature_arc.c', > 'graph_pcap.c', > 'rte_graph_worker.c', > 'rte_graph_model_mcore_dispatch.c', > ) > headers = files('rte_graph.h', 'rte_graph_worker.h') > +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') > indirect_headers += files( > 'rte_graph_model_mcore_dispatch.h', > 'rte_graph_model_rtc.h', > diff --git a/lib/graph/rte_graph_feature_arc.h > b/lib/graph/rte_graph_feature_arc.h > new file mode 100644 > index 0000000000..e3bf4eb73d > --- /dev/null > +++ b/lib/graph/rte_graph_feature_arc.h > @@ -0,0 +1,373 @@ > +/* SPDX-License-Identifier: BSD-3-Clause > + * Copyright(C) 2024 Marvell International Ltd. > + */ > + > +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ > +#define _RTE_GRAPH_FEATURE_ARC_H_ > + > +#include <assert.h> > +#include <errno.h> > +#include <signal.h> > +#include <stddef.h> > +#include <stdint.h> > +#include <stdio.h> > +#include <stdlib.h> > +#include <string.h> > + > +#include <rte_common.h> > +#include <rte_compat.h> > +#include <rte_debug.h> > +#include <rte_graph.h> > +#include <rte_graph_worker.h> > + > +#ifdef __cplusplus > +extern "C" { > +#endif > + > +/** > + * @file > + * > + * rte_graph_feature_arc.h > + * > + * Define APIs and structures/variables with respect to feature arc > + * > + * - Feature arc(s) > + * - Feature(s) > + * > + * A feature arc represents an ordered list of features/protocol-nodes at a > + * given networking layer. Feature arc provides a high level abstraction to > + * connect various *rte_graph* nodes, designated as *feature nodes*, and > + * allowing steering of packets across these feature nodes fast path > processing > + * in a generic manner. In a typical network stack, often a protocol or feature > + * must be first enabled on a given interface, before any packet is steered > + * towards it for feature processing. For eg: incoming IPv4 packets are sent to > + * routing sub-system only after a valid IPv4 address is assigned to the > + * received interface. In other words, often packets needs to be steered across > + * features not based on the packet content but based on whether a feature is > + * enable or disable on a given incoming/outgoing interface. Feature arc > + * provides mechanism to enable/disable feature(s) on each interface at > runtime > + * and allow seamless packet steering across runtime enabled feature nodes > in > + * fast path. > + * > + * Feature arc also provides a way to steer packets from standard nodes to > + * custom/user-defined *feature nodes* without any change in standard > node's > + * fast path functions > + * > + * On a given interface multiple feature(s) might be enabled in a particular > + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" > + * features may be enabled on "eth0" interface in "L3-output" feature arc. > + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" > + * interface in same "L3-output" feature arc. > + * > + * When multiple features are present in a given feature arc, its imperative > + * to allow each feature processing in a particular sequential order. For > + * instance, in "L3-input" feature arc it may be required to run "IPsec > + * input" feature first, for packet decryption, before "ip-lookup". So a > + * sequential order must be maintained among features present in a feature > arc. > + * > + * Features are enabled/disabled multiple times at runtime to some or all > + * available interfaces present in the system. Features can be > enabled/disabled > + * even after @b rte_graph_create() is called. Enable/disabling features on > one > + * interface is independent of other interface. > + * > + * A given feature might consume packet (if it's configured to consume) or > may > + * forward it to next enabled feature. For instance, "IPsec input" feature may > + * consume/drop all packets with "Protect" policy action while all packets with > + * policy action as "Bypass" may be forwarded to next enabled feature (with > in > + * same feature arc) > + * > + * This library facilitates rte graph based applications to steer packets in > + * fast path to different feature nodes with-in a feature arc and support all > + * functionalities described above > + * > + * In order to use feature-arc APIs, applications needs to do following in > + * control path: > + * - Initialize feature arc library via rte_graph_feature_arc_init() > + * - Create feature arc via rte_graph_feature_arc_create() > + * - *Before calling rte_graph_create()*, features must be added to feature- > arc > + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding > + * features in a sequential order with "runs_after" and "runs_before" > + * constraints. > + * - Post rte_graph_create(), features can be enabled/disabled at runtime on > + * any interface via rte_graph_feature_enable()/rte_graph_feature_disable() > + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() > + * > + * In fast path, APIs are provided to steer packets towards feature path from > + * - start_node (provided as an argument to rte_graph_feature_arc_create()) > + * - feature nodes (which are added via rte_graph_feature_add()) > + * > + * For typical steering of packets across feature nodes, application required > + * to know "rte_edges" which are saved in feature data object. Feature data > + * object is unique for every interface per feature with in a feature arc. > + * > + * When steering packets from start_node to feature node: > + * - rte_graph_feature_arc_first_feature_get() provides first enabled feature. > + * - Next rte_edge from start_node to first enabled feature can be obtained > via > + * rte_graph_feature_arc_feature_set() > + * > + * rte_mbuf can carry [current feature, index] from start_node of an arc to > other > + * feature nodes > + * > + * In feature node, application can get 32-bit user_data > + * via_rte_graph_feature_user_data_get() which is provided in > + * rte_graph_feature_enable(). User data can hold feature specific cookie like > + * IPsec policy database index (if more than one are supported) > + * > + * If feature node is not consuming packet, next enabled feature and next > + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() > + * > + * It is application responsibility to ensure that at-least *last feature*(or sink > + * feature) must be enabled from where packet can exit feature-arc path, if > + * *NO* intermediate feature is consuming the packet and it has reached till > + * the end of feature arc path > + * > + * Synchronization among cores > + * --------------------------- > + * Subsequent calls to rte_graph_feature_enable() is allowed while worker > cores > + * are processing in rte_graph_walk() loop. However, for > + * rte_graph_feature_disable() application must use RCU based > synchronization > + */ > + > +/**< Initializer value for rte_graph_feature_arc_t */ > +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER > ((rte_graph_feature_arc_t)UINT64_MAX) > + > +/** Max number of features supported in a given feature arc */ > +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 > + > +/** Length of feature arc name */ > +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE > + > +/** @internal */ > +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) > + > +/**< Initializer value for rte_graph_feature_arc_t */ > +#define RTE_GRAPH_FEATURE_INVALID > rte_graph_feature_cast(UINT8_MAX) > + > +/** rte_graph feature arc object */ > +typedef uint64_t rte_graph_feature_arc_t; > + > +/** rte_graph feature object */ > +typedef uint8_t rte_graph_feature_t; > + > +/** runtime active feature list index with in feature arc*/ > +typedef uint8_t rte_graph_feature_rt_list_t; > + > +/** per feature arc monotonically increasing counter to synchronize fast path > APIs */ > +typedef uint16_t rte_graph_feature_counter_t; > + > +/** > + * Initialize feature arc subsystem > + * > + * @param max_feature_arcs > + * Maximum number of feature arcs required to be supported > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_arc_init(int max_feature_arcs); > + > +/** > + * Create a feature arc > + * > + * @param feature_arc_name > + * Feature arc name with max length of @ref > RTE_GRAPH_FEATURE_ARC_NAMELEN > + * @param max_features > + * Maximum number of features to be supported in this feature arc > + * @param max_indexes > + * Maximum number of interfaces/ports/indexes to be supported > + * @param start_node > + * Base node where this feature arc's features are checked in fast path > + * @param[out] _arc > + * Feature arc object > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_arc_create(const char *feature_arc_name, int > max_features, int max_indexes, > + struct rte_node_register *start_node, > + rte_graph_feature_arc_t *_arc); > + > +/** > + * Get feature arc object with name > + * > + * @param arc_name > + * Feature arc name provided to successful @ref > rte_graph_feature_arc_create > + * @param[out] _arc > + * Feature arc object returned > + * > + * @return > + * 0: Success > + * <0: Failure. > + */ > +__rte_experimental > +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, > rte_graph_feature_arc_t *_arc); > + > +/** > + * Add a feature to already created feature arc. For instance > + * > + * 1. Add first feature node: "ipv4-input" to input arc > + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); > + * > + * 2. Add "ipsec-input" feature node after "ipv4-input" node > + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", > NULL); > + * > + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" node > + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", NULL, > "ipv4-input"); > + * > + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input > + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4-input", > "ipsec-input"); > + * > + * @param _arc > + * Feature arc handle returned from @ref rte_graph_feature_arc_create() > + * @param feature_node > + * Graph node representing feature. On success, feature_node is next_node > of > + * feature_arc->start_node > + * @param runs_after > + * Add this feature_node after already added "runs_after". Creates > + * start_node -> runs_after -> this_feature sequence > + * @param runs_before > + * Add this feature_node before already added "runs_before". Creates > + * start_node -> this_feature -> runs_before sequence > + * > + * <I> Must be called before rte_graph_create() </I> > + * <I> rte_graph_feature_add() is not allowed after call to > + * rte_graph_feature_enable() so all features must be added before they can > be > + * enabled </I> > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct > rte_node_register *feature_node, > + const char *runs_after, const char *runs_before); > + > +/** > + * Enable feature within a feature arc > + * > + * Must be called after @b rte_graph_create(). > + * > + * @param _arc > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > @ref > + * rte_graph_feature_arc_lookup_by_name > + * @param index > + * Application specific index. Can be corresponding to interface_id/port_id > etc > + * @param feature_name > + * Name of the node which is already added via @ref rte_graph_feature_add > + * @param user_data > + * Application specific data which is retrieved in fast path > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, > const char *feature_name, > + int32_t user_data); > + > +/** > + * Validate whether subsequent enable/disable feature would succeed or not. > + * API is thread-safe > + * > + * @param _arc > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > @ref > + * rte_graph_feature_arc_lookup_by_name > + * @param index > + * Application specific index. Can be corresponding to interface_id/port_id > etc > + * @param feature_name > + * Name of the node which is already added via @ref rte_graph_feature_add > + * @param is_enable_disable > + * If 1, validate whether subsequent @ref rte_graph_feature_enable would > pass or not > + * If 0, validate whether subsequent @ref rte_graph_feature_disable would > pass or not > + * > + * @return > + * 0: Subsequent enable/disable API would pass > + * <0: Subsequent enable/disable API would not pass > + */ > +__rte_experimental > +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, > + const char *feature_name, int is_enable_disable); > + > +/** > + * Disable already enabled feature within a feature arc > + * > + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe > + * > + * @param _arc > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > @ref > + * rte_graph_feature_arc_lookup_by_name > + * @param index > + * Application specific index. Can be corresponding to interface_id/port_id > etc > + * @param feature_name > + * Name of the node which is already added via @ref rte_graph_feature_add > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, > + const char *feature_name); > + > +/** > + * Get rte_graph_feature_t object from feature name > + * > + * @param arc > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > @ref > + * rte_graph_feature_arc_lookup_by_name > + * @param feature_name > + * Feature name provided to @ref rte_graph_feature_add > + * @param[out] feature > + * Feature object > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char > *feature_name, > + rte_graph_feature_t *feature); > + > +/** > + * Delete feature_arc object > + * > + * @param _arc > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > @ref > + * rte_graph_feature_arc_lookup_by_name > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); > + > +/** > + * Cleanup all feature arcs > + * > + * @return > + * 0: Success > + * <0: Failure > + */ > +__rte_experimental > +int rte_graph_feature_arc_cleanup(void); > + > +/** > + * Slow path API to know how many features are currently enabled within a > featur-arc > + * > + * @param _arc > + * Feature arc object > + * > + * @return: Number of enabled features > + */ > +__rte_experimental > +int rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t > _arc); > +#ifdef __cplusplus > +} > +#endif > + > +#endif > diff --git a/lib/graph/rte_graph_feature_arc_worker.h > b/lib/graph/rte_graph_feature_arc_worker.h > new file mode 100644 > index 0000000000..6019d74853 > --- /dev/null > +++ b/lib/graph/rte_graph_feature_arc_worker.h > @@ -0,0 +1,548 @@ > +/* SPDX-License-Identifier: BSD-3-Clause > + * Copyright(C) 2024 Marvell International Ltd. > + */ > + > +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ > +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ > + > +#include <stddef.h> > +#include <rte_graph_feature_arc.h> > +#include <rte_bitops.h> > + > +/** > + * @file > + * > + * rte_graph_feature_arc_worker.h > + * > + * Defines fast path structure > + */ > + > +#ifdef __cplusplus > +extern "C" { > +#endif > + > +/** @internal > + * > + * Slow path feature node info list > + */ > +struct rte_graph_feature_node_list { > + /** Next feature */ > + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; > + > + /** node representing feature */ > + struct rte_node_register *feature_node; > + > + /** How many indexes/interfaces using this feature */ > + int32_t ref_count; > + > + /* node_index in list (after feature_enable())*/ > + uint32_t node_index; > + > + /** Back pointer to feature arc */ > + void *feature_arc; > + > + /** rte_edge_t to this feature node from feature_arc->start_node */ > + rte_edge_t edge_to_this_feature; > +}; > + > +/** > + * Fast path holding rte_edge_t and next enabled feature for an feature > + */ > +typedef struct __rte_packed rte_graph_feature_data { > + /* next node to which current mbuf should go*/ > + rte_edge_t next_edge; > + > + /* next enabled feature on this arc for current index */ > + union { > + uint16_t reserved; > + struct { > + rte_graph_feature_t next_enabled_feature; > + }; > + }; > + > + /* user_data */ > + int32_t user_data; > +} rte_graph_feature_data_t; > + > +/** > + * Fast path feature structure. Holds re_graph_feature_data_t per index > + */ > +struct __rte_cache_aligned rte_graph_feature { > + uint16_t this_feature_index; > + > + /* Array of size arc->feature_data_size > + * [data-index-0][data-index-1]... > + * Each index of size: sizeof(rte_graph_feature_data_t) > + */ > + uint8_t feature_data_by_index[]; > +}; > + > +/** > + * fast path cache aligned feature list holding all features > + * There are two feature lists: active, passive > + * > + * Fast APIs works on active list while control plane updates passive list > + * A atomic update to arc->active_feature_list is done to switch between > active > + * and passive > + */ > +typedef struct __rte_cache_aligned rte_graph_feature_list { > + /** > + * fast path array holding per_feature data. > + * Duplicate entry as feature-arc also hold this pointer > + * arc->features[] > + * > + *<-------------feature-0 ---------><CEIL><---------feature-1 -------------- > >... > + *[index-0][index-1]...[max_index-1] [index-0][index-1] > ...[max_index-1]... > + */ > + struct rte_graph_feature *indexed_by_features; > + /* > + * fast path array holding first enabled feature per index > + * (Required in start_node. In non start_node, mbuf can hold next > enabled > + * feature) > + */ > + rte_graph_feature_t first_enabled_feature_by_index[]; > +} rte_graph_feature_list_t; > + > +/** > + * rte_graph feature arc object > + * > + * A feature-arc can only hold RTE_GRAPH_FEATURE_MAX_PER_ARC features > but no > + * limit to interface index > + * > + * Representing a feature arc holding all features which are enabled/disabled > + * on any interfaces > + */ > +struct __rte_cache_aligned rte_graph_feature_arc { > + /* First 64B is fast path variables */ > + RTE_MARKER fast_path_variables; > + > + /** runtime active feature list */ > + rte_graph_feature_rt_list_t active_feature_list; > + > + /* Actual Size of feature_list0 */ > + uint16_t feature_list_size; > + > + /** > + * Size each feature in fastpath. > + * sizeof(arc->active_list->indexed_by_feature[0]) > + */ > + uint16_t feature_size; > + > + /* Size of arc->max_index * sizeof(rte_graph_feature_data_t) */ > + uint16_t feature_data_size; > + > + /** > + * Fast path bitmask indicating if a feature is enabled or not Number > + * of bits: RTE_GRAPH_FEATURE_MAX_PER_ARC > + */ > + uint64_t feature_enable_bitmask[2]; > + rte_graph_feature_list_t *feature_list[2]; > + struct rte_graph_feature *features[2]; > + > + /** index in feature_arc_main */ > + uint16_t feature_arc_index; > + > + uint16_t reserved[3]; > + > + /** Slow path variables follows*/ > + RTE_MARKER slow_path_variables; > + > + /** feature arc name */ > + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; > + > + /** All feature lists */ > + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; > + > + uint32_t runtime_enabled_features; > + > + /** Back pointer to feature_arc_main */ > + void *feature_arc_main; > + > + /* start_node */ > + struct rte_node_register *start_node; > + > + /* maximum number of features supported by this arc */ > + uint32_t max_features; > + > + /* maximum number of index supported by this arc */ > + uint32_t max_indexes; > + > + /* Slow path bit mask per feature per index */ > + uint64_t feature_bit_mask_by_index[]; > +}; > + > +/** Feature arc main */ > +typedef struct feature_arc_main { > + /** number of feature arcs created by application */ > + uint32_t num_feature_arcs; > + > + /** max features arcs allowed */ > + uint32_t max_feature_arcs; > + > + /** feature arcs */ > + rte_graph_feature_arc_t feature_arcs[]; > +} rte_graph_feature_arc_main_t; > + > +/** @internal Get feature arc pointer from object */ > +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc *)arc) > + > +extern rte_graph_feature_arc_main_t *__feature_arc_main; > + > +/** > + * API to know if feature is valid or not > + */ > + > +static __rte_always_inline int > +rte_graph_feature_is_valid(rte_graph_feature_t feature) > +{ > + return (feature != RTE_GRAPH_FEATURE_INVALID); > +} > + > +/** > + * Get rte_graph_feature object with no checks > + * > + * @param arc > + * Feature arc pointer > + * @param feature > + * Feature index > + * @param feature_list > + * active feature list retrieved from > rte_graph_feature_arc_has_any_feature() > + * or rte_graph_feature_arc_has_feature() > + * > + * @return > + * Internal feature object. > + */ > +static __rte_always_inline struct rte_graph_feature * > +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, > rte_graph_feature_t feature, > + const rte_graph_feature_rt_list_t feature_list) > +{ > + return ((struct rte_graph_feature *)((uint8_t *)(arc- > >features[feature_list] + > + (feature * arc->feature_size)))); > +} > + > +/** > + * Get rte_graph_feature object for a given interface/index from feature arc > + * > + * @param arc > + * Feature arc pointer > + * @param feature > + * Feature index > + * > + * @return > + * Internal feature object. > + */ > +static __rte_always_inline struct rte_graph_feature * > +rte_graph_feature_get(struct rte_graph_feature_arc *arc, > rte_graph_feature_t feature) > +{ > + RTE_VERIFY(feature < arc->max_features); > + > + if (likely(rte_graph_feature_is_valid(feature))) > + return __rte_graph_feature_get(arc, feature, arc- > >active_feature_list); > + > + return NULL; > +} > + > +static __rte_always_inline rte_graph_feature_data_t * > +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct > rte_graph_feature *feature, > + uint8_t index) > +{ > + RTE_SET_USED(arc); > + return ((rte_graph_feature_data_t *)(feature->feature_data_by_index > + > + (index * > sizeof(rte_graph_feature_data_t)))); > +} > + > +/** > + * Get rte_graph feature data object for a index in feature > + * > + * @param arc > + * feature arc > + * @param feature > + * Pointer to feature object > + * @param index > + * Index of feature maintained in slow path linked list > + * > + * @return > + * Valid feature data > + */ > +static __rte_always_inline rte_graph_feature_data_t * > +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct > rte_graph_feature *feature, > + uint8_t index) > +{ > + if (likely(index < arc->max_indexes)) > + return __rte_graph_feature_data_get(arc, feature, index); > + > + RTE_VERIFY(0); > +} > + > +/** > + * Fast path API to check if any feature enabled on a feature arc > + * Typically from arc->start_node process function > + * > + * @param arc > + * Feature arc object > + * @param[out] plist > + * Pointer to runtime active feature list which needs to be provided to other > + * fast path APIs > + * > + * @return > + * 0: If no feature enabled > + * Non-Zero: Bitmask of features enabled. plist is valid > + * > + */ > +static __rte_always_inline uint64_t > +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, > + rte_graph_feature_rt_list_t *plist) > +{ > + *plist = __atomic_load_n(&arc->active_feature_list, > __ATOMIC_RELAXED); > + > + return (__atomic_load_n(arc->feature_enable_bitmask + > (uint8_t)*plist, > + __ATOMIC_RELAXED)); > +} > + > +/** > + * Fast path API to check if provided feature is enabled on any interface/index > + * or not > + * > + * @param arc > + * Feature arc object > + * @param feature > + * Input rte_graph_feature_t that needs to be checked > + * @param[out] plist > + * Returns active list to caller which needs to be provided to other fast path > + * APIs > + * > + * @return > + * 1: If feature is enabled in arc > + * 0: If feature is not enabled in arc > + */ > +static __rte_always_inline int > +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, > + rte_graph_feature_t feature, > + rte_graph_feature_rt_list_t *plist) > +{ > + uint64_t bitmask = RTE_BIT64(feature); > + > + *plist = __atomic_load_n(&arc->active_feature_list, > __ATOMIC_RELAXED); > + > + return (bitmask & __atomic_load_n(arc->feature_enable_bitmask + > (uint8_t)*plist, > + __ATOMIC_RELAXED)); > +} > + > +/** > + * Prefetch feature arc fast path cache line > + * > + * @param arc > + * RTE_GRAPH feature arc object > + */ > +static __rte_always_inline void > +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) > +{ > + rte_prefetch0((void *)&arc->fast_path_variables); > +} > + > +/** > + * Prefetch feature related fast path cache line > + * > + * @param arc > + * RTE_GRAPH feature arc object > + * @param list > + * Pointer to runtime active feature list from > rte_graph_feature_arc_has_any_feature(); > + * @param feature > + * Pointer to feature object > + */ > +static __rte_always_inline void > +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, > + const rte_graph_feature_rt_list_t list, > + rte_graph_feature_t feature) > +{ > + /* feature cache line */ > + if (likely(rte_graph_feature_is_valid(feature))) > + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, > list)); > +} > + > +/** > + * Prefetch feature data upfront. Perform sanity > + * > + * @param _arc > + * RTE_GRAPH feature arc object > + * @param list > + * Pointer to runtime active feature list from > rte_graph_feature_arc_has_any_feature(); > + * @param feature > + * Pointer to feature object returned from @ref > + * rte_graph_feature_arc_first_feature_get() > + * @param index > + * Interface/index > + */ > +static __rte_always_inline void > +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, > + const rte_graph_feature_rt_list_t list, > + rte_graph_feature_t feature, uint32_t index) > +{ > + if (likely(rte_graph_feature_is_valid(feature))) > + rte_prefetch0((void *)((uint8_t *)arc->features[list] + > + offsetof(struct rte_graph_feature, > feature_data_by_index) + > + (index * sizeof(rte_graph_feature_data_t)))); > +} > + > +/** > + * Fast path API to get first enabled feature on interface index > + * Typically required in arc->start_node so that from returned feature, > + * feature-data can be retrieved to steer packets > + * > + * @param arc > + * Feature arc object > + * @param list > + * Pointer to runtime active feature list from > + * rte_graph_feature_arc_has_any_feature() or > + * rte_graph_feature_arc_has_feature() > + * @param index > + * Interface Index > + * @param[out] feature > + * Pointer to rte_graph_feature_t. > + * > + * @return > + * 0. Success. feature field is valid > + * 1. Failure. feature field is invalid > + * > + */ > +static __rte_always_inline int > +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, > + const rte_graph_feature_rt_list_t list, > + uint32_t index, > + rte_graph_feature_t *feature) > +{ > + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; > + > + *feature = feature_list->first_enabled_feature_by_index[index]; > + > + return rte_graph_feature_is_valid(*feature); > +} > + > +/** > + * Fast path API to get next enabled feature on interface index with provided > + * input feature > + * > + * @param arc > + * Feature arc object > + * @param list > + * Pointer to runtime active feature list from > + * rte_graph_feature_arc_has_any_feature() or > + * @param index > + * Interface Index > + * @param[in][out] feature > + * Pointer to rte_graph_feature_t. Input feature set to next enabled feature > + * after success return > + * @param[out] next_edge > + * Edge from current feature to next feature. Valid only if next feature is > valid > + * > + * @return > + * 0. Success. next enabled feature is valid. > + * 1. Failure. next enabled feature is invalid > + */ > +static __rte_always_inline int > +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc *arc, > + const rte_graph_feature_rt_list_t list, > + uint32_t index, > + rte_graph_feature_t *feature, > + rte_edge_t *next_edge) > +{ > + rte_graph_feature_data_t *feature_data = NULL; > + struct rte_graph_feature *f = NULL; > + > + if (likely(rte_graph_feature_is_valid(*feature))) { > + f = __rte_graph_feature_get(arc, *feature, list); > + feature_data = rte_graph_feature_data_get(arc, f, index); > + *feature = feature_data->next_enabled_feature; > + *next_edge = feature_data->next_edge; > + return (*feature == RTE_GRAPH_FEATURE_INVALID); > + } > + > + return 1; > +} > + > +/** > + * Set fields with respect to first enabled feature in an arc and return edge > + * Typically returned feature and interface index must be saved in rte_mbuf > + * structure to pass this information to next feature node > + * > + * @param arc > + * Feature arc object > + * @param list > + * Pointer to runtime active feature list from > rte_graph_feature_arc_has_any_feature(); > + * @param index > + * Index (of interface) > + * @param[out] gf > + * Pointer to rte_graph_feature_t. Valid if API returns Success > + * @param[out] edge > + * Edge to steer packet from arc->start_node to first enabled feature. Valid > + * only if API returns Success > + * > + * @return > + * 0: If valid feature is set by API > + * 1: If valid feature is NOT set by API > + */ > +static __rte_always_inline rte_graph_feature_t > +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, > + const rte_graph_feature_rt_list_t list, > + uint32_t index, > + rte_graph_feature_t *gf, > + rte_edge_t *edge) > +{ > + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; > + struct rte_graph_feature_data *feature_data = NULL; > + struct rte_graph_feature *feature = NULL; > + rte_graph_feature_t f; > + > + /* reset */ > + *gf = RTE_GRAPH_FEATURE_INVALID; > + f = feature_list->first_enabled_feature_by_index[index]; > + > + if (unlikely(rte_graph_feature_is_valid(f))) { > + feature = __rte_graph_feature_get(arc, f, list); > + feature_data = rte_graph_feature_data_get(arc, feature, > index); > + *gf = f; > + *edge = feature_data->next_edge; > + return 0; > + } > + > + return 1; > +} > + > +/** > + * Get user data corresponding to current feature set by application in > + * rte_graph_feature_enable() > + * > + * @param arc > + * Feature arc object > + * @param list > + * Pointer to runtime active feature list from > rte_graph_feature_arc_has_any_feature(); > + * @param feature > + * Feature index > + * @param index > + * Interface index > + * > + * @return > + * UINT32_MAX: Failure > + * Valid user data: Success > + */ > +static __rte_always_inline uint32_t > +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, > + const rte_graph_feature_rt_list_t list, > + rte_graph_feature_t feature, > + uint32_t index) > +{ > + rte_graph_feature_data_t *fdata = NULL; > + struct rte_graph_feature *f = NULL; > + > + if (likely(rte_graph_feature_is_valid(feature))) { > + f = __rte_graph_feature_get(arc, feature, list); > + fdata = rte_graph_feature_data_get(arc, f, index); > + return fdata->user_data; > + } > + > + return UINT32_MAX; > +} > +#ifdef __cplusplus > +} > +#endif > +#endif > diff --git a/lib/graph/version.map b/lib/graph/version.map > index 2c83425ddc..82b2469fba 100644 > --- a/lib/graph/version.map > +++ b/lib/graph/version.map > @@ -52,3 +52,20 @@ DPDK_25 { > > local: *; > }; > + > +EXPERIMENTAL { > + global: > + > + # added in 24.11 > + rte_graph_feature_arc_init; > + rte_graph_feature_arc_create; > + rte_graph_feature_arc_lookup_by_name; > + rte_graph_feature_add; > + rte_graph_feature_enable; > + rte_graph_feature_validate; > + rte_graph_feature_disable; > + rte_graph_feature_lookup; > + rte_graph_feature_arc_destroy; > + rte_graph_feature_arc_cleanup; > + rte_graph_feature_arc_num_enabled_features; > +}; > -- > 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* RE: [RFC PATCH 1/3] graph: add feature arc support 2024-09-11 4:41 ` Kiran Kumar Kokkilagadda @ 2024-10-10 4:42 ` Nitin Saxena 0 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 4:42 UTC (permalink / raw) To: Kiran Kumar Kokkilagadda, Jerin Jacob, Nithin Kumar Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena Hi Kiran, See my inline comments. Somehow I forgot to respond earlier Thanks, Nitin > -----Original Message----- > From: Kiran Kumar Kokkilagadda <kirankumark@marvell.com> > Sent: Wednesday, September 11, 2024 10:11 AM > To: Nitin Saxena <nsaxena@marvell.com>; Jerin Jacob <jerinj@marvell.com>; > Nithin Kumar Dabilpuram <ndabilpuram@marvell.com>; Zhirun Yan > <yanzhirun_163@163.com> > Cc: dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com> > Subject: RE: [RFC PATCH 1/3] graph: add feature arc support > > > > > -----Original Message----- > > From: Nitin Saxena <nsaxena@marvell.com> > > Sent: Saturday, September 7, 2024 1:01 PM > > To: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda > > <kirankumark@marvell.com>; Nithin Kumar Dabilpuram > > <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com> > > Cc: dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com> > > Subject: [RFC PATCH 1/3] graph: add feature arc support > > > > add feature arc to allow dynamic steering of packets across graph nodes > > based on protocol features enabled on incoming or outgoing interface > > > > Signed-off-by: Nitin Saxena <nsaxena@marvell.com> > > --- > > lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ > > lib/graph/meson.build | 2 + > > lib/graph/rte_graph_feature_arc.h | 373 +++++++++ > > lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ > > lib/graph/version.map | 17 + > > 5 files changed, 1899 insertions(+) > > create mode 100644 lib/graph/graph_feature_arc.c > > create mode 100644 lib/graph/rte_graph_feature_arc.h > > create mode 100644 lib/graph/rte_graph_feature_arc_worker.h > > > > diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c > > new file mode 100644 > > index 0000000000..3b05bac137 > > --- /dev/null > > +++ b/lib/graph/graph_feature_arc.c > > @@ -0,0 +1,959 @@ > > +/* SPDX-License-Identifier: BSD-3-Clause > > + * Copyright(C) 2024 Marvell International Ltd. > > + */ > > + > > +#include "graph_private.h" > > +#include <rte_graph_feature_arc_worker.h> > > +#include <rte_malloc.h> > > + > > +#define __RTE_GRAPH_FEATURE_ARC_MAX 32 > > + > > +#define ARC_PASSIVE_LIST(arc) (arc->active_feature_list ^ 0x1) > > + > > +#define rte_graph_uint_cast(x) ((unsigned int)x) > > +#define feat_dbg graph_err > > + > > +rte_graph_feature_arc_main_t *__feature_arc_main; > > + > > +/* Make sure fast path cache line is compact */ > > +_Static_assert((offsetof(struct rte_graph_feature_arc, > slow_path_variables) > > + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) > > + <= RTE_CACHE_LINE_SIZE); > > + > > + > > +static int > > +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, > > + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) > > +{ > > + struct rte_graph_feature_node_list *finfo = NULL; > > + const char *name; > > + > > + if (!feat_name) > > + return -1; > > + > > + if (slot) > > + *slot = 0; > > + > > + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { > > + RTE_VERIFY(finfo->feature_arc == arc); > > + name = rte_node_id_to_name(finfo->feature_node->id); > > + if (!strncmp(name, feat_name, RTE_GRAPH_NAMESIZE)) { > > + if (ffinfo) > > + *ffinfo = finfo; > > + return 0; > > + } > > + if (slot) > > + (*slot)++; > > + } > > + return -1; > > +} > > + > > +static int > > +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t > > feature_index, > > + struct rte_graph_feature_node_list **ppfinfo) > > +{ > > + struct rte_graph_feature_node_list *finfo = NULL; > > + uint32_t index = 0; > > + > > + if (!ppfinfo) > > + return -1; > > + > > + *ppfinfo = NULL; > > + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { > > + if (index == feature_index) { > > + if (finfo->node_index == feature_index) > > + return -1; > > + *ppfinfo = finfo; > > + } > > + index++; > > + } > > + if (feature_index && (index >= feature_index)) > > + return -1; > > + > > + return 0; > > +} > > + > > +static void > > +prepare_feature_arc(struct rte_graph_feature_arc *arc) > > +{ > > + struct rte_graph_feature_node_list *finfo = NULL; > > + uint32_t index = 0; > > + > > + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { > > + finfo->node_index = index; > > + index++; > > + } > > +} > > + > > +static int > > +feature_arc_lookup(rte_graph_feature_arc_t _arc) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > > + uint32_t iter; > > + > > + if (!__feature_arc_main) > > + return -1; > > + > > + for (iter = 0; iter < dm->max_feature_arcs; iter++) { > > + if (dm->feature_arcs[iter] == > > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > > + continue; > > + > > + if (arc == (rte_graph_feature_arc_get(dm- > > >feature_arcs[iter]))) > > + return 0; > > + } > > + return -1; > > +} > > + > > +static int > > +get_existing_edge(const char *arc_name, struct rte_node_register > > *parent_node, > > + struct rte_node_register *child_node, rte_edge_t *_edge) > > +{ > > + char **next_edges = NULL; > > + uint32_t count, i; > > + > > + RTE_SET_USED(arc_name); > > + > > + count = rte_node_edge_get(parent_node->id, NULL); > > + next_edges = malloc(count); > > + > > + if (!next_edges) > > + return -1; > > + > > + count = rte_node_edge_get(parent_node->id, next_edges); > > + for (i = 0; i < count; i++) { > > + if (strstr(child_node->name, next_edges[i])) { > > + feat_dbg("%s: Edge exists [%s[%u]: \"%s\"]", > > arc_name, > > + parent_node->name, i, child_node->name); > > + if (_edge) > > + *_edge = (rte_edge_t)i; > > + > > + free(next_edges); > > + return 0; > > + } > > + } > > + free(next_edges); > > + > > + return -1; > > +} > > + > > +static int > > +connect_graph_nodes(struct rte_node_register *parent_node, struct > > rte_node_register *child_node, > > + rte_edge_t *_edge, char *arc_name) > > +{ > > + const char *next_node = NULL; > > + rte_edge_t edge; > > + > > + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { > > + feat_dbg("%s: add_feature: Edge reused [%s[%u]: \"%s\"]", > > arc_name, > > + parent_node->name, edge, child_node->name); > > + > > + if (_edge) > > + *_edge = edge; > > + > > + return 0; > > + } > > + > > + /* Node to be added */ > > + next_node = child_node->name; > > + > > + edge = rte_node_edge_update(parent_node->id, > > RTE_EDGE_ID_INVALID, &next_node, 1); > > + > > + if (edge == RTE_EDGE_ID_INVALID) { > > + graph_err("edge invalid"); > > + return -1; > > + } > > + edge = rte_node_edge_count(parent_node->id) - 1; > > + > > + feat_dbg("%s: add_feature: edge added [%s[%u]: \"%s\"]", arc_name, > > parent_node->name, edge, > > + child_node->name); > > + > > + if (_edge) > > + *_edge = edge; > > + > > + return 0; > > +} > > + > > +static int > > +feature_arc_init(rte_graph_feature_arc_main_t **pfl, uint32_t > > max_feature_arcs) > > +{ > > + rte_graph_feature_arc_main_t *pm = NULL; > > + uint32_t i; > > + size_t sz; > > + > > + if (!pfl) > > + return -1; > > + > > + sz = sizeof(rte_graph_feature_arc_main_t) + > > + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); > > + > > + pm = malloc(sz); > > + if (!pm) > > + return -1; > > + > > + memset(pm, 0, sz); > > + > > + for (i = 0; i < max_feature_arcs; i++) > > + pm->feature_arcs[i] = > > RTE_GRAPH_FEATURE_ARC_INITIALIZER; > > + > > + pm->max_feature_arcs = max_feature_arcs; > > + > > + *pfl = pm; > > + > > + return 0; > > +} > > + > > +int > > +rte_graph_feature_arc_init(int max_feature_arcs) > > +{ > > + if (!max_feature_arcs) > > + return -1; > > + > > + if (__feature_arc_main) > > + return -1; > > + > > + return feature_arc_init(&__feature_arc_main, max_feature_arcs); > > +} > > + > > +static void > > +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t > list_index) > > +{ > > + rte_graph_feature_data_t *fdata = NULL; > > + rte_graph_feature_list_t *list = NULL; > > + struct rte_graph_feature *feat = NULL; > > + uint32_t i, j; > > + > > + list = arc->feature_list[list_index]; > > + feat = arc->features[list_index]; > > + > > + /*Initialize variables*/ > > + memset(feat, 0, arc->feature_size); > > + memset(list, 0, arc->feature_list_size); > > + > > + /* Initialize feature and feature_data */ > > + for (i = 0; i < arc->max_features; i++) { > > + feat = __rte_graph_feature_get(arc, i, list_index); > > + feat->this_feature_index = i; > > + > > + for (j = 0; j < arc->max_indexes; j++) { > > + fdata = rte_graph_feature_data_get(arc, feat, j); > > + fdata->next_enabled_feature = > > RTE_GRAPH_FEATURE_INVALID; > > + fdata->next_edge = UINT16_MAX; > > + fdata->user_data = UINT32_MAX; > > + } > > + } > > + > > + for (i = 0; i < arc->max_indexes; i++) > > + list->first_enabled_feature_by_index[i] = > > RTE_GRAPH_FEATURE_INVALID; > > +} > > + > > +static int > > +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char > > *flist_name, > > + rte_graph_feature_list_t **pplist, > > + struct rte_graph_feature **ppfeature, uint32_t > > list_index) > > +{ > > + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; > > + size_t list_size, feat_size, fdata_size; > > + rte_graph_feature_list_t *list = NULL; > > + struct rte_graph_feature *feat = NULL; > > + > > + list_size = sizeof(list->first_enabled_feature_by_index[0]) * arc- > > >max_indexes; > > + > > + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); > > + if (!list) > > + return -ENOMEM; > > + > > + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); > > + > > + /* Let one feature capture complete cache lines */ > > + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + > > fdata_size, > > + RTE_CACHE_LINE_SIZE); > > + > > + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, > > "feat"); > > + > > + feat = rte_malloc(fname, feat_size * arc->max_features, > > RTE_CACHE_LINE_SIZE); > > + if (!feat) { > > + rte_free(list); > > + return -ENOMEM; > > + } > > + arc->feature_size = feat_size; > > + arc->feature_data_size = fdata_size; > > + arc->feature_list_size = list_size; > > + > > + /* Initialize list */ > > + list->indexed_by_features = feat; > > + *pplist = list; > > + *ppfeature = feat; > > + > > + feature_arc_list_reset(arc, list_index); > > + > > + return 0; > > +} > > + > > +static void > > +feature_arc_list_destroy(rte_graph_feature_list_t *list) > > +{ > > + rte_free(list->indexed_by_features); > Do you need to free individual rte_graph_feature here, that is allocated in > arc_list_init? Nitin> It seems correct to me. feature_arc_list_destroy() frees all memory allocated in feature_arc_list_init(). So feature_arc_list_destroy() is calling rte_free() for every rte_malloc() happened in feature_arc_list_init(). To make it clear, I have refactor the function and has added appropriate comment from v2 patch set onwards > > > + rte_free(list); > > +} > > + > > +int > > +rte_graph_feature_arc_create(const char *feature_arc_name, int > > max_features, int max_indexes, > > + struct rte_node_register *start_node, > > rte_graph_feature_arc_t *_arc) > > +{ > > + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; > > + rte_graph_feature_arc_main_t *dfm = NULL; > > + struct rte_graph_feature_arc *arc = NULL; > > + struct rte_graph_feature_data *gfd = NULL; > > + struct rte_graph_feature *df = NULL; > > + uint32_t iter, j, arc_index; > > + size_t sz; > > + > > + if (!_arc) > > + return -1; > > + > > + if (max_features < 2) > > + return -1; > > + > > + if (!start_node) > > + return -1; > > + > > + if (!feature_arc_name) > > + return -1; > > + > > + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) { > > + graph_err("Invalid max features: %u", max_features); > > + return -1; > > + } > > + > > + /* > > + * Application hasn't called rte_graph_feature_arc_init(). Initialize > with > > + * default values > > + */ > > + if (!__feature_arc_main) { > > + if > > (rte_graph_feature_arc_init((int)__RTE_GRAPH_FEATURE_ARC_MAX) < 0) { > > + graph_err("rte_graph_feature_arc_init() failed"); > > + return -1; > > + } > > + } > > + > > + dfm = __feature_arc_main; > > + > > + /* threshold check */ > > + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) { > > + graph_err("max threshold for num_feature_arcs: %d > > reached", > > + dfm->max_feature_arcs - 1); > > + return -1; > > + } > > + /* Find the free slot for feature arc */ > > + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { > > + if (dfm->feature_arcs[iter] == > > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > > + break; > > + } > > + arc_index = iter; > > + > > + if (arc_index >= dfm->max_feature_arcs) { > > + graph_err("No free slot found for num_feature_arc"); > > + return -1; > > + } > > + > > + /* This should not happen */ > > + RTE_VERIFY(dfm->feature_arcs[arc_index] == > > RTE_GRAPH_FEATURE_ARC_INITIALIZER); > > + > > + /* size of feature arc + feature_bit_mask_by_index */ > > + sz = sizeof(*arc) + (sizeof(uint64_t) * max_indexes); > > + > > + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); > > + > > + if (!arc) { > > + graph_err("malloc failed for feature_arc_create()"); > > + return -1; > > + } > > + > > + memset(arc, 0, sz); > > + > > + /* Initialize rte_graph port group fixed variables */ > > + STAILQ_INIT(&arc->all_features); > > + strncpy(arc->feature_arc_name, feature_arc_name, > > RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); > > + arc->feature_arc_main = (void *)dfm; > > + arc->start_node = start_node; > > + arc->max_features = max_features; > > + arc->max_indexes = max_indexes; > > + > > + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); > > + > > + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc- > > >features[0], 0) < 0) { > > + rte_free(arc); > > + graph_err("feature_arc_list_init(0) failed"); > > + return -1; > > + } > > + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); > > + > > + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc- > > >features[1], 1) < 0) { > > + feature_arc_list_destroy(arc->feature_list[0]); > > + graph_err("feature_arc_list_init(1) failed"); > > + return -1; > > + } > > + > > + for (iter = 0; iter < arc->max_features; iter++) { > > + df = rte_graph_feature_get(arc, iter); > > + for (j = 0; j < arc->max_indexes; j++) { > > + gfd = rte_graph_feature_data_get(arc, df, j); > > + gfd->next_enabled_feature = > > RTE_GRAPH_FEATURE_INVALID; > > + } > > + } > > + arc->feature_arc_index = arc_index; > > + dfm->feature_arcs[arc->feature_arc_index] = > > (rte_graph_feature_arc_t)arc; > > + dfm->num_feature_arcs++; > > + > > + if (_arc) > > + *_arc = (rte_graph_feature_arc_t)arc; > > + > > + return 0; > > +} > > + > > +int > > +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct > > rte_node_register *feature_node, > > + const char *after_feature, const char *before_feature) > > +{ > > + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo > > = NULL; > > + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + uint32_t slot, add_flag; > > + rte_edge_t edge = -1; > > + > > + RTE_VERIFY(arc->feature_arc_main == __feature_arc_main); > > + > > + if (feature_arc_lookup(_arc)) { > > + graph_err("invalid feature arc: 0x%016" PRIx64, > > (uint64_t)_arc); > > + return -1; > > + } > > + > > + if (arc->runtime_enabled_features) { > > + graph_err("adding features after enabling any one of them is > > not supported"); > > + return -1; > > + } > > + > > + if ((after_feature != NULL) && (before_feature != NULL) && > > + (after_feature == before_feature)) { > > + graph_err("after_feature and before_feature are same > > '%s:%s]", after_feature, > > + before_feature); > > + return -1; > > + } > > + > > + if (!feature_node) { > > + graph_err("feature_node: %p invalid", feature_node); > > + return -1; > > + } > > + > > + arc = rte_graph_feature_arc_get(_arc); > > + > > + if (feature_node->id == RTE_NODE_ID_INVALID) { > > + graph_err("Invalid node: %s", feature_node->name); > > + return -1; > > + } > > + > > + if (!feature_lookup(arc, feature_node->name, &finfo, &slot)) { > > + graph_err("%s feature already added", feature_node->name); > > + return -1; > > + } > > + > > + if (slot >= RTE_GRAPH_FEATURE_MAX_PER_ARC) { > > + graph_err("Max slot %u reached for feature addition", slot); > > + return -1; > > + } > > + > > + if (strstr(feature_node->name, arc->start_node->name)) { > > + graph_err("Feature %s cannot point to itself: %s", > > feature_node->name, > > + arc->start_node->name); > > + return -1; > > + } > > + > > + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc- > > >feature_arc_name)) { > > + graph_err("unable to connect %s -> %s", arc->start_node- > > >name, feature_node->name); > > + return -1; > > + } > > + > > + finfo = malloc(sizeof(*finfo)); > > + if (!finfo) > > + return -1; > > + > > + memset(finfo, 0, sizeof(*finfo)); > > + > > + finfo->feature_arc = (void *)arc; > > + finfo->feature_node = feature_node; > > + finfo->edge_to_this_feature = edge; > > + > > + /* Check for before and after constraints */ > > + if (before_feature) { > > + /* before_feature sanity */ > > + if (feature_lookup(arc, before_feature, &before_finfo, NULL)) > > + SET_ERR_JMP(EINVAL, finfo_free, > > + "Invalid before feature name: %s", > > before_feature); > > + > > + if (!before_finfo) > > + SET_ERR_JMP(EINVAL, finfo_free, > > + "before_feature %s does not exist", > > before_feature); > > + > > + /* > > + * Starting from 0 to before_feature, continue connecting > > edges > > + */ > > + add_flag = 1; > > + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { > > + /* > > + * As soon as we see before_feature. stop adding > > edges > > + */ > > + if (!strncmp(temp->feature_node->name, > > before_feature, > > + RTE_GRAPH_NAMESIZE)) > > + if (!connect_graph_nodes(finfo- > > >feature_node, temp->feature_node, > > + &edge, arc- > > >feature_arc_name)) > > + add_flag = 0; > > + > > + if (add_flag) > > + connect_graph_nodes(temp->feature_node, > > finfo->feature_node, NULL, > > + arc->feature_arc_name); > > + } > > + } > > + > > + if (after_feature) { > > + if (feature_lookup(arc, after_feature, &after_finfo, NULL)) > > + SET_ERR_JMP(EINVAL, finfo_free, > > + "Invalid after feature_name %s", > > after_feature); > > + > > + if (!after_finfo) > > + SET_ERR_JMP(EINVAL, finfo_free, > > + "after_feature %s does not exist", > > after_feature); > > + > > + /* Starting from after_feature to end continue connecting > > edges */ > > + add_flag = 0; > > + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { > > + /* We have already seen after_feature now */ > > + if (add_flag) > > + /* Add all features as next node to current > > feature*/ > > + connect_graph_nodes(finfo->feature_node, > > temp->feature_node, NULL, > > + arc->feature_arc_name); > > + > > + /* as soon as we see after_feature. start adding edges > > + * from next iteration > > + */ > > + if (!strncmp(temp->feature_node->name, > > after_feature, RTE_GRAPH_NAMESIZE)) > > + /* connect after_feature to this feature */ > > + if (!connect_graph_nodes(temp- > > >feature_node, finfo->feature_node, > > + &edge, arc- > > >feature_arc_name)) > > + add_flag = 1; > > + } > > + > > + /* add feature next to after_feature */ > > + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, > > next_feature); > > + } else { > > + if (before_finfo) { > > + after_finfo = NULL; > > + STAILQ_FOREACH(temp, &arc->all_features, > > next_feature) { > > + if (before_finfo == temp) { > > + if (after_finfo) > > + STAILQ_INSERT_AFTER(&arc- > > >all_features, after_finfo, > > + finfo, > > next_feature); > > + else > > + STAILQ_INSERT_HEAD(&arc- > > >all_features, finfo, > > + > > next_feature); > > + > > + return 0; > > + } > > + after_finfo = temp; > > + } > > + } else { > > + STAILQ_INSERT_TAIL(&arc->all_features, finfo, > > next_feature); > > + } > > + } > > + > > + return 0; > > + > > +finfo_free: > > + free(finfo); > > + > > + return -1; > > +} > > + > > +int > > +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char > > *feature_name, > > + rte_graph_feature_t *feat) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + struct rte_graph_feature_node_list *finfo = NULL; > > + uint32_t slot; > > + > > + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { > > + *feat = (rte_graph_feature_t) slot; > > + return 0; > > + } > > + > > + return -1; > > +} > > + > > +int > > +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, > > const char *feature_name, > > + int is_enable_disable) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + struct rte_graph_feature_node_list *finfo = NULL; > > + struct rte_graph_feature *gf = NULL; > > + uint32_t slot; > > + > > + /* validate _arc */ > > + if (arc->feature_arc_main != __feature_arc_main) { > > + graph_err("invalid feature arc: 0x%016" PRIx64, > > (uint64_t)_arc); > > + return -EINVAL; > > + } > > + > > + /* validate index */ > > + if (index >= arc->max_indexes) { > > + graph_err("%s: Invalid provided index: %u >= %u configured", > > arc->feature_arc_name, > > + index, arc->max_indexes); > > + return -1; > > + } > > + > > + /* validate feature_name is already added or not */ > > + if (feature_lookup(arc, feature_name, &finfo, &slot)) { > > + graph_err("%s: No feature %s added", arc- > > >feature_arc_name, feature_name); > > + return -EINVAL; > > + } > > + > > + if (!finfo) { > > + graph_err("%s: No feature: %s found", arc- > > >feature_arc_name, feature_name); > > + return -EINVAL; > > + } > > + > > + /* slot should be in valid range */ > > + if (slot >= arc->max_features) { > > + graph_err("%s/%s: Invalid free slot %u(max=%u) for feature", > > arc->feature_arc_name, > > + feature_name, slot, arc->max_features); > > + return -EINVAL; > > + } > > + > > + /* slot should be in range of 0 - 63 */ > > + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { > > + graph_err("%s/%s: Invalid slot: %u", arc->feature_arc_name, > > + feature_name, slot); > > + return -EINVAL; > > + } > > + > > + if (finfo->node_index != slot) { > > + graph_err("%s/%s: feature lookup slot mismatch with finfo > > index: %u and lookup slot: %u", > > + arc->feature_arc_name, feature_name, finfo- > > >node_index, slot); > > + return -1; > > + } > > + > > + /* Get feature from active list */ > > + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(arc)); > > + if (gf->this_feature_index != slot) { > > + graph_err("%s: %s received feature_index: %u does not > match > > with saved feature_index: %u", > > + arc->feature_arc_name, feature_name, slot, gf- > > >this_feature_index); > > + return -1; > > + } > > + > > + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & > > + RTE_BIT64(slot))) { > > + graph_err("%s: %s already enabled on index: %u", > > + arc->feature_arc_name, feature_name, index); > > + return -1; > > + } > > + > > + if (!is_enable_disable && !arc->runtime_enabled_features) { > > + graph_err("%s: No feature enabled to disable", arc- > > >feature_arc_name); > > + return -1; > > + } > > + > > + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & > > RTE_BIT64(slot))) { > > + graph_err("%s: %s not enabled in bitmask for index: %u", > > + arc->feature_arc_name, feature_name, index); > > + return -1; > > + } > > + > > + return 0; > > +} > > + > > +static void > > +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t > > dest_list_index, > > + uint16_t src_list_index) > > +{ > > + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; > > + struct rte_graph_feature *sgf = NULL, *dgf = NULL; > > + uint32_t i, j; > > + > > + for (i = 0; i < arc->max_features; i++) { > > + sgf = __rte_graph_feature_get(arc, i, src_list_index); > > + dgf = __rte_graph_feature_get(arc, i, dest_list_index); > > + for (j = 0; j < arc->max_indexes; j++) { > > + sgfd = rte_graph_feature_data_get(arc, sgf, j); > > + dgfd = rte_graph_feature_data_get(arc, dgf, j); > > + dgfd->user_data = sgfd->user_data; > > + } > > + } > > +} > > + > > +static void > > +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t > > list_index) > > +{ > > + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; > > + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; > > + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; > > + rte_graph_feature_list_t *flist = NULL; > > + uint32_t fi, di, prev_fi; > > + uint64_t bitmask; > > + rte_edge_t edge; > > + > > + flist = arc->feature_list[list_index]; > > + > > + for (di = 0; di < arc->max_indexes; di++) { > > + bitmask = arc->feature_bit_mask_by_index[di]; > > + prev_fi = RTE_GRAPH_FEATURE_INVALID; > > + /* for each feature set for index, set fast path data */ > > + while (rte_bsf64_safe(bitmask, &fi)) { > > + gf = __rte_graph_feature_get(arc, fi, list_index); > > + gfd = rte_graph_feature_data_get(arc, gf, di); > > + feature_arc_node_info_lookup(arc, fi, &finfo); > > + > > + /* If previous feature_index was valid in last loop */ > > + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { > > + prev_gf = __rte_graph_feature_get(arc, > > prev_fi, list_index); > > + prev_gfd = rte_graph_feature_data_get(arc, > > prev_gf, di); > > + /* > > + * Get edge of previous feature node > > connecting to this feature node > > + */ > > + feature_arc_node_info_lookup(arc, prev_fi, > > &prev_finfo); > > + if (!get_existing_edge(arc->feature_arc_name, > > + prev_finfo->feature_node, > > + finfo->feature_node, > > &edge)) { > > + feat_dbg("[%s/%s(%2u)/idx:%2u]: > > %s[%u] = %s", > > + arc->feature_arc_name, > > + prev_finfo->feature_node- > > >name, prev_fi, di, > > + prev_finfo->feature_node- > > >name, > > + edge, finfo->feature_node- > > >name); > > + /* Copy feature index for next > > iteration*/ > > + gfd->next_edge = edge; > > + prev_fi = fi; > > + /* > > + * Fill current feature as next enabled > > + * feature to previous one > > + */ > > + prev_gfd->next_enabled_feature = fi; > > + } else { > > + /* Should not fail */ > > + RTE_VERIFY(0); > > + } > > + } > > + /* On first feature edge of the node to be added */ > > + if (fi == rte_bsf64(arc- > > >feature_bit_mask_by_index[di])) { > > + if (!get_existing_edge(arc->feature_arc_name, > > arc->start_node, > > + finfo->feature_node, > > + &edge)) { > > + feat_dbg("[%s/%s/%2u/idx:%2u]: 1st > > feat %s[%u] = %s", > > + arc->feature_arc_name, > > + arc->start_node->name, fi, di, > > + arc->start_node->name, > > edge, > > + finfo->feature_node->name); > > + /* Copy feature index for next > > iteration*/ > > + gfd->next_edge = edge; > > + prev_fi = fi; > > + /* Set first feature set array for > > index*/ > > + flist- > > >first_enabled_feature_by_index[di] = fi; > > + } else { > > + /* Should not fail */ > > + RTE_VERIFY(0); > > + } > > + } > > + /* Clear current feature index */ > > + bitmask &= ~RTE_BIT64(fi); > > + } > > + } > > +} > > + > > +int > > +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, > > const > > + char *feature_name, int32_t user_data) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + struct rte_graph_feature_node_list *finfo = NULL; > > + struct rte_graph_feature_data *gfd = NULL; > > + rte_graph_feature_rt_list_t passive_list; > > + struct rte_graph_feature *gf = NULL; > > + uint64_t fp_bitmask; > > + uint32_t slot; > > + > > + if (rte_graph_feature_validate(_arc, index, feature_name, 1)) > > + return -1; > > + > > + /** This should not fail as validate() has passed */ > > + if (feature_lookup(arc, feature_name, &finfo, &slot)) > > + RTE_VERIFY(0); > > + > > + if (!arc->runtime_enabled_features) > > + prepare_feature_arc(arc); > > + > > + passive_list = ARC_PASSIVE_LIST(arc); > > + > > + gf = __rte_graph_feature_get(arc, slot, passive_list); > > + gfd = rte_graph_feature_data_get(arc, gf, index); > > + > > + feat_dbg("%s/%s: Enabling feature on list: %u for index: %u at feature > > slot %u", > > + arc->feature_arc_name, feature_name, passive_list, index, > > slot); > > + > > + /* Reset feature list */ > > + feature_arc_list_reset(arc, passive_list); > > + > > + /* Copy user-data */ > > + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); > > + > > + /* Set current user-data */ > > + gfd->user_data = user_data; > > + > > + /* Set bitmask in control path bitmask */ > > + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc- > > >feature_bit_mask_by_index[index]); > > + refill_feature_fastpath_data(arc, passive_list); > > + > > + /* Set fast path enable bitmask */ > > + fp_bitmask = __atomic_load_n(&arc- > > >feature_enable_bitmask[passive_list], __ATOMIC_RELAXED); > > + fp_bitmask |= RTE_BIT64(slot); > > + __atomic_store(&arc->feature_enable_bitmask[passive_list], > > &fp_bitmask, __ATOMIC_RELAXED); > > + > > + /* Slow path updates */ > > + arc->runtime_enabled_features++; > > + > > + /* Increase feature node info reference count */ > > + finfo->ref_count++; > > + > > + /* Store release semantics for active_list update */ > > + __atomic_store(&arc->active_feature_list, &passive_list, > > __ATOMIC_RELEASE); > > + > > + return 0; > > +} > > + > > +int > > +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, > > const char *feature_name) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + struct rte_graph_feature_data *gfd = NULL; > > + struct rte_graph_feature_node_list *finfo = NULL; > > + rte_graph_feature_rt_list_t passive_list; > > + struct rte_graph_feature *gf = NULL; > > + uint32_t slot; > > + > > + if (rte_graph_feature_validate(_arc, index, feature_name, 0)) > > + return -1; > > + > > + if (feature_lookup(arc, feature_name, &finfo, &slot)) > > + return -1; > > + > > + passive_list = ARC_PASSIVE_LIST(arc); > > + > > + gf = __rte_graph_feature_get(arc, slot, passive_list); > > + gfd = rte_graph_feature_data_get(arc, gf, index); > > + > > + feat_dbg("%s/%s: Disabling feature for index: %u at feature slot %u", > > arc->feature_arc_name, > > + feature_name, index, slot); > > + > > + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc- > > >feature_bit_mask_by_index[index]); > > + > > + /* Set fast path enable bitmask */ > > + arc->feature_enable_bitmask[passive_list] &= ~(RTE_BIT64(slot)); > > + > > + /* Reset feature list */ > > + feature_arc_list_reset(arc, passive_list); > > + > > + /* Copy user-data */ > > + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); > > + > > + /* Reset current user-data */ > > + gfd->user_data = ~0; > > + > > + refill_feature_fastpath_data(arc, passive_list); > > + > > + finfo->ref_count--; > > + arc->runtime_enabled_features--; > > + > > + /* Store release semantics for active_list update */ > > + __atomic_store(&arc->active_feature_list, &passive_list, > > __ATOMIC_RELEASE); > > + > > + return 0; > > +} > > + > > +int > > +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > > + struct rte_graph_feature_node_list *node_info = NULL; > > + > > + while (!STAILQ_EMPTY(&arc->all_features)) { > > + node_info = STAILQ_FIRST(&arc->all_features); > > + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); > > + free(node_info); > > + } > > + feature_arc_list_destroy(arc->feature_list[0]); > > + feature_arc_list_destroy(arc->feature_list[1]); > > + rte_free(arc->features[0]); > > + rte_free(arc->features[1]); > > + > > + dm->feature_arcs[arc->feature_arc_index] = > > RTE_GRAPH_FEATURE_ARC_INITIALIZER; > > + > > + rte_free(arc); > > + return 0; > > +} > > + > > +int > > +rte_graph_feature_arc_cleanup(void) > > +{ > > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > > + uint32_t iter; > > + > > + if (!__feature_arc_main) > > + return -1; > > + > > + for (iter = 0; iter < dm->max_feature_arcs; iter++) { > > + if (dm->feature_arcs[iter] == > > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > > + continue; > > + > > + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm- > > >feature_arcs[iter]); > > + } > > + free(dm); > > + > > + __feature_arc_main = NULL; > > + > > + return 0; > > +} > > + > > +int > > +rte_graph_feature_arc_lookup_by_name(const char *arc_name, > > rte_graph_feature_arc_t *_arc) > > +{ > > + rte_graph_feature_arc_main_t *dm = __feature_arc_main; > > + struct rte_graph_feature_arc *arc = NULL; > > + uint32_t iter; > > + > > + if (!__feature_arc_main) > > + return -1; > > + > > + for (iter = 0; iter < dm->max_feature_arcs; iter++) { > > + if (dm->feature_arcs[iter] == > > RTE_GRAPH_FEATURE_ARC_INITIALIZER) > > + continue; > > + > > + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); > > + > > + if (strstr(arc_name, arc->feature_arc_name)) { > > + if (_arc) > > + *_arc = (rte_graph_feature_arc_t)arc; > > + return 0; > > + } > > + } > > + > > + return -1; > > +} > > + > > +int > > +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t > > _arc) > > +{ > > + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); > > + > > + return arc->runtime_enabled_features; > > +} > > + > > + > > diff --git a/lib/graph/meson.build b/lib/graph/meson.build > > index 0cb15442ab..d916176fb7 100644 > > --- a/lib/graph/meson.build > > +++ b/lib/graph/meson.build > > @@ -14,11 +14,13 @@ sources = files( > > 'graph_debug.c', > > 'graph_stats.c', > > 'graph_populate.c', > > + 'graph_feature_arc.c', > > 'graph_pcap.c', > > 'rte_graph_worker.c', > > 'rte_graph_model_mcore_dispatch.c', > > ) > > headers = files('rte_graph.h', 'rte_graph_worker.h') > > +headers += files('rte_graph_feature_arc.h', > 'rte_graph_feature_arc_worker.h') > > indirect_headers += files( > > 'rte_graph_model_mcore_dispatch.h', > > 'rte_graph_model_rtc.h', > > diff --git a/lib/graph/rte_graph_feature_arc.h > > b/lib/graph/rte_graph_feature_arc.h > > new file mode 100644 > > index 0000000000..e3bf4eb73d > > --- /dev/null > > +++ b/lib/graph/rte_graph_feature_arc.h > > @@ -0,0 +1,373 @@ > > +/* SPDX-License-Identifier: BSD-3-Clause > > + * Copyright(C) 2024 Marvell International Ltd. > > + */ > > + > > +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ > > +#define _RTE_GRAPH_FEATURE_ARC_H_ > > + > > +#include <assert.h> > > +#include <errno.h> > > +#include <signal.h> > > +#include <stddef.h> > > +#include <stdint.h> > > +#include <stdio.h> > > +#include <stdlib.h> > > +#include <string.h> > > + > > +#include <rte_common.h> > > +#include <rte_compat.h> > > +#include <rte_debug.h> > > +#include <rte_graph.h> > > +#include <rte_graph_worker.h> > > + > > +#ifdef __cplusplus > > +extern "C" { > > +#endif > > + > > +/** > > + * @file > > + * > > + * rte_graph_feature_arc.h > > + * > > + * Define APIs and structures/variables with respect to feature arc > > + * > > + * - Feature arc(s) > > + * - Feature(s) > > + * > > + * A feature arc represents an ordered list of features/protocol-nodes at a > > + * given networking layer. Feature arc provides a high level abstraction to > > + * connect various *rte_graph* nodes, designated as *feature nodes*, and > > + * allowing steering of packets across these feature nodes fast path > > processing > > + * in a generic manner. In a typical network stack, often a protocol or > feature > > + * must be first enabled on a given interface, before any packet is steered > > + * towards it for feature processing. For eg: incoming IPv4 packets are sent > to > > + * routing sub-system only after a valid IPv4 address is assigned to the > > + * received interface. In other words, often packets needs to be steered > across > > + * features not based on the packet content but based on whether a > feature is > > + * enable or disable on a given incoming/outgoing interface. Feature arc > > + * provides mechanism to enable/disable feature(s) on each interface at > > runtime > > + * and allow seamless packet steering across runtime enabled feature > nodes > > in > > + * fast path. > > + * > > + * Feature arc also provides a way to steer packets from standard nodes to > > + * custom/user-defined *feature nodes* without any change in standard > > node's > > + * fast path functions > > + * > > + * On a given interface multiple feature(s) might be enabled in a particular > > + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" > > + * features may be enabled on "eth0" interface in "L3-output" feature arc. > > + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" > > + * interface in same "L3-output" feature arc. > > + * > > + * When multiple features are present in a given feature arc, its imperative > > + * to allow each feature processing in a particular sequential order. For > > + * instance, in "L3-input" feature arc it may be required to run "IPsec > > + * input" feature first, for packet decryption, before "ip-lookup". So a > > + * sequential order must be maintained among features present in a > feature > > arc. > > + * > > + * Features are enabled/disabled multiple times at runtime to some or all > > + * available interfaces present in the system. Features can be > > enabled/disabled > > + * even after @b rte_graph_create() is called. Enable/disabling features on > > one > > + * interface is independent of other interface. > > + * > > + * A given feature might consume packet (if it's configured to consume) or > > may > > + * forward it to next enabled feature. For instance, "IPsec input" feature > may > > + * consume/drop all packets with "Protect" policy action while all packets > with > > + * policy action as "Bypass" may be forwarded to next enabled feature > (with > > in > > + * same feature arc) > > + * > > + * This library facilitates rte graph based applications to steer packets in > > + * fast path to different feature nodes with-in a feature arc and support all > > + * functionalities described above > > + * > > + * In order to use feature-arc APIs, applications needs to do following in > > + * control path: > > + * - Initialize feature arc library via rte_graph_feature_arc_init() > > + * - Create feature arc via rte_graph_feature_arc_create() > > + * - *Before calling rte_graph_create()*, features must be added to feature- > > arc > > + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding > > + * features in a sequential order with "runs_after" and "runs_before" > > + * constraints. > > + * - Post rte_graph_create(), features can be enabled/disabled at runtime > on > > + * any interface via > rte_graph_feature_enable()/rte_graph_feature_disable() > > + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() > > + * > > + * In fast path, APIs are provided to steer packets towards feature path > from > > + * - start_node (provided as an argument to > rte_graph_feature_arc_create()) > > + * - feature nodes (which are added via rte_graph_feature_add()) > > + * > > + * For typical steering of packets across feature nodes, application required > > + * to know "rte_edges" which are saved in feature data object. Feature > data > > + * object is unique for every interface per feature with in a feature arc. > > + * > > + * When steering packets from start_node to feature node: > > + * - rte_graph_feature_arc_first_feature_get() provides first enabled > feature. > > + * - Next rte_edge from start_node to first enabled feature can be obtained > > via > > + * rte_graph_feature_arc_feature_set() > > + * > > + * rte_mbuf can carry [current feature, index] from start_node of an arc to > > other > > + * feature nodes > > + * > > + * In feature node, application can get 32-bit user_data > > + * via_rte_graph_feature_user_data_get() which is provided in > > + * rte_graph_feature_enable(). User data can hold feature specific cookie > like > > + * IPsec policy database index (if more than one are supported) > > + * > > + * If feature node is not consuming packet, next enabled feature and next > > + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() > > + * > > + * It is application responsibility to ensure that at-least *last feature*(or > sink > > + * feature) must be enabled from where packet can exit feature-arc path, if > > + * *NO* intermediate feature is consuming the packet and it has reached > till > > + * the end of feature arc path > > + * > > + * Synchronization among cores > > + * --------------------------- > > + * Subsequent calls to rte_graph_feature_enable() is allowed while worker > > cores > > + * are processing in rte_graph_walk() loop. However, for > > + * rte_graph_feature_disable() application must use RCU based > > synchronization > > + */ > > + > > +/**< Initializer value for rte_graph_feature_arc_t */ > > +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER > > ((rte_graph_feature_arc_t)UINT64_MAX) > > + > > +/** Max number of features supported in a given feature arc */ > > +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 > > + > > +/** Length of feature arc name */ > > +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE > > + > > +/** @internal */ > > +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) > > + > > +/**< Initializer value for rte_graph_feature_arc_t */ > > +#define RTE_GRAPH_FEATURE_INVALID > > rte_graph_feature_cast(UINT8_MAX) > > + > > +/** rte_graph feature arc object */ > > +typedef uint64_t rte_graph_feature_arc_t; > > + > > +/** rte_graph feature object */ > > +typedef uint8_t rte_graph_feature_t; > > + > > +/** runtime active feature list index with in feature arc*/ > > +typedef uint8_t rte_graph_feature_rt_list_t; > > + > > +/** per feature arc monotonically increasing counter to synchronize fast > path > > APIs */ > > +typedef uint16_t rte_graph_feature_counter_t; > > + > > +/** > > + * Initialize feature arc subsystem > > + * > > + * @param max_feature_arcs > > + * Maximum number of feature arcs required to be supported > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_arc_init(int max_feature_arcs); > > + > > +/** > > + * Create a feature arc > > + * > > + * @param feature_arc_name > > + * Feature arc name with max length of @ref > > RTE_GRAPH_FEATURE_ARC_NAMELEN > > + * @param max_features > > + * Maximum number of features to be supported in this feature arc > > + * @param max_indexes > > + * Maximum number of interfaces/ports/indexes to be supported > > + * @param start_node > > + * Base node where this feature arc's features are checked in fast path > > + * @param[out] _arc > > + * Feature arc object > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_arc_create(const char *feature_arc_name, int > > max_features, int max_indexes, > > + struct rte_node_register *start_node, > > + rte_graph_feature_arc_t *_arc); > > + > > +/** > > + * Get feature arc object with name > > + * > > + * @param arc_name > > + * Feature arc name provided to successful @ref > > rte_graph_feature_arc_create > > + * @param[out] _arc > > + * Feature arc object returned > > + * > > + * @return > > + * 0: Success > > + * <0: Failure. > > + */ > > +__rte_experimental > > +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, > > rte_graph_feature_arc_t *_arc); > > + > > +/** > > + * Add a feature to already created feature arc. For instance > > + * > > + * 1. Add first feature node: "ipv4-input" to input arc > > + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); > > + * > > + * 2. Add "ipsec-input" feature node after "ipv4-input" node > > + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", > > NULL); > > + * > > + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" node > > + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", > NULL, > > "ipv4-input"); > > + * > > + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input > > + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4- > input", > > "ipsec-input"); > > + * > > + * @param _arc > > + * Feature arc handle returned from @ref rte_graph_feature_arc_create() > > + * @param feature_node > > + * Graph node representing feature. On success, feature_node is > next_node > > of > > + * feature_arc->start_node > > + * @param runs_after > > + * Add this feature_node after already added "runs_after". Creates > > + * start_node -> runs_after -> this_feature sequence > > + * @param runs_before > > + * Add this feature_node before already added "runs_before". Creates > > + * start_node -> this_feature -> runs_before sequence > > + * > > + * <I> Must be called before rte_graph_create() </I> > > + * <I> rte_graph_feature_add() is not allowed after call to > > + * rte_graph_feature_enable() so all features must be added before they > can > > be > > + * enabled </I> > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct > > rte_node_register *feature_node, > > + const char *runs_after, const char *runs_before); > > + > > +/** > > + * Enable feature within a feature arc > > + * > > + * Must be called after @b rte_graph_create(). > > + * > > + * @param _arc > > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > > @ref > > + * rte_graph_feature_arc_lookup_by_name > > + * @param index > > + * Application specific index. Can be corresponding to > interface_id/port_id > > etc > > + * @param feature_name > > + * Name of the node which is already added via @ref > rte_graph_feature_add > > + * @param user_data > > + * Application specific data which is retrieved in fast path > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t > index, > > const char *feature_name, > > + int32_t user_data); > > + > > +/** > > + * Validate whether subsequent enable/disable feature would succeed or > not. > > + * API is thread-safe > > + * > > + * @param _arc > > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > > @ref > > + * rte_graph_feature_arc_lookup_by_name > > + * @param index > > + * Application specific index. Can be corresponding to > interface_id/port_id > > etc > > + * @param feature_name > > + * Name of the node which is already added via @ref > rte_graph_feature_add > > + * @param is_enable_disable > > + * If 1, validate whether subsequent @ref rte_graph_feature_enable > would > > pass or not > > + * If 0, validate whether subsequent @ref rte_graph_feature_disable > would > > pass or not > > + * > > + * @return > > + * 0: Subsequent enable/disable API would pass > > + * <0: Subsequent enable/disable API would not pass > > + */ > > +__rte_experimental > > +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t > index, > > + const char *feature_name, int is_enable_disable); > > + > > +/** > > + * Disable already enabled feature within a feature arc > > + * > > + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe > > + * > > + * @param _arc > > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > > @ref > > + * rte_graph_feature_arc_lookup_by_name > > + * @param index > > + * Application specific index. Can be corresponding to > interface_id/port_id > > etc > > + * @param feature_name > > + * Name of the node which is already added via @ref > rte_graph_feature_add > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t > index, > > + const char *feature_name); > > + > > +/** > > + * Get rte_graph_feature_t object from feature name > > + * > > + * @param arc > > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > > @ref > > + * rte_graph_feature_arc_lookup_by_name > > + * @param feature_name > > + * Feature name provided to @ref rte_graph_feature_add > > + * @param[out] feature > > + * Feature object > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char > > *feature_name, > > + rte_graph_feature_t *feature); > > + > > +/** > > + * Delete feature_arc object > > + * > > + * @param _arc > > + * Feature arc object returned by @ref rte_graph_feature_arc_create or > > @ref > > + * rte_graph_feature_arc_lookup_by_name > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); > > + > > +/** > > + * Cleanup all feature arcs > > + * > > + * @return > > + * 0: Success > > + * <0: Failure > > + */ > > +__rte_experimental > > +int rte_graph_feature_arc_cleanup(void); > > + > > +/** > > + * Slow path API to know how many features are currently enabled within a > > featur-arc > > + * > > + * @param _arc > > + * Feature arc object > > + * > > + * @return: Number of enabled features > > + */ > > +__rte_experimental > > +int > rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t > > _arc); > > +#ifdef __cplusplus > > +} > > +#endif > > + > > +#endif > > diff --git a/lib/graph/rte_graph_feature_arc_worker.h > > b/lib/graph/rte_graph_feature_arc_worker.h > > new file mode 100644 > > index 0000000000..6019d74853 > > --- /dev/null > > +++ b/lib/graph/rte_graph_feature_arc_worker.h > > @@ -0,0 +1,548 @@ > > +/* SPDX-License-Identifier: BSD-3-Clause > > + * Copyright(C) 2024 Marvell International Ltd. > > + */ > > + > > +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ > > +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ > > + > > +#include <stddef.h> > > +#include <rte_graph_feature_arc.h> > > +#include <rte_bitops.h> > > + > > +/** > > + * @file > > + * > > + * rte_graph_feature_arc_worker.h > > + * > > + * Defines fast path structure > > + */ > > + > > +#ifdef __cplusplus > > +extern "C" { > > +#endif > > + > > +/** @internal > > + * > > + * Slow path feature node info list > > + */ > > +struct rte_graph_feature_node_list { > > + /** Next feature */ > > + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; > > + > > + /** node representing feature */ > > + struct rte_node_register *feature_node; > > + > > + /** How many indexes/interfaces using this feature */ > > + int32_t ref_count; > > + > > + /* node_index in list (after feature_enable())*/ > > + uint32_t node_index; > > + > > + /** Back pointer to feature arc */ > > + void *feature_arc; > > + > > + /** rte_edge_t to this feature node from feature_arc->start_node */ > > + rte_edge_t edge_to_this_feature; > > +}; > > + > > +/** > > + * Fast path holding rte_edge_t and next enabled feature for an feature > > + */ > > +typedef struct __rte_packed rte_graph_feature_data { > > + /* next node to which current mbuf should go*/ > > + rte_edge_t next_edge; > > + > > + /* next enabled feature on this arc for current index */ > > + union { > > + uint16_t reserved; > > + struct { > > + rte_graph_feature_t next_enabled_feature; > > + }; > > + }; > > + > > + /* user_data */ > > + int32_t user_data; > > +} rte_graph_feature_data_t; > > + > > +/** > > + * Fast path feature structure. Holds re_graph_feature_data_t per index > > + */ > > +struct __rte_cache_aligned rte_graph_feature { > > + uint16_t this_feature_index; > > + > > + /* Array of size arc->feature_data_size > > + * [data-index-0][data-index-1]... > > + * Each index of size: sizeof(rte_graph_feature_data_t) > > + */ > > + uint8_t feature_data_by_index[]; > > +}; > > + > > +/** > > + * fast path cache aligned feature list holding all features > > + * There are two feature lists: active, passive > > + * > > + * Fast APIs works on active list while control plane updates passive list > > + * A atomic update to arc->active_feature_list is done to switch between > > active > > + * and passive > > + */ > > +typedef struct __rte_cache_aligned rte_graph_feature_list { > > + /** > > + * fast path array holding per_feature data. > > + * Duplicate entry as feature-arc also hold this pointer > > + * arc->features[] > > + * > > + *<-------------feature-0 ---------><CEIL><---------feature-1 -------------- > > >... > > + *[index-0][index-1]...[max_index-1] [index-0][index-1] > > ...[max_index-1]... > > + */ > > + struct rte_graph_feature *indexed_by_features; > > + /* > > + * fast path array holding first enabled feature per index > > + * (Required in start_node. In non start_node, mbuf can hold next > > enabled > > + * feature) > > + */ > > + rte_graph_feature_t first_enabled_feature_by_index[]; > > +} rte_graph_feature_list_t; > > + > > +/** > > + * rte_graph feature arc object > > + * > > + * A feature-arc can only hold RTE_GRAPH_FEATURE_MAX_PER_ARC > features > > but no > > + * limit to interface index > > + * > > + * Representing a feature arc holding all features which are > enabled/disabled > > + * on any interfaces > > + */ > > +struct __rte_cache_aligned rte_graph_feature_arc { > > + /* First 64B is fast path variables */ > > + RTE_MARKER fast_path_variables; > > + > > + /** runtime active feature list */ > > + rte_graph_feature_rt_list_t active_feature_list; > > + > > + /* Actual Size of feature_list0 */ > > + uint16_t feature_list_size; > > + > > + /** > > + * Size each feature in fastpath. > > + * sizeof(arc->active_list->indexed_by_feature[0]) > > + */ > > + uint16_t feature_size; > > + > > + /* Size of arc->max_index * sizeof(rte_graph_feature_data_t) */ > > + uint16_t feature_data_size; > > + > > + /** > > + * Fast path bitmask indicating if a feature is enabled or not Number > > + * of bits: RTE_GRAPH_FEATURE_MAX_PER_ARC > > + */ > > + uint64_t feature_enable_bitmask[2]; > > + rte_graph_feature_list_t *feature_list[2]; > > + struct rte_graph_feature *features[2]; > > + > > + /** index in feature_arc_main */ > > + uint16_t feature_arc_index; > > + > > + uint16_t reserved[3]; > > + > > + /** Slow path variables follows*/ > > + RTE_MARKER slow_path_variables; > > + > > + /** feature arc name */ > > + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; > > + > > + /** All feature lists */ > > + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; > > + > > + uint32_t runtime_enabled_features; > > + > > + /** Back pointer to feature_arc_main */ > > + void *feature_arc_main; > > + > > + /* start_node */ > > + struct rte_node_register *start_node; > > + > > + /* maximum number of features supported by this arc */ > > + uint32_t max_features; > > + > > + /* maximum number of index supported by this arc */ > > + uint32_t max_indexes; > > + > > + /* Slow path bit mask per feature per index */ > > + uint64_t feature_bit_mask_by_index[]; > > +}; > > + > > +/** Feature arc main */ > > +typedef struct feature_arc_main { > > + /** number of feature arcs created by application */ > > + uint32_t num_feature_arcs; > > + > > + /** max features arcs allowed */ > > + uint32_t max_feature_arcs; > > + > > + /** feature arcs */ > > + rte_graph_feature_arc_t feature_arcs[]; > > +} rte_graph_feature_arc_main_t; > > + > > +/** @internal Get feature arc pointer from object */ > > +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc > *)arc) > > + > > +extern rte_graph_feature_arc_main_t *__feature_arc_main; > > + > > +/** > > + * API to know if feature is valid or not > > + */ > > + > > +static __rte_always_inline int > > +rte_graph_feature_is_valid(rte_graph_feature_t feature) > > +{ > > + return (feature != RTE_GRAPH_FEATURE_INVALID); > > +} > > + > > +/** > > + * Get rte_graph_feature object with no checks > > + * > > + * @param arc > > + * Feature arc pointer > > + * @param feature > > + * Feature index > > + * @param feature_list > > + * active feature list retrieved from > > rte_graph_feature_arc_has_any_feature() > > + * or rte_graph_feature_arc_has_feature() > > + * > > + * @return > > + * Internal feature object. > > + */ > > +static __rte_always_inline struct rte_graph_feature * > > +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, > > rte_graph_feature_t feature, > > + const rte_graph_feature_rt_list_t feature_list) > > +{ > > + return ((struct rte_graph_feature *)((uint8_t *)(arc- > > >features[feature_list] + > > + (feature * arc->feature_size)))); > > +} > > + > > +/** > > + * Get rte_graph_feature object for a given interface/index from feature arc > > + * > > + * @param arc > > + * Feature arc pointer > > + * @param feature > > + * Feature index > > + * > > + * @return > > + * Internal feature object. > > + */ > > +static __rte_always_inline struct rte_graph_feature * > > +rte_graph_feature_get(struct rte_graph_feature_arc *arc, > > rte_graph_feature_t feature) > > +{ > > + RTE_VERIFY(feature < arc->max_features); > > + > > + if (likely(rte_graph_feature_is_valid(feature))) > > + return __rte_graph_feature_get(arc, feature, arc- > > >active_feature_list); > > + > > + return NULL; > > +} > > + > > +static __rte_always_inline rte_graph_feature_data_t * > > +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct > > rte_graph_feature *feature, > > + uint8_t index) > > +{ > > + RTE_SET_USED(arc); > > + return ((rte_graph_feature_data_t *)(feature->feature_data_by_index > > + > > + (index * > > sizeof(rte_graph_feature_data_t)))); > > +} > > + > > +/** > > + * Get rte_graph feature data object for a index in feature > > + * > > + * @param arc > > + * feature arc > > + * @param feature > > + * Pointer to feature object > > + * @param index > > + * Index of feature maintained in slow path linked list > > + * > > + * @return > > + * Valid feature data > > + */ > > +static __rte_always_inline rte_graph_feature_data_t * > > +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct > > rte_graph_feature *feature, > > + uint8_t index) > > +{ > > + if (likely(index < arc->max_indexes)) > > + return __rte_graph_feature_data_get(arc, feature, index); > > + > > + RTE_VERIFY(0); > > +} > > + > > +/** > > + * Fast path API to check if any feature enabled on a feature arc > > + * Typically from arc->start_node process function > > + * > > + * @param arc > > + * Feature arc object > > + * @param[out] plist > > + * Pointer to runtime active feature list which needs to be provided to > other > > + * fast path APIs > > + * > > + * @return > > + * 0: If no feature enabled > > + * Non-Zero: Bitmask of features enabled. plist is valid > > + * > > + */ > > +static __rte_always_inline uint64_t > > +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, > > + rte_graph_feature_rt_list_t *plist) > > +{ > > + *plist = __atomic_load_n(&arc->active_feature_list, > > __ATOMIC_RELAXED); > > + > > + return (__atomic_load_n(arc->feature_enable_bitmask + > > (uint8_t)*plist, > > + __ATOMIC_RELAXED)); > > +} > > + > > +/** > > + * Fast path API to check if provided feature is enabled on any > interface/index > > + * or not > > + * > > + * @param arc > > + * Feature arc object > > + * @param feature > > + * Input rte_graph_feature_t that needs to be checked > > + * @param[out] plist > > + * Returns active list to caller which needs to be provided to other fast > path > > + * APIs > > + * > > + * @return > > + * 1: If feature is enabled in arc > > + * 0: If feature is not enabled in arc > > + */ > > +static __rte_always_inline int > > +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, > > + rte_graph_feature_t feature, > > + rte_graph_feature_rt_list_t *plist) > > +{ > > + uint64_t bitmask = RTE_BIT64(feature); > > + > > + *plist = __atomic_load_n(&arc->active_feature_list, > > __ATOMIC_RELAXED); > > + > > + return (bitmask & __atomic_load_n(arc->feature_enable_bitmask + > > (uint8_t)*plist, > > + __ATOMIC_RELAXED)); > > +} > > + > > +/** > > + * Prefetch feature arc fast path cache line > > + * > > + * @param arc > > + * RTE_GRAPH feature arc object > > + */ > > +static __rte_always_inline void > > +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) > > +{ > > + rte_prefetch0((void *)&arc->fast_path_variables); > > +} > > + > > +/** > > + * Prefetch feature related fast path cache line > > + * > > + * @param arc > > + * RTE_GRAPH feature arc object > > + * @param list > > + * Pointer to runtime active feature list from > > rte_graph_feature_arc_has_any_feature(); > > + * @param feature > > + * Pointer to feature object > > + */ > > +static __rte_always_inline void > > +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, > > + const rte_graph_feature_rt_list_t list, > > + rte_graph_feature_t feature) > > +{ > > + /* feature cache line */ > > + if (likely(rte_graph_feature_is_valid(feature))) > > + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, > > list)); > > +} > > + > > +/** > > + * Prefetch feature data upfront. Perform sanity > > + * > > + * @param _arc > > + * RTE_GRAPH feature arc object > > + * @param list > > + * Pointer to runtime active feature list from > > rte_graph_feature_arc_has_any_feature(); > > + * @param feature > > + * Pointer to feature object returned from @ref > > + * rte_graph_feature_arc_first_feature_get() > > + * @param index > > + * Interface/index > > + */ > > +static __rte_always_inline void > > +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, > > + const rte_graph_feature_rt_list_t list, > > + rte_graph_feature_t feature, uint32_t > index) > > +{ > > + if (likely(rte_graph_feature_is_valid(feature))) > > + rte_prefetch0((void *)((uint8_t *)arc->features[list] + > > + offsetof(struct rte_graph_feature, > > feature_data_by_index) + > > + (index * sizeof(rte_graph_feature_data_t)))); > > +} > > + > > +/** > > + * Fast path API to get first enabled feature on interface index > > + * Typically required in arc->start_node so that from returned feature, > > + * feature-data can be retrieved to steer packets > > + * > > + * @param arc > > + * Feature arc object > > + * @param list > > + * Pointer to runtime active feature list from > > + * rte_graph_feature_arc_has_any_feature() or > > + * rte_graph_feature_arc_has_feature() > > + * @param index > > + * Interface Index > > + * @param[out] feature > > + * Pointer to rte_graph_feature_t. > > + * > > + * @return > > + * 0. Success. feature field is valid > > + * 1. Failure. feature field is invalid > > + * > > + */ > > +static __rte_always_inline int > > +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, > > + const rte_graph_feature_rt_list_t list, > > + uint32_t index, > > + rte_graph_feature_t *feature) > > +{ > > + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; > > + > > + *feature = feature_list->first_enabled_feature_by_index[index]; > > + > > + return rte_graph_feature_is_valid(*feature); > > +} > > + > > +/** > > + * Fast path API to get next enabled feature on interface index with > provided > > + * input feature > > + * > > + * @param arc > > + * Feature arc object > > + * @param list > > + * Pointer to runtime active feature list from > > + * rte_graph_feature_arc_has_any_feature() or > > + * @param index > > + * Interface Index > > + * @param[in][out] feature > > + * Pointer to rte_graph_feature_t. Input feature set to next enabled > feature > > + * after success return > > + * @param[out] next_edge > > + * Edge from current feature to next feature. Valid only if next feature is > > valid > > + * > > + * @return > > + * 0. Success. next enabled feature is valid. > > + * 1. Failure. next enabled feature is invalid > > + */ > > +static __rte_always_inline int > > +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc > *arc, > > + const rte_graph_feature_rt_list_t list, > > + uint32_t index, > > + rte_graph_feature_t *feature, > > + rte_edge_t *next_edge) > > +{ > > + rte_graph_feature_data_t *feature_data = NULL; > > + struct rte_graph_feature *f = NULL; > > + > > + if (likely(rte_graph_feature_is_valid(*feature))) { > > + f = __rte_graph_feature_get(arc, *feature, list); > > + feature_data = rte_graph_feature_data_get(arc, f, index); > > + *feature = feature_data->next_enabled_feature; > > + *next_edge = feature_data->next_edge; > > + return (*feature == RTE_GRAPH_FEATURE_INVALID); > > + } > > + > > + return 1; > > +} > > + > > +/** > > + * Set fields with respect to first enabled feature in an arc and return edge > > + * Typically returned feature and interface index must be saved in > rte_mbuf > > + * structure to pass this information to next feature node > > + * > > + * @param arc > > + * Feature arc object > > + * @param list > > + * Pointer to runtime active feature list from > > rte_graph_feature_arc_has_any_feature(); > > + * @param index > > + * Index (of interface) > > + * @param[out] gf > > + * Pointer to rte_graph_feature_t. Valid if API returns Success > > + * @param[out] edge > > + * Edge to steer packet from arc->start_node to first enabled feature. Valid > > + * only if API returns Success > > + * > > + * @return > > + * 0: If valid feature is set by API > > + * 1: If valid feature is NOT set by API > > + */ > > +static __rte_always_inline rte_graph_feature_t > > +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, > > + const rte_graph_feature_rt_list_t list, > > + uint32_t index, > > + rte_graph_feature_t *gf, > > + rte_edge_t *edge) > > +{ > > + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; > > + struct rte_graph_feature_data *feature_data = NULL; > > + struct rte_graph_feature *feature = NULL; > > + rte_graph_feature_t f; > > + > > + /* reset */ > > + *gf = RTE_GRAPH_FEATURE_INVALID; > > + f = feature_list->first_enabled_feature_by_index[index]; > > + > > + if (unlikely(rte_graph_feature_is_valid(f))) { > > + feature = __rte_graph_feature_get(arc, f, list); > > + feature_data = rte_graph_feature_data_get(arc, feature, > > index); > > + *gf = f; > > + *edge = feature_data->next_edge; > > + return 0; > > + } > > + > > + return 1; > > +} > > + > > +/** > > + * Get user data corresponding to current feature set by application in > > + * rte_graph_feature_enable() > > + * > > + * @param arc > > + * Feature arc object > > + * @param list > > + * Pointer to runtime active feature list from > > rte_graph_feature_arc_has_any_feature(); > > + * @param feature > > + * Feature index > > + * @param index > > + * Interface index > > + * > > + * @return > > + * UINT32_MAX: Failure > > + * Valid user data: Success > > + */ > > +static __rte_always_inline uint32_t > > +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, > > + const rte_graph_feature_rt_list_t list, > > + rte_graph_feature_t feature, > > + uint32_t index) > > +{ > > + rte_graph_feature_data_t *fdata = NULL; > > + struct rte_graph_feature *f = NULL; > > + > > + if (likely(rte_graph_feature_is_valid(feature))) { > > + f = __rte_graph_feature_get(arc, feature, list); > > + fdata = rte_graph_feature_data_get(arc, f, index); > > + return fdata->user_data; > > + } > > + > > + return UINT32_MAX; > > +} > > +#ifdef __cplusplus > > +} > > +#endif > > +#endif > > diff --git a/lib/graph/version.map b/lib/graph/version.map > > index 2c83425ddc..82b2469fba 100644 > > --- a/lib/graph/version.map > > +++ b/lib/graph/version.map > > @@ -52,3 +52,20 @@ DPDK_25 { > > > > local: *; > > }; > > + > > +EXPERIMENTAL { > > + global: > > + > > + # added in 24.11 > > + rte_graph_feature_arc_init; > > + rte_graph_feature_arc_create; > > + rte_graph_feature_arc_lookup_by_name; > > + rte_graph_feature_add; > > + rte_graph_feature_enable; > > + rte_graph_feature_validate; > > + rte_graph_feature_disable; > > + rte_graph_feature_lookup; > > + rte_graph_feature_arc_destroy; > > + rte_graph_feature_arc_cleanup; > > + rte_graph_feature_arc_num_enabled_features; > > +}; > > -- > > 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH 2/3] graph: add feature arc option in graph create 2024-09-07 7:31 [RFC PATCH 0/3] add feature arc in rte_graph Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 1/3] graph: add feature arc support Nitin Saxena @ 2024-09-07 7:31 ` Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 3/3] graph: add IPv4 output feature arc Nitin Saxena ` (2 subsequent siblings) 4 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-09-07 7:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena, Pavan Nikhilesh Added option in graph create to call feature-specific process node functions. This removes extra overhead for checking feature arc status in nodes where application is not using feature arc processing Signed-off-by: Pavan Nikhilesh <pbhagavatula@marvell.com> --- lib/graph/graph.c | 1 + lib/graph/graph_populate.c | 7 ++++++- lib/graph/graph_private.h | 3 +++ lib/graph/node.c | 2 ++ lib/graph/rte_graph.h | 3 +++ 5 files changed, 15 insertions(+), 1 deletion(-) diff --git a/lib/graph/graph.c b/lib/graph/graph.c index d5b8c9f918..b0ad3a83ae 100644 --- a/lib/graph/graph.c +++ b/lib/graph/graph.c @@ -455,6 +455,7 @@ rte_graph_create(const char *name, struct rte_graph_param *prm) graph->parent_id = RTE_GRAPH_ID_INVALID; graph->lcore_id = RTE_MAX_LCORE; graph->num_pkt_to_capture = prm->num_pkt_to_capture; + graph->feature_arc_enabled = prm->feature_arc_enable; if (prm->pcap_filename) rte_strscpy(graph->pcap_filename, prm->pcap_filename, RTE_GRAPH_PCAP_FILE_SZ); diff --git a/lib/graph/graph_populate.c b/lib/graph/graph_populate.c index ed596a7711..2892f3cce2 100644 --- a/lib/graph/graph_populate.c +++ b/lib/graph/graph_populate.c @@ -79,8 +79,13 @@ graph_nodes_populate(struct graph *_graph) if (graph_pcap_is_enable()) { node->process = graph_pcap_dispatch; node->original_process = graph_node->node->process; - } else + if (_graph->feature_arc_enabled) + node->original_process = graph_node->node->feat_arc_proc; + } else { node->process = graph_node->node->process; + if (_graph->feature_arc_enabled) + node->process = graph_node->node->feat_arc_proc; + } memcpy(node->name, graph_node->node->name, RTE_GRAPH_NAMESIZE); pid = graph_node->node->parent_id; if (pid != RTE_NODE_ID_INVALID) { /* Cloned node */ diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index d557d55f2d..58ba0abeff 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -56,6 +56,7 @@ struct node { unsigned int lcore_id; /**< Node runs on the Lcore ID used for mcore dispatch model. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Allocated identifier for the node. */ @@ -126,6 +127,8 @@ struct graph { /**< Number of packets to be captured per core. */ char pcap_filename[RTE_GRAPH_PCAP_FILE_SZ]; /**< pcap file name/path. */ + uint8_t feature_arc_enabled; + /**< Graph feature arc. */ STAILQ_HEAD(gnode_list, graph_node) node_list; /**< Nodes in a graph. */ }; diff --git a/lib/graph/node.c b/lib/graph/node.c index 99a9622779..d8fd273543 100644 --- a/lib/graph/node.c +++ b/lib/graph/node.c @@ -90,6 +90,7 @@ __rte_node_register(const struct rte_node_register *reg) goto free; node->flags = reg->flags; node->process = reg->process; + node->feat_arc_proc = reg->feat_arc_proc; node->init = reg->init; node->fini = reg->fini; node->nb_edges = reg->nb_edges; @@ -137,6 +138,7 @@ node_clone(struct node *node, const char *name) /* Clone the source node */ reg->flags = node->flags; reg->process = node->process; + reg->feat_arc_proc = node->feat_arc_proc; reg->init = node->init; reg->fini = node->fini; reg->nb_edges = node->nb_edges; diff --git a/lib/graph/rte_graph.h b/lib/graph/rte_graph.h index ecfec2068a..ebbdbbea48 100644 --- a/lib/graph/rte_graph.h +++ b/lib/graph/rte_graph.h @@ -163,6 +163,8 @@ struct rte_graph_param { uint64_t num_pkt_to_capture; /**< Number of packets to capture. */ char *pcap_filename; /**< Filename in which packets to be captured.*/ + bool feature_arc_enable; /**< Enable Graph feature arc. */ + union { struct { uint64_t rsvd; /**< Reserved for rtc model. */ @@ -470,6 +472,7 @@ struct rte_node_register { uint64_t flags; /**< Node configuration flag. */ #define RTE_NODE_SOURCE_F (1ULL << 0) /**< Node type is source. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Node Identifier. */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH 3/3] graph: add IPv4 output feature arc 2024-09-07 7:31 [RFC PATCH 0/3] add feature arc in rte_graph Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 1/3] graph: add feature arc support Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 2/3] graph: add feature arc option in graph create Nitin Saxena @ 2024-09-07 7:31 ` Nitin Saxena 2024-10-08 8:04 ` [RFC PATCH 0/3] add feature arc in rte_graph David Marchand 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena 4 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-09-07 7:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena add ipv4-output feature arc in ipv4-rewrite node to allow custom/standard nodes(like outbound IPsec policy node) in outgoing forwarding path Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/node/ip4_rewrite.c | 476 +++++++++++++++++++++++++++++------- lib/node/ip4_rewrite_priv.h | 9 +- lib/node/node_private.h | 19 +- lib/node/rte_node_ip4_api.h | 3 + 4 files changed, 411 insertions(+), 96 deletions(-) diff --git a/lib/node/ip4_rewrite.c b/lib/node/ip4_rewrite.c index 34a920df5e..916f180a3d 100644 --- a/lib/node/ip4_rewrite.c +++ b/lib/node/ip4_rewrite.c @@ -15,39 +15,156 @@ #include "ip4_rewrite_priv.h" #include "node_private.h" +#define ALL_PKT_MASK 0xf + struct ip4_rewrite_node_ctx { + rte_graph_feature_arc_t output_feature_arc; /* Dynamic offset to mbuf priv1 */ int mbuf_priv1_off; /* Cached next index */ uint16_t next_index; + uint16_t last_tx; }; +typedef struct rewrite_priv_vars { + union { + struct { + rte_xmm_t xmm1; + }; + struct __rte_packed { + uint16_t next0; + uint16_t next1; + uint16_t next2; + uint16_t next3; + uint16_t last_tx_interface; + uint16_t last_if_feature; + uint16_t actual_feat_mask; + uint16_t speculative_feat_mask; + }; + }; +} rewrite_priv_vars_t; + static struct ip4_rewrite_node_main *ip4_rewrite_nm; #define IP4_REWRITE_NODE_LAST_NEXT(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->next_index) +#define IP4_REWRITE_NODE_LAST_TX(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->last_tx) + #define IP4_REWRITE_NODE_PRIV1_OFF(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->mbuf_priv1_off) -static uint16_t -ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, - void **objs, uint16_t nb_objs) +#define IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->output_feature_arc) + +static __rte_always_inline void +prefetch_mbuf_and_dynfield(struct rte_mbuf *mbuf) { + /* prefetch first cache line required for accessing buf_addr */ + rte_prefetch0((void *)mbuf); +} + +static __rte_always_inline void +check_output_feature_x4(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t flist, + rewrite_priv_vars_t *pvar, struct node_mbuf_priv1 *priv0, + struct node_mbuf_priv1 *priv1, struct node_mbuf_priv1 *priv2, + struct node_mbuf_priv1 *priv3) +{ + uint32_t mask = 0; + uint16_t xor = 0; + + /* + * interface edge's start from 1 and not from 0 as "pkt_drop" + * is next node at 0th index + */ + priv0->if_index = pvar->next0 - 1; + priv1->if_index = pvar->next1 - 1; + priv2->if_index = pvar->next2 - 1; + priv3->if_index = pvar->next3 - 1; + + /* Find out if all packets are sent to last_tx_interface */ + xor = pvar->last_tx_interface ^ priv0->if_index; + xor += priv0->if_index ^ priv1->if_index; + xor += priv1->if_index ^ priv2->if_index; + xor += priv2->if_index ^ priv3->if_index; + + if (likely(!xor)) { + /* copy last interface feature and feature mask */ + priv0->current_feature = priv1->current_feature = + priv2->current_feature = priv3->current_feature = + pvar->last_if_feature; + pvar->actual_feat_mask = pvar->speculative_feat_mask; + } else { + /* create a mask for index which does not have feature + * Also override next edge and if feature enabled, get feature + */ + mask = rte_graph_feature_arc_feature_set(arc, flist, priv0->if_index, + &priv0->current_feature, + &pvar->next0); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv1->if_index, + &priv1->current_feature, + &pvar->next1)) << 1); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv2->if_index, + &priv2->current_feature, + &pvar->next2)) << 2); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv3->if_index, + &priv3->current_feature, + &pvar->next3)) << 3); + + /* + * add last tx and last feature regardless even if feature is + * valid or not + */ + pvar->last_tx_interface = priv3->if_index; + pvar->last_if_feature = priv3->current_feature; + /* Set 0xf if invalid feature to last packet, else 0 */ + pvar->speculative_feat_mask = (priv3->current_feature == + RTE_GRAPH_FEATURE_INVALID) ? ALL_PKT_MASK : 0x0; + pvar->actual_feat_mask = mask; + } +} + +static __rte_always_inline uint16_t +__ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + struct rte_graph_feature_arc *out_feature_arc, + void **objs, uint16_t nb_objs, + const int dyn, const int check_enabled_features, + const rte_graph_feature_rt_list_t flist) +{ + struct node_mbuf_priv1 *priv0 = NULL, *priv1 = NULL, *priv2 = NULL, *priv3 = NULL; struct rte_mbuf *mbuf0, *mbuf1, *mbuf2, *mbuf3, **pkts; struct ip4_rewrite_nh_header *nh = ip4_rewrite_nm->nh; - const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); - uint16_t next0, next1, next2, next3, next_index; - struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; uint16_t n_left_from, held = 0, last_spec = 0; + struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; + rewrite_priv_vars_t pvar; + int64_t fd0, fd1, fd2, fd3; + rte_edge_t fix_spec = 0; void *d0, *d1, *d2, *d3; void **to_next, **from; + uint16_t next_index; rte_xmm_t priv01; rte_xmm_t priv23; int i; - /* Speculative next as last next */ + RTE_SET_USED(fd0); + RTE_SET_USED(fd1); + RTE_SET_USED(fd2); + RTE_SET_USED(fd3); + + /* Initialize speculative variables.*/ + + /* Last interface */ + pvar.last_tx_interface = IP4_REWRITE_NODE_LAST_TX(node->ctx); + /*last next from node ctx*/ next_index = IP4_REWRITE_NODE_LAST_NEXT(node->ctx); + pvar.speculative_feat_mask = ALL_PKT_MASK; + pvar.actual_feat_mask = 0; + rte_prefetch0(nh); pkts = (struct rte_mbuf **)objs; @@ -55,20 +172,47 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, n_left_from = nb_objs; for (i = 0; i < 4 && i < n_left_from; i++) - rte_prefetch0(pkts[i]); + prefetch_mbuf_and_dynfield(pkts[i]); /* Get stream for the speculated next node */ to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + + /* prefetch speculative feature and corresponding data */ + if (check_enabled_features) { + /* + * Get first feature enabled, if any, on last_tx_interface + */ + if (unlikely(!rte_graph_feature_arc_first_feature_get(out_feature_arc, + flist, + pvar.last_tx_interface, + (rte_graph_feature_t *) + &pvar.last_if_feature))) { + /* prefetch feature cache line */ + rte_graph_feature_arc_feature_prefetch(out_feature_arc, flist, + pvar.last_if_feature); + + /* prefetch feature data cache line */ + rte_graph_feature_arc_data_prefetch(out_feature_arc, flist, + pvar.last_if_feature, + pvar.last_tx_interface); + /* + * Set speculativa_feat mask to indicate, all 4 packets + * going to feature path + */ + pvar.speculative_feat_mask = 0; + } + } + /* Update Ethernet header of pkts */ while (n_left_from >= 4) { if (likely(n_left_from > 7)) { /* Prefetch only next-mbuf struct and priv area. * Data need not be prefetched as we only write. */ - rte_prefetch0(pkts[4]); - rte_prefetch0(pkts[5]); - rte_prefetch0(pkts[6]); - rte_prefetch0(pkts[7]); + prefetch_mbuf_and_dynfield(pkts[4]); + prefetch_mbuf_and_dynfield(pkts[5]); + prefetch_mbuf_and_dynfield(pkts[6]); + prefetch_mbuf_and_dynfield(pkts[7]); } mbuf0 = pkts[0]; @@ -78,66 +222,138 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 4; n_left_from -= 4; + + /* Copy mbuf private data into private variables */ priv01.u64[0] = node_mbuf_priv1(mbuf0, dyn)->u; priv01.u64[1] = node_mbuf_priv1(mbuf1, dyn)->u; priv23.u64[0] = node_mbuf_priv1(mbuf2, dyn)->u; priv23.u64[1] = node_mbuf_priv1(mbuf3, dyn)->u; - /* Increment checksum by one. */ - priv01.u32[1] += rte_cpu_to_be_16(0x0100); - priv01.u32[3] += rte_cpu_to_be_16(0x0100); - priv23.u32[1] += rte_cpu_to_be_16(0x0100); - priv23.u32[3] += rte_cpu_to_be_16(0x0100); - - /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, - nh[priv01.u16[0]].rewrite_len); - - next0 = nh[priv01.u16[0]].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - ip0->time_to_live = priv01.u16[1] - 1; - ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ - d1 = rte_pktmbuf_mtod(mbuf1, void *); - rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, - nh[priv01.u16[4]].rewrite_len); - - next1 = nh[priv01.u16[4]].tx_node; - ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + - sizeof(struct rte_ether_hdr)); - ip1->time_to_live = priv01.u16[5] - 1; - ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ - d2 = rte_pktmbuf_mtod(mbuf2, void *); - rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, - nh[priv23.u16[0]].rewrite_len); - next2 = nh[priv23.u16[0]].tx_node; - ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + - sizeof(struct rte_ether_hdr)); - ip2->time_to_live = priv23.u16[1] - 1; - ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ - d3 = rte_pktmbuf_mtod(mbuf3, void *); - rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, - nh[priv23.u16[4]].rewrite_len); - - next3 = nh[priv23.u16[4]].tx_node; - ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + - sizeof(struct rte_ether_hdr)); - ip3->time_to_live = priv23.u16[5] - 1; - ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + /* Copy next edge from next hop */ + pvar.next0 = nh[priv01.u16[0]].tx_node; + pvar.next1 = nh[priv01.u16[4]].tx_node; + pvar.next2 = nh[priv23.u16[0]].tx_node; + pvar.next3 = nh[priv23.u16[4]].tx_node; + + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + priv1 = node_mbuf_priv1(mbuf1, dyn); + priv2 = node_mbuf_priv1(mbuf2, dyn); + priv3 = node_mbuf_priv1(mbuf3, dyn); + + /* If feature is enabled, override next edge for each mbuf + * and set node_mbuf_priv data appropriately + */ + check_output_feature_x4(out_feature_arc, flist, + &pvar, priv0, priv1, priv2, priv3); + + /* check_output_feature_x4() returns bit mask which indicates + * which packet is not following feature path, hence normal processing + * has to happen on them + */ + if (unlikely(pvar.actual_feat_mask)) { + if (pvar.actual_feat_mask & 0x1) { + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + } + if (pvar.actual_feat_mask & 0x2) { + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + } + if (pvar.actual_feat_mask & 0x4) { + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + } + if (pvar.actual_feat_mask & 0x8) { + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } + } + } else { + /* Case when no feature is enabled */ + + /* Increment checksum by one. */ + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } /* Enqueue four to next node */ - rte_edge_t fix_spec = - ((next_index == next0) && (next0 == next1) && - (next1 == next2) && (next2 == next3)); + fix_spec = next_index ^ pvar.next0; + fix_spec += next_index ^ pvar.next1; + fix_spec += next_index ^ pvar.next2; + fix_spec += next_index ^ pvar.next3; - if (unlikely(fix_spec == 0)) { + if (unlikely(fix_spec != 0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -146,56 +362,56 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, last_spec = 0; /* next0 */ - if (next_index == next0) { + if (next_index == pvar.next0) { to_next[0] = from[0]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next0, + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); } /* next1 */ - if (next_index == next1) { + if (next_index == pvar.next1) { to_next[0] = from[1]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next1, + rte_node_enqueue_x1(graph, node, pvar.next1, from[1]); } /* next2 */ - if (next_index == next2) { + if (next_index == pvar.next2) { to_next[0] = from[2]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next2, + rte_node_enqueue_x1(graph, node, pvar.next2, from[2]); } /* next3 */ - if (next_index == next3) { + if (next_index == pvar.next3) { to_next[0] = from[3]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next3, + rte_node_enqueue_x1(graph, node, pvar.next3, from[3]); } from += 4; /* Change speculation if last two are same */ - if ((next_index != next3) && (next2 == next3)) { + if ((next_index != pvar.next3) && (pvar.next2 == pvar.next3)) { /* Put the current speculated node */ rte_node_next_stream_put(graph, node, next_index, held); held = 0; /* Get next speculated stream */ - next_index = next3; + next_index = pvar.next3; to_next = rte_node_next_stream_get( graph, node, next_index, nb_objs); } @@ -212,20 +428,41 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 1; n_left_from -= 1; - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, - nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); - - next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + - rte_cpu_to_be_16(0x0100); - chksum += chksum >= 0xffff; - ip0->hdr_checksum = chksum; - ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; - - if (unlikely(next_index ^ next0)) { + pvar.next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + if (pvar.next0 != (pvar.last_tx_interface + 1)) { + priv0->if_index = pvar.next0 - 1; + rte_graph_feature_arc_feature_set(out_feature_arc, flist, + priv0->if_index, + &priv0->current_feature, + &pvar.next0); + pvar.last_tx_interface = priv0->if_index; + pvar.last_if_feature = priv0->current_feature; + } else { + /* current mbuf index is same as last_tx_interface */ + priv0->if_index = pvar.last_tx_interface; + priv0->current_feature = pvar.last_if_feature; + } + } + /* Do the needful if either feature arc is disabled OR + * Invalid feature is present + */ + if (!check_enabled_features || + (priv0->current_feature == RTE_GRAPH_FEATURE_INVALID)) { + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, + nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + + rte_cpu_to_be_16(0x0100); + chksum += chksum >= 0xffff; + ip0->hdr_checksum = chksum; + ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; + } + if (unlikely(next_index ^ pvar.next0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -233,13 +470,15 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, held += last_spec; last_spec = 0; - rte_node_enqueue_x1(graph, node, next0, from[0]); + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); from += 1; } else { last_spec += 1; } } + IP4_REWRITE_NODE_LAST_TX(node->ctx) = pvar.last_tx_interface; + /* !!! Home run !!! */ if (likely(last_spec == nb_objs)) { rte_node_next_stream_move(graph, node, next_index); @@ -255,22 +494,78 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, return nb_objs; } +static uint16_t +ip4_rewrite_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = + rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + rte_graph_feature_rt_list_t flist; + + /* If any feature is enabled on this arc */ + if (unlikely(rte_graph_feature_arc_has_any_feature(arc, &flist))) { + if (flist) + return __ip4_rewrite_node_process(graph, node, arc, objs, nb_objs, + dyn, + 1 /* check features */, + (rte_graph_feature_rt_list_t)1); + else + return __ip4_rewrite_node_process(graph, node, arc, objs, nb_objs, + dyn, + 1 /* check features */, + (rte_graph_feature_rt_list_t)0); + } else { + return __ip4_rewrite_node_process(graph, node, arc, objs, nb_objs, dyn, + 0/* don't check features*/, + 0/* don't care */); + } + return 0; +} + +static uint16_t +ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + return __ip4_rewrite_node_process(graph, node, NULL, objs, nb_objs, dyn, + 0/* don't check features*/, + 0/* don't care */); +} + static int ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) { + rte_graph_feature_arc_t feature_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; static bool init_once; RTE_SET_USED(graph); RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_node_ctx) > RTE_NODE_CTX_SZ); + RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_nh_header) != RTE_CACHE_LINE_MIN_SIZE); if (!init_once) { node_mbuf_priv1_dynfield_offset = rte_mbuf_dynfield_register( &node_mbuf_priv1_dynfield_desc); if (node_mbuf_priv1_dynfield_offset < 0) return -rte_errno; - init_once = true; + + /* Create ipv4-output feature arc, if not created + */ + if (rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + NULL) < 0) { + if (rte_graph_feature_arc_create(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + RTE_GRAPH_FEATURE_MAX_PER_ARC, + RTE_MAX_ETHPORTS, + ip4_rewrite_node_get(), &feature_arc)) { + return -rte_errno; + } + init_once = true; + } } IP4_REWRITE_NODE_PRIV1_OFF(node->ctx) = node_mbuf_priv1_dynfield_offset; + IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx) = feature_arc; + IP4_REWRITE_NODE_LAST_TX(node->ctx) = UINT16_MAX; node_dbg("ip4_rewrite", "Initialized ip4_rewrite node initialized"); @@ -329,6 +624,7 @@ rte_node_ip4_rewrite_add(uint16_t next_hop, uint8_t *rewrite_data, static struct rte_node_register ip4_rewrite_node = { .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, .name = "ip4_rewrite", /* Default edge i.e '0' is pkt drop */ .nb_edges = 1, diff --git a/lib/node/ip4_rewrite_priv.h b/lib/node/ip4_rewrite_priv.h index 5105ec1d29..27ccc67489 100644 --- a/lib/node/ip4_rewrite_priv.h +++ b/lib/node/ip4_rewrite_priv.h @@ -5,6 +5,7 @@ #define __INCLUDE_IP4_REWRITE_PRIV_H__ #include <rte_common.h> +#include <rte_graph_feature_arc.h> #define RTE_GRAPH_IP4_REWRITE_MAX_NH 64 #define RTE_GRAPH_IP4_REWRITE_MAX_LEN 56 @@ -15,11 +16,9 @@ * Ipv4 rewrite next hop header data structure. Used to store port specific * rewrite data. */ -struct ip4_rewrite_nh_header { - uint16_t rewrite_len; /**< Header rewrite length. */ +struct __rte_cache_min_aligned ip4_rewrite_nh_header { uint16_t tx_node; /**< Tx node next index identifier. */ - uint16_t enabled; /**< NH enable flag */ - uint16_t rsvd; + uint16_t rewrite_len; /**< Header rewrite length. */ union { struct { struct rte_ether_addr dst; @@ -30,6 +29,8 @@ struct ip4_rewrite_nh_header { uint8_t rewrite_data[RTE_GRAPH_IP4_REWRITE_MAX_LEN]; /**< Generic rewrite data */ }; + /* used in control path */ + uint8_t enabled; /**< NH enable flag */ }; /** diff --git a/lib/node/node_private.h b/lib/node/node_private.h index 1de7306792..7c26686948 100644 --- a/lib/node/node_private.h +++ b/lib/node/node_private.h @@ -12,6 +12,9 @@ #include <rte_mbuf.h> #include <rte_mbuf_dyn.h> +#include <rte_graph_worker_common.h> +#include <rte_graph_feature_arc_worker.h> + extern int rte_node_logtype; #define RTE_LOGTYPE_NODE rte_node_logtype @@ -29,13 +32,25 @@ extern int rte_node_logtype; */ struct node_mbuf_priv1 { union { - /* IP4/IP6 rewrite */ + /** + * IP4/IP6 rewrite + * only used to pass lookup data from + * ip4-lookup to ip4-rewrite + */ struct { uint16_t nh; uint16_t ttl; uint32_t cksum; }; - + /** + * Feature arc data + */ + struct { + /** interface index */ + uint16_t if_index; + /** feature that current mbuf holds */ + rte_graph_feature_t current_feature; + }; uint64_t u; }; }; diff --git a/lib/node/rte_node_ip4_api.h b/lib/node/rte_node_ip4_api.h index 24f8ec843a..0de06f7fc7 100644 --- a/lib/node/rte_node_ip4_api.h +++ b/lib/node/rte_node_ip4_api.h @@ -23,6 +23,7 @@ extern "C" { #include <rte_compat.h> #include <rte_graph.h> +#include <rte_graph_feature_arc_worker.h> /** * IP4 lookup next nodes. @@ -67,6 +68,8 @@ struct rte_node_ip4_reassembly_cfg { /**< Node identifier to configure. */ }; +#define RTE_IP4_OUTPUT_FEATURE_ARC_NAME "ipv4-output" + /** * Add ipv4 route to lookup table. * -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-09-07 7:31 [RFC PATCH 0/3] add feature arc in rte_graph Nitin Saxena ` (2 preceding siblings ...) 2024-09-07 7:31 ` [RFC PATCH 3/3] graph: add IPv4 output feature arc Nitin Saxena @ 2024-10-08 8:04 ` David Marchand 2024-10-08 14:26 ` [EXTERNAL] " Nitin Saxena 2024-10-14 11:11 ` Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena 4 siblings, 2 replies; 55+ messages in thread From: David Marchand @ 2024-10-08 8:04 UTC (permalink / raw) To: Nitin Saxena Cc: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, dev, Nitin Saxena, Robin Jarry, Christophe Fontaine Hi graph guys, On Sat, Sep 7, 2024 at 9:31 AM Nitin Saxena <nsaxena@marvell.com> wrote: > > Feature arc represents an ordered list of features/protocols at a given > networking layer. It is a high level abstraction to connect various > rte_graph nodes, as feature nodes, and allow packets steering across > these nodes in a generic manner. > > Features (or feature nodes) are nodes which handles partial or complete > handling of a protocol in fast path. Like ipv4-rewrite node, which adds > rewrite data to an outgoing IPv4 packet. > > However in above example, outgoing interface(say "eth0") may have > outbound IPsec policy enabled, hence packets must be steered from > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > policy lookup. On the other hand, packets routed to another interface > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > is disabled on eth1. Feature-arc allows rte_graph applications to manage > such constraints easily > > Feature arc abstraction allows rte_graph based application to > > 1. Seamlessly steer packets across feature nodes based on wheter feature > is enabled or disabled on an interface. Features enabled on one > interface may not be enabled on another interface with in a same feature > arc. > > 2. Allow enabling/disabling of features on an interface at runtime, > so that if a feature is disabled, packets associated with that interface > won't be steered to corresponding feature node. > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > node and allow packet steering from feature node to custom node without > changing former's fast path function > > 4. Allow expressing features in a particular sequential order so that > packets are steered in an ordered way across nodes in fast path. For > eg: if IPsec and IPv4 features are enabled on an ingress interface, > packets must be sent to IPsec inbound policy node first and then to ipv4 > lookup node. > > This patch series adds feature arc library in rte_graph and also adds > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > Nitin Saxena (3): > graph: add feature arc support > graph: add feature arc option in graph create > graph: add IPv4 output feature arc > > lib/graph/graph.c | 1 + > lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ > lib/graph/graph_populate.c | 7 +- > lib/graph/graph_private.h | 3 + > lib/graph/meson.build | 2 + > lib/graph/node.c | 2 + > lib/graph/rte_graph.h | 3 + > lib/graph/rte_graph_feature_arc.h | 373 +++++++++ > lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ > lib/graph/version.map | 17 + > lib/node/ip4_rewrite.c | 476 ++++++++--- > lib/node/ip4_rewrite_priv.h | 9 +- > lib/node/node_private.h | 19 +- > lib/node/rte_node_ip4_api.h | 3 + > 14 files changed, 2325 insertions(+), 97 deletions(-) > create mode 100644 lib/graph/graph_feature_arc.c > create mode 100644 lib/graph/rte_graph_feature_arc.h > create mode 100644 lib/graph/rte_graph_feature_arc_worker.h I see no non-RFC series following this original submission. It will slip to next release unless there is an objection. Btw, I suggest copying Robin (and Christophe) for graph related changes. -- David Marchand ^ permalink raw reply [flat|nested] 55+ messages in thread
* RE: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-08 8:04 ` [RFC PATCH 0/3] add feature arc in rte_graph David Marchand @ 2024-10-08 14:26 ` Nitin Saxena 2024-10-14 11:11 ` Nitin Saxena 1 sibling, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 14:26 UTC (permalink / raw) To: David Marchand Cc: Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Nitin Saxena, Robin Jarry, Christophe Fontaine Hi David, I just sent v2 version for this patch series. Will add Robin and Christophe from next version onwards Thanks, Nitin > -----Original Message----- > From: David Marchand <david.marchand@redhat.com> > Sent: Tuesday, October 8, 2024 1:34 PM > To: Nitin Saxena <nsaxena@marvell.com> > Cc: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda > <kirankumark@marvell.com>; Nithin Kumar Dabilpuram > <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com>; > dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com>; Robin Jarry > <rjarry@redhat.com>; Christophe Fontaine <cfontain@redhat.com> > Subject: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph > > Hi graph guys, On Sat, Sep 7, 2024 at 9: 31 AM Nitin Saxena > <nsaxena@ marvell. com> wrote: > > Feature arc represents an ordered list of > features/protocols at a given > networking layer. It is a high level abstraction > to connect > Hi graph guys, > > On Sat, Sep 7, 2024 at 9:31 AM Nitin Saxena <nsaxena@marvell.com> wrote: > > > > Feature arc represents an ordered list of features/protocols at a > > given networking layer. It is a high level abstraction to connect > > various rte_graph nodes, as feature nodes, and allow packets steering > > across these nodes in a generic manner. > > > > Features (or feature nodes) are nodes which handles partial or > > complete handling of a protocol in fast path. Like ipv4-rewrite node, > > which adds rewrite data to an outgoing IPv4 packet. > > > > However in above example, outgoing interface(say "eth0") may have > > outbound IPsec policy enabled, hence packets must be steered from > > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > > policy lookup. On the other hand, packets routed to another interface > > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > > is disabled on eth1. Feature-arc allows rte_graph applications to > > manage such constraints easily > > > > Feature arc abstraction allows rte_graph based application to > > > > 1. Seamlessly steer packets across feature nodes based on wheter > > feature is enabled or disabled on an interface. Features enabled on > > one interface may not be enabled on another interface with in a same > > feature arc. > > > > 2. Allow enabling/disabling of features on an interface at runtime, so > > that if a feature is disabled, packets associated with that interface > > won't be steered to corresponding feature node. > > > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > > node and allow packet steering from feature node to custom node > > without changing former's fast path function > > > > 4. Allow expressing features in a particular sequential order so that > > packets are steered in an ordered way across nodes in fast path. For > > eg: if IPsec and IPv4 features are enabled on an ingress interface, > > packets must be sent to IPsec inbound policy node first and then to > > ipv4 lookup node. > > > > This patch series adds feature arc library in rte_graph and also adds > > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > > > Nitin Saxena (3): > > graph: add feature arc support > > graph: add feature arc option in graph create > > graph: add IPv4 output feature arc > > > > lib/graph/graph.c | 1 + > > lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ > > lib/graph/graph_populate.c | 7 +- > > lib/graph/graph_private.h | 3 + > > lib/graph/meson.build | 2 + > > lib/graph/node.c | 2 + > > lib/graph/rte_graph.h | 3 + > > lib/graph/rte_graph_feature_arc.h | 373 +++++++++ > > lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ > > lib/graph/version.map | 17 + > > lib/node/ip4_rewrite.c | 476 ++++++++--- > > lib/node/ip4_rewrite_priv.h | 9 +- > > lib/node/node_private.h | 19 +- > > lib/node/rte_node_ip4_api.h | 3 + > > 14 files changed, 2325 insertions(+), 97 deletions(-) create mode > > 100644 lib/graph/graph_feature_arc.c create mode 100644 > > lib/graph/rte_graph_feature_arc.h create mode 100644 > > lib/graph/rte_graph_feature_arc_worker.h > > I see no non-RFC series following this original submission. > It will slip to next release unless there is an objection. > > Btw, I suggest copying Robin (and Christophe) for graph related changes. > > > -- > David Marchand ^ permalink raw reply [flat|nested] 55+ messages in thread
* RE: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-08 8:04 ` [RFC PATCH 0/3] add feature arc in rte_graph David Marchand 2024-10-08 14:26 ` [EXTERNAL] " Nitin Saxena @ 2024-10-14 11:11 ` Nitin Saxena 2024-10-16 9:24 ` David Marchand 1 sibling, 1 reply; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 11:11 UTC (permalink / raw) To: David Marchand Cc: Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Nitin Saxena, Robin Jarry, Christophe Fontaine Hi David and all, >> I see no non-RFC series following this original submission. >> It will slip to next release unless there is an objection. I had pushed non RFC patch series before -rc1 date (11th oct). We have an ABI change in this patch series https://patches.dpdk.org/project/dpdk/patch/20241010133111.2764712-3-nsaxena@marvell.com/ Could you help merge this patch series in rc2 otherwise it has to wait for next LTS Thanks, Nitin > -----Original Message----- > From: David Marchand <david.marchand@redhat.com> > Sent: Tuesday, October 8, 2024 1:34 PM > To: Nitin Saxena <nsaxena@marvell.com> > Cc: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda > <kirankumark@marvell.com>; Nithin Kumar Dabilpuram > <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com>; > dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com>; Robin Jarry > <rjarry@redhat.com>; Christophe Fontaine <cfontain@redhat.com> > Subject: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph > > Hi graph guys, On Sat, Sep 7, 2024 at 9: 31 AM Nitin Saxena > <nsaxena@ marvell. com> wrote: > > Feature arc represents an ordered list of > features/protocols at a given > networking layer. It is a high level abstraction > to connect > Hi graph guys, > > On Sat, Sep 7, 2024 at 9:31 AM Nitin Saxena <nsaxena@marvell.com> wrote: > > > > Feature arc represents an ordered list of features/protocols at a > > given networking layer. It is a high level abstraction to connect > > various rte_graph nodes, as feature nodes, and allow packets steering > > across these nodes in a generic manner. > > > > Features (or feature nodes) are nodes which handles partial or > > complete handling of a protocol in fast path. Like ipv4-rewrite node, > > which adds rewrite data to an outgoing IPv4 packet. > > > > However in above example, outgoing interface(say "eth0") may have > > outbound IPsec policy enabled, hence packets must be steered from > > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > > policy lookup. On the other hand, packets routed to another interface > > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > > is disabled on eth1. Feature-arc allows rte_graph applications to > > manage such constraints easily > > > > Feature arc abstraction allows rte_graph based application to > > > > 1. Seamlessly steer packets across feature nodes based on wheter > > feature is enabled or disabled on an interface. Features enabled on > > one interface may not be enabled on another interface with in a same > > feature arc. > > > > 2. Allow enabling/disabling of features on an interface at runtime, so > > that if a feature is disabled, packets associated with that interface > > won't be steered to corresponding feature node. > > > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > > node and allow packet steering from feature node to custom node > > without changing former's fast path function > > > > 4. Allow expressing features in a particular sequential order so that > > packets are steered in an ordered way across nodes in fast path. For > > eg: if IPsec and IPv4 features are enabled on an ingress interface, > > packets must be sent to IPsec inbound policy node first and then to > > ipv4 lookup node. > > > > This patch series adds feature arc library in rte_graph and also adds > > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > > > Nitin Saxena (3): > > graph: add feature arc support > > graph: add feature arc option in graph create > > graph: add IPv4 output feature arc > > > > lib/graph/graph.c | 1 + > > lib/graph/graph_feature_arc.c | 959 +++++++++++++++++++++++ > > lib/graph/graph_populate.c | 7 +- > > lib/graph/graph_private.h | 3 + > > lib/graph/meson.build | 2 + > > lib/graph/node.c | 2 + > > lib/graph/rte_graph.h | 3 + > > lib/graph/rte_graph_feature_arc.h | 373 +++++++++ > > lib/graph/rte_graph_feature_arc_worker.h | 548 +++++++++++++ > > lib/graph/version.map | 17 + > > lib/node/ip4_rewrite.c | 476 ++++++++--- > > lib/node/ip4_rewrite_priv.h | 9 +- > > lib/node/node_private.h | 19 +- > > lib/node/rte_node_ip4_api.h | 3 + > > 14 files changed, 2325 insertions(+), 97 deletions(-) create mode > > 100644 lib/graph/graph_feature_arc.c create mode 100644 > > lib/graph/rte_graph_feature_arc.h create mode 100644 > > lib/graph/rte_graph_feature_arc_worker.h > > I see no non-RFC series following this original submission. > It will slip to next release unless there is an objection. > > Btw, I suggest copying Robin (and Christophe) for graph related changes. > > > -- > David Marchand ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-14 11:11 ` Nitin Saxena @ 2024-10-16 9:24 ` David Marchand 2024-10-16 9:38 ` Robin Jarry 0 siblings, 1 reply; 55+ messages in thread From: David Marchand @ 2024-10-16 9:24 UTC (permalink / raw) To: Nitin Saxena Cc: Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Nitin Saxena, Robin Jarry, Christophe Fontaine On Mon, Oct 14, 2024 at 1:12 PM Nitin Saxena <nsaxena@marvell.com> wrote: > I had pushed non RFC patch series before -rc1 date (11th oct). > We have an ABI change in this patch series https://patches.dpdk.org/project/dpdk/patch/20241010133111.2764712-3-nsaxena@marvell.com/ > Could you help merge this patch series in rc2 otherwise it has to wait for next LTS Just read through the series, I am not confident with this addition. It requires a lot of changes in the node code for supporting it, where it should be something handled in/facilitated by the graph library itself. I did not read much from Robin or Christophe who have been writing more node code than me. I would prefer their opinion before going forward. On the ABI topic. As far as I can see, the only issue would be in extending struct rte_node_register, but this can be solved with function versioning. That change would have to be announced. Am I missing something else? -- David Marchand ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-16 9:24 ` David Marchand @ 2024-10-16 9:38 ` Robin Jarry 2024-10-16 13:50 ` Nitin Saxena 0 siblings, 1 reply; 55+ messages in thread From: Robin Jarry @ 2024-10-16 9:38 UTC (permalink / raw) To: David Marchand, Nitin Saxena Cc: Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Nitin Saxena, Christophe Fontaine Hi folks, David Marchand, Oct 16, 2024 at 11:24: > On Mon, Oct 14, 2024 at 1:12 PM Nitin Saxena <nsaxena@marvell.com> wrote: >> I had pushed non RFC patch series before -rc1 date (11th oct). >> We have an ABI change in this patch series https://patches.dpdk.org/project/dpdk/patch/20241010133111.2764712-3-nsaxena@marvell.com/ >> Could you help merge this patch series in rc2 otherwise it has to wait for next LTS > > Just read through the series, I am not confident with this addition. > It requires a lot of changes in the node code for supporting it, where > it should be something handled in/facilitated by the graph library > itself. As far as I can tell, it will be very complicated (if not impossible) to determine in a generic manner whether a packet must be steered towards a sub tree or not. The decision *must* come from the originating node in some way or another. > I did not read much from Robin or Christophe who have been writing > more node code than me. > I would prefer their opinion before going forward. This series is indeed very dense. I like the concept of having extensible sub trees in the graph but it feels like the implementation is more complex than it should be. Lacking of another solution, we went for a naive approach in grout. Basically, some nodes have undefined next nodes which are extended using a dedicated API. https://github.com/DPDK/grout/blob/v0.2/modules/infra/datapath/eth_input.c#L23-L31 This API can be used by other nodes to attach themselves to these extensible nodes: https://github.com/DPDK/grout/blob/v0.2/modules/ip/datapath/arp_input.c#L143 https://github.com/DPDK/grout/blob/v0.2/modules/ip/datapath/ip_input.c#L124 https://github.com/DPDK/grout/blob/v0.2/modules/ip6/datapath/ip6_input.c#L122 After which, the extensible nodes can steer the packets towards the correct downstream edge based on the dedicated classifier field: https://github.com/DPDK/grout/blob/v0.2/modules/infra/datapath/eth_input.c#L79 Obviously, this does not natively support a per-interface sub tree traversal, but it can be done in the originating node based on packet private context data. This raises a more important question: how can we standardize the way private application data is passed from node to node? And how could we enforce this declaratively in the node register API? Do you think we could find some middle ground that would not require such extensive changes? Cheers, Robin ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-16 9:38 ` Robin Jarry @ 2024-10-16 13:50 ` Nitin Saxena 2024-10-17 7:03 ` Nitin Saxena 0 siblings, 1 reply; 55+ messages in thread From: Nitin Saxena @ 2024-10-16 13:50 UTC (permalink / raw) To: Robin Jarry Cc: David Marchand, Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Christophe Fontaine Hi Robin, Thanks for the review Please see my replies inline Thanks, Nitin On Wed, Oct 16, 2024 at 3:08 PM Robin Jarry <rjarry@redhat.com> wrote: > > Hi folks, > > David Marchand, Oct 16, 2024 at 11:24: > > On Mon, Oct 14, 2024 at 1:12 PM Nitin Saxena <nsaxena@marvell.com> wrote: > >> I had pushed non RFC patch series before -rc1 date (11th oct). > >> We have an ABI change in this patch series https://patches.dpdk.org/project/dpdk/patch/20241010133111.2764712-3-nsaxena@marvell.com/ > >> Could you help merge this patch series in rc2 otherwise it has to wait for next LTS > > > > Just read through the series, I am not confident with this addition. > > It requires a lot of changes in the node code for supporting it, where > > it should be something handled in/facilitated by the graph library > > itself. > > As far as I can tell, it will be very complicated (if not impossible) to > determine in a generic manner whether a packet must be steered towards > a sub tree or not. The decision *must* come from the originating node in > some way or another. Nitin> I am not sure if it *must* always be from the originating node? What about a control plane which wants to enable "IP4 feature" on interface 'X' by assigning IP address? A originating node (say: ip4-input) *must not* activate IP4 lookup sub-graph for interface "X " until control plane assigns any IP address to it. Regarding the complexity of adopting feature arc changes in fast path, - a sub-optimal change for feature-arc would be simple and trivial but at the cost of performance. - Complexity increases when feature arc changes are optimally integrated (like "ip4_rewrite" changes in the patch) with no performance regression > > > I did not read much from Robin or Christophe who have been writing > > more node code than me. > > I would prefer their opinion before going forward. > > This series is indeed very dense. I like the concept of having > extensible sub trees in the graph but it feels like the implementation > is more complex than it should be. > > Lacking of another solution, we went for a naive approach in grout. > Basically, some nodes have undefined next nodes which are extended using > a dedicated API. Nitin> With an initial glance, it looks like "grout" is trying to solve a use-case where a child is being added to the parent's undefined next node. This is trying to create a runtime parent-child relationship On the other hand, feature arc not just create parent-child relationships but also sibling-sibling relationships as well. Also enabling sub-graph per interface is critical functionality in feature arc that adds complexity Let's assume a use-case in ingress direction, at the IPv4 layer, where IPv4-input is the *originating node* and - On interface X, IPsec-policy, IP4-classify() and IPv4-lookup sub-graphs are enabled in a sequential order - On interface Y, IP4-classify() and IPv4-lookup sub-graphs are enabled. in a sequential order. i.e. IPsec-policy is *disabled* on interface Y In fast path, following processing should happen for "mbuf0" which is received on interface "X" - "ipv4-input" sends mbuf0 to the first enabled sub-graph node for interface X, "IPsec-policy" - In "IPsec-policy" node processing, if policy action results in "bypass" action for mbuf0, it must then be sent to next enabled sub-graph i.e. "IPv4-classify" (from "IPsec-policy" node) - In "IPv4-classify" node processing, if classify fails for mbuf0 then it should finally be sent to "IPv4-lookup" node (from "IPv4-classify" node) whereas for "mbuf1" received on interface Y following fast path processing must happen - "Ipv4-input" sends mbuf1 to the first enabled sub-graph node for interface Y, "IPv4-classify" - If "IPv4-classify" fails for mbuf1, then it should finally be sent to IPv4-lookup node To behave differently for interface X and interface Y as above - First of all, IPsec-policy/IPv4-classify/IPv4-lookup must be connected to "ipv4-input" node (Parent-Child relationship) - Also, IPsec-policy/IPv4-classify/IPv4-lookup must also be connected with each other (Sibling-Sibling relationship) - Fast path APIs provide *rte_edges_t* to send mbuf from one node to another node 1. Based on interface (either Interface X or Interface Y) 2. Based on which node, fast path APIs are called. Next enabled feature/sub-graph can only be determined from previous enabled feature/sub-graph in fast path Not sure if grout handles above use-cases in the same manner. AFAIR , for any control plane change grout re-creates "graph" objects which may not be required with feature arc. > > https://github.com/DPDK/grout/blob/v0.2/modules/infra/datapath/eth_input.c#L23-L31 > > This API can be used by other nodes to attach themselves to these > extensible nodes: > > https://github.com/DPDK/grout/blob/v0.2/modules/ip/datapath/arp_input.c#L143 > https://github.com/DPDK/grout/blob/v0.2/modules/ip/datapath/ip_input.c#L124 > https://github.com/DPDK/grout/blob/v0.2/modules/ip6/datapath/ip6_input.c#L122 > > After which, the extensible nodes can steer the packets towards the > correct downstream edge based on the dedicated classifier field: > > https://github.com/DPDK/grout/blob/v0.2/modules/infra/datapath/eth_input.c#L79 > > Obviously, this does not natively support a per-interface sub tree > traversal, but it can be done in the originating node based on packet > private context data. Nitin> Expressing per interface sub-tree traversal is the key functionality of feature arc. > > This raises a more important question: how can we standardize the way > private application data is passed from node to node? And how could we > enforce this declaratively in the node register API? Nitin> What you are suggesting here (node to node application data exchange) can be done in rte_node_register API but, IMO, this is not related to feature arc. Feature arc is not just between parent and child nodes but also between siblings (as explained above) > > Do you think we could find some middle ground that would not require > such extensive changes? Nitin> If you are pointing to ipv4-rewrite changes, I had an internal review comment of adding those changes without any *performance regression*. A sub-optimal code would be much simpler but at the cost of performance. > > Cheers, > Robin > ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-16 13:50 ` Nitin Saxena @ 2024-10-17 7:03 ` Nitin Saxena 2024-10-17 7:50 ` Robin Jarry 0 siblings, 1 reply; 55+ messages in thread From: Nitin Saxena @ 2024-10-17 7:03 UTC (permalink / raw) To: Robin Jarry Cc: David Marchand, Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Christophe Fontaine Hi Robin/David and all, We realized the feature arc patch series is difficult to understand as a new concept. Our objectives are following with feature arc changes 1. Allow reusability of standard DPDK nodes (defined in lib/nodes/*) with out-of-tree applications (like grout). Currently out-of-tree graph applications are duplicating standard nodes but not reusing the standard ones which are available. In the long term, we would like to mature standard DPDK nodes with flexibility of hooking them to out-of-tree application nodes. 2. Flexibility to enable/disable sub-graphs per interface based on the runtime configuration updates. Protocol sub-graphs can be selectively enabled for few (or all interfaces) at runtime 3. More than one sub-graphs/features can be enabled on an interface. So a packet has to follow a sequential ordering node path on worker cores. Packets may need to move from one sub-graph to another sub-graph per interface 4. Last but not least, an optimized implementation which does not (or minimally) stop worker cores for any control plane runtime updates. Any performance regression should also be avoided I am planning to create a draft presentation on feature arc which I can share, when ready, to discuss. If needed, I can also plan to present that in one of the DPDK community meetings. Their we can also discuss if there are any alternatives of achieving above objectives Thanks, Nitin . On Wed, Oct 16, 2024 at 7:20 PM Nitin Saxena <nsaxena16@gmail.com> wrote: > > Hi Robin, > > Thanks for the review > Please see my replies inline > > Thanks, > Nitin > > On Wed, Oct 16, 2024 at 3:08 PM Robin Jarry <rjarry@redhat.com> wrote: > > > > Hi folks, > > > > David Marchand, Oct 16, 2024 at 11:24: > > > On Mon, Oct 14, 2024 at 1:12 PM Nitin Saxena <nsaxena@marvell.com> wrote: > > >> I had pushed non RFC patch series before -rc1 date (11th oct). > > >> We have an ABI change in this patch series https://patches.dpdk.org/project/dpdk/patch/20241010133111.2764712-3-nsaxena@marvell.com/ > > >> Could you help merge this patch series in rc2 otherwise it has to wait for next LTS > > > > > > Just read through the series, I am not confident with this addition. > > > It requires a lot of changes in the node code for supporting it, where > > > it should be something handled in/facilitated by the graph library > > > itself. > > > > As far as I can tell, it will be very complicated (if not impossible) to > > determine in a generic manner whether a packet must be steered towards > > a sub tree or not. The decision *must* come from the originating node in > > some way or another. > > Nitin> I am not sure if it *must* always be from the originating node? > What about a control plane which wants to enable "IP4 feature" on > interface 'X' by assigning IP address? > A originating node (say: ip4-input) *must not* activate IP4 lookup > sub-graph for interface "X " until control plane assigns any IP > address to it. > > Regarding the complexity of adopting feature arc changes in fast path, > - a sub-optimal change for feature-arc would be simple and trivial but > at the cost of performance. > - Complexity increases when feature arc changes are optimally > integrated (like "ip4_rewrite" changes in the patch) with no > performance regression > > > > > > I did not read much from Robin or Christophe who have been writing > > > more node code than me. > > > I would prefer their opinion before going forward. > > > > This series is indeed very dense. I like the concept of having > > extensible sub trees in the graph but it feels like the implementation > > is more complex than it should be. > > > > Lacking of another solution, we went for a naive approach in grout. > > Basically, some nodes have undefined next nodes which are extended using > > a dedicated API. > > Nitin> With an initial glance, it looks like "grout" is trying to > solve a use-case where a child is being added to the parent's > undefined next node. This is trying to create a runtime parent-child > relationship > > On the other hand, feature arc not just create parent-child > relationships but also sibling-sibling relationships as well. Also > enabling sub-graph per interface is critical functionality in feature > arc that adds complexity > > Let's assume a use-case in ingress direction, at the IPv4 layer, > where IPv4-input is the *originating node* and > > - On interface X, IPsec-policy, IP4-classify() and IPv4-lookup > sub-graphs are enabled in a sequential order > - On interface Y, IP4-classify() and IPv4-lookup sub-graphs are > enabled. in a sequential order. i.e. IPsec-policy is *disabled* on > interface Y > > In fast path, following processing should happen for "mbuf0" which is > received on interface "X" > - "ipv4-input" sends mbuf0 to the first enabled sub-graph node for > interface X, "IPsec-policy" > - In "IPsec-policy" node processing, if policy action results in > "bypass" action for mbuf0, it must then be sent to next enabled > sub-graph i.e. "IPv4-classify" (from "IPsec-policy" node) > - In "IPv4-classify" node processing, if classify fails for mbuf0 then > it should finally be sent to "IPv4-lookup" node (from "IPv4-classify" > node) > > whereas for "mbuf1" received on interface Y following fast path > processing must happen > - "Ipv4-input" sends mbuf1 to the first enabled sub-graph node for > interface Y, "IPv4-classify" > - If "IPv4-classify" fails for mbuf1, then it should finally be sent > to IPv4-lookup node > > To behave differently for interface X and interface Y as above > - First of all, IPsec-policy/IPv4-classify/IPv4-lookup must be > connected to "ipv4-input" node (Parent-Child relationship) > - Also, IPsec-policy/IPv4-classify/IPv4-lookup must also be connected > with each other (Sibling-Sibling relationship) > - Fast path APIs provide *rte_edges_t* to send mbuf from one node to > another node > 1. Based on interface (either Interface X or Interface Y) > 2. Based on which node, fast path APIs are called. Next enabled > feature/sub-graph can only be determined from previous enabled > feature/sub-graph in fast path > > Not sure if grout handles above use-cases in the same manner. AFAIR , > for any control plane change grout re-creates "graph" objects which > may not be required with feature arc. > > > > > https://github.com/DPDK/grout/blob/v0.2/modules/infra/datapath/eth_input.c#L23-L31 > > > > This API can be used by other nodes to attach themselves to these > > extensible nodes: > > > > https://github.com/DPDK/grout/blob/v0.2/modules/ip/datapath/arp_input.c#L143 > > https://github.com/DPDK/grout/blob/v0.2/modules/ip/datapath/ip_input.c#L124 > > https://github.com/DPDK/grout/blob/v0.2/modules/ip6/datapath/ip6_input.c#L122 > > > > After which, the extensible nodes can steer the packets towards the > > correct downstream edge based on the dedicated classifier field: > > > > https://github.com/DPDK/grout/blob/v0.2/modules/infra/datapath/eth_input.c#L79 > > > > Obviously, this does not natively support a per-interface sub tree > > traversal, but it can be done in the originating node based on packet > > private context data. > > Nitin> Expressing per interface sub-tree traversal is the key > functionality of feature arc. > > > > > This raises a more important question: how can we standardize the way > > private application data is passed from node to node? And how could we > > enforce this declaratively in the node register API? > > Nitin> What you are suggesting here (node to node application data > exchange) can be done in rte_node_register API but, IMO, this is not > related to feature arc. > Feature arc is not just between parent and child nodes but also > between siblings (as explained above) > > > > > Do you think we could find some middle ground that would not require > > such extensive changes? > > Nitin> If you are pointing to ipv4-rewrite changes, I had an internal > review comment of adding those changes without any *performance > regression*. > A sub-optimal code would be much simpler but at the cost of performance. > > > > > Cheers, > > Robin > > ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-17 7:03 ` Nitin Saxena @ 2024-10-17 7:50 ` Robin Jarry 2024-10-17 8:32 ` [EXTERNAL] " Christophe Fontaine 2024-10-17 8:48 ` [EXTERNAL] " Nitin Saxena 0 siblings, 2 replies; 55+ messages in thread From: Robin Jarry @ 2024-10-17 7:50 UTC (permalink / raw) To: Nitin Saxena Cc: David Marchand, Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Christophe Fontaine Hi Nitin, all, Nitin Saxena, Oct 17, 2024 at 09:03: > Hi Robin/David and all, > > We realized the feature arc patch series is difficult to understand as > a new concept. Our objectives are following with feature arc changes > > 1. Allow reusability of standard DPDK nodes (defined in lib/nodes/*) > with out-of-tree applications (like grout). Currently out-of-tree > graph applications are duplicating standard nodes but not reusing > the standard ones which are available. In the long term, we would > like to mature standard DPDK nodes with flexibility of hooking them > to out-of-tree application nodes. It would be ideal if the in-built nodes could be reused. When we started working on grout, I tried multiple approaches where I could reuse these nodes, but all failed. The nodes public API seems tailored for app/graph but does not fit well with other control plane implementations. One of the main issues I had is that the ethdev_rx and ethdev_tx nodes are cloned per rxq / txq associated with a graph worker. The rte_node API requires that every clone has a unique name. This in turn makes hot plugging of DPDK ports very complex, if not impossible. For example, with the in-built nodes, it is not possible to change the number of ports or their number of RX queues without destroying the whole graph and creating a new one from scratch. Also, the current implementation of "ip{4,6}-rewrite" handles writing ethernet header data. This would prevent it from using this node for an IP-in-IP tunnel interface as we did in grout. Do you think we could change the in-built nodes to enforce OSI layer separation of concerns? It would make them much more flexible. It may cause a slight drop of performance because you'd be splitting processing in two different nodes. But I think flexibility is more important. Otherwise, the in-built nodes can only be used for very specific use-cases. Finally, I would like to improve the rte_node API to allow defining and enforcing per-packet metadata that every node expects as input. The current in-built nodes rely on mbuf dynamic fields for this but this means you only have 9x32 bits available. And using all of these may break some drivers (ixgbe) that rely on dynfields to work. Have you considered using mbuf private data for this? > > 2. Flexibility to enable/disable sub-graphs per interface based on the > runtime configuration updates. Protocol sub-graphs can be > selectively enabled for few (or all interfaces) at runtime > > 3. More than one sub-graphs/features can be enabled on an interface. > So a packet has to follow a sequential ordering node path on worker > cores. Packets may need to move from one sub-graph to another > sub-graph per interface > > 4. Last but not least, an optimized implementation which does not (or > minimally) stop worker cores for any control plane runtime updates. > Any performance regression should also be avoided > > I am planning to create a draft presentation on feature arc which > I can share, when ready, to discuss. If needed, I can also plan to > present that in one of the DPDK community meetings. Their we can also > discuss if there are any alternatives of achieving above objectives Looking forward to this. Thanks! ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-17 7:50 ` Robin Jarry @ 2024-10-17 8:32 ` Christophe Fontaine 2024-10-17 10:56 ` Nitin Saxena 2024-10-17 8:48 ` [EXTERNAL] " Nitin Saxena 1 sibling, 1 reply; 55+ messages in thread From: Christophe Fontaine @ 2024-10-17 8:32 UTC (permalink / raw) To: Robin Jarry Cc: Nitin Saxena, David Marchand, Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev Hi all, What about the following steps: - update the nodes so they work on the current layer (example: for all L3 nodes, the current mbuf data offset *must* be pointing to the IP header) - define a public data structure that would be shared across nodes through priv data, and not dynfields ? This structure would be the "internal api" (so, that has to be tracked across dpdk releases) between nodes. We’d need common data shared for all the nodes as well as specific data between 2 nodes. As we get to this point, this (hopefully) will help with the node reusability. - Update the feature arcs to leverage this well known structure, and refine the api - Define which part of the stack needs to be defined as a feature arc, with the benefit of the generic API to enable/disable that feature, and which part needs to be dynamically pluggable. For instance, for a router, it may not make sense to define IPv4 support as a feature arc. So, we’d statically connect eth_input to ip_input. Yet, lldp support is a good candidate for a feature arc: we need to configure it per interface, and this is independent of the main graph. WDYT? Christophe > On 17 Oct 2024, at 09:50, Robin Jarry <rjarry@redhat.com> wrote: > > Hi Nitin, all, > > Nitin Saxena, Oct 17, 2024 at 09:03: >> Hi Robin/David and all, >> >> We realized the feature arc patch series is difficult to understand as a new concept. Our objectives are following with feature arc changes >> >> 1. Allow reusability of standard DPDK nodes (defined in lib/nodes/*) with out-of-tree applications (like grout). Currently out-of-tree graph applications are duplicating standard nodes but not reusing the standard ones which are available. In the long term, we would like to mature standard DPDK nodes with flexibility of hooking them to out-of-tree application nodes. > > It would be ideal if the in-built nodes could be reused. When we started working on grout, I tried multiple approaches where I could reuse these nodes, but all failed. The nodes public API seems tailored for app/graph but does not fit well with other control plane implementations. > > One of the main issues I had is that the ethdev_rx and ethdev_tx nodes are cloned per rxq / txq associated with a graph worker. The rte_node API requires that every clone has a unique name. This in turn makes hot plugging of DPDK ports very complex, if not impossible. > > For example, with the in-built nodes, it is not possible to change the number of ports or their number of RX queues without destroying the whole graph and creating a new one from scratch. > > Also, the current implementation of "ip{4,6}-rewrite" handles writing ethernet header data. This would prevent it from using this node for an IP-in-IP tunnel interface as we did in grout. > > Do you think we could change the in-built nodes to enforce OSI layer separation of concerns? It would make them much more flexible. It may cause a slight drop of performance because you'd be splitting processing in two different nodes. But I think flexibility is more important. Otherwise, the in-built nodes can only be used for very specific use-cases. > > Finally, I would like to improve the rte_node API to allow defining and enforcing per-packet metadata that every node expects as input. The current in-built nodes rely on mbuf dynamic fields for this but this means you only have 9x32 bits available. And using all of these may break some drivers (ixgbe) that rely on dynfields to work. Have you considered using mbuf private data for this? > >> >> 2. Flexibility to enable/disable sub-graphs per interface based on the runtime configuration updates. Protocol sub-graphs can be selectively enabled for few (or all interfaces) at runtime >> >> 3. More than one sub-graphs/features can be enabled on an interface. So a packet has to follow a sequential ordering node path on worker cores. Packets may need to move from one sub-graph to another sub-graph per interface >> >> 4. Last but not least, an optimized implementation which does not (or minimally) stop worker cores for any control plane runtime updates. Any performance regression should also be avoided >> >> I am planning to create a draft presentation on feature arc which I can share, when ready, to discuss. If needed, I can also plan to present that in one of the DPDK community meetings. Their we can also discuss if there are any alternatives of achieving above objectives > > Looking forward to this. > > Thanks! > ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-17 8:32 ` [EXTERNAL] " Christophe Fontaine @ 2024-10-17 10:56 ` Nitin Saxena 0 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-17 10:56 UTC (permalink / raw) To: Christophe Fontaine Cc: Robin Jarry, David Marchand, Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev Hi Christophe, Please see inline comments Thanks, Nitin On Thu, Oct 17, 2024 at 2:02 PM Christophe Fontaine <cfontain@redhat.com> wrote: > > Hi all, > > What about the following steps: > - update the nodes so they work on the current layer (example: for all L3 nodes, the current mbuf data offset *must* be pointing to the IP header) Agreed. It would be better if nodes uses rte_pktmbuf_[append()/shrink() etc..] APIs to manipulate layer data offset > - define a public data structure that would be shared across nodes through priv data, and not dynfields ? Eventually public data structures should be defined to serve *a purpose*. Do you refer to creating a generic public structure? If yes, IMO, it may not be tuned for performance IMO, we need to create public structures for each specific purpose. Feature arc is also a public data structure which optimally saves following variables in 8 byte compact structure (rte_graph_feature_daa_t) for every interface - rte_edge_t (uint16_t) - next enabled feature (uint8_t) per index (from current node) - Per interface feature specific user_data (uint32_t) Due to its compact nature, 8 such objects per interface can be saved in one 64B cache line. So IMO, it is better to create public structures for a given purpose and optimally define them fields and APIs. Also feature arc specific 3B data is saved in mbuf dynfield. Hard to say if priv data would provide a better solution. > This structure would be the "internal api" (so, that has to be tracked across dpdk releases) between nodes. > We’d need common data shared for all the nodes as well as specific data between 2 nodes. > As we get to this point, this (hopefully) will help with the node reusability. Feature arc also maintains data between 2 nodes per interface and also for all nodes which are added as features. > > - Update the feature arcs to leverage this well known structure, and refine the api > - Define which part of the stack needs to be defined as a feature arc, with the benefit of the generic API to enable/disable that feature, and which part needs to be dynamically pluggable. > For instance, for a router, it may not make sense to define IPv4 support as a feature arc. > So, we’d statically connect eth_input to ip_input. Agreed > Yet, lldp support is a good candidate for a feature arc: we need to configure it per interface, and this is independent of the main graph. > There would be more protocols which need to be enabled per interface > WDYT? > Christophe > > > On 17 Oct 2024, at 09:50, Robin Jarry <rjarry@redhat.com> wrote: > > > > Hi Nitin, all, > > > > Nitin Saxena, Oct 17, 2024 at 09:03: > >> Hi Robin/David and all, > >> > >> We realized the feature arc patch series is difficult to understand as a new concept. Our objectives are following with feature arc changes > >> > >> 1. Allow reusability of standard DPDK nodes (defined in lib/nodes/*) with out-of-tree applications (like grout). Currently out-of-tree graph applications are duplicating standard nodes but not reusing the standard ones which are available. In the long term, we would like to mature standard DPDK nodes with flexibility of hooking them to out-of-tree application nodes. > > > > It would be ideal if the in-built nodes could be reused. When we started working on grout, I tried multiple approaches where I could reuse these nodes, but all failed. The nodes public API seems tailored for app/graph but does not fit well with other control plane implementations. > > > > One of the main issues I had is that the ethdev_rx and ethdev_tx nodes are cloned per rxq / txq associated with a graph worker. The rte_node API requires that every clone has a unique name. This in turn makes hot plugging of DPDK ports very complex, if not impossible. > > > > For example, with the in-built nodes, it is not possible to change the number of ports or their number of RX queues without destroying the whole graph and creating a new one from scratch. > > > > Also, the current implementation of "ip{4,6}-rewrite" handles writing ethernet header data. This would prevent it from using this node for an IP-in-IP tunnel interface as we did in grout. > > > > Do you think we could change the in-built nodes to enforce OSI layer separation of concerns? It would make them much more flexible. It may cause a slight drop of performance because you'd be splitting processing in two different nodes. But I think flexibility is more important. Otherwise, the in-built nodes can only be used for very specific use-cases. > > > > Finally, I would like to improve the rte_node API to allow defining and enforcing per-packet metadata that every node expects as input. The current in-built nodes rely on mbuf dynamic fields for this but this means you only have 9x32 bits available. And using all of these may break some drivers (ixgbe) that rely on dynfields to work. Have you considered using mbuf private data for this? > > > >> > >> 2. Flexibility to enable/disable sub-graphs per interface based on the runtime configuration updates. Protocol sub-graphs can be selectively enabled for few (or all interfaces) at runtime > >> > >> 3. More than one sub-graphs/features can be enabled on an interface. So a packet has to follow a sequential ordering node path on worker cores. Packets may need to move from one sub-graph to another sub-graph per interface > >> > >> 4. Last but not least, an optimized implementation which does not (or minimally) stop worker cores for any control plane runtime updates. Any performance regression should also be avoided > >> > >> I am planning to create a draft presentation on feature arc which I can share, when ready, to discuss. If needed, I can also plan to present that in one of the DPDK community meetings. Their we can also discuss if there are any alternatives of achieving above objectives > > > > Looking forward to this. > > > > Thanks! > > > ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] Re: [RFC PATCH 0/3] add feature arc in rte_graph 2024-10-17 7:50 ` Robin Jarry 2024-10-17 8:32 ` [EXTERNAL] " Christophe Fontaine @ 2024-10-17 8:48 ` Nitin Saxena 1 sibling, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-17 8:48 UTC (permalink / raw) To: Robin Jarry Cc: David Marchand, Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, dev, Christophe Fontaine Hi Robin, See inline comments Thanks, Nitin On Thu, Oct 17, 2024 at 1:20 PM Robin Jarry <rjarry@redhat.com> wrote: > > Hi Nitin, all, > > Nitin Saxena, Oct 17, 2024 at 09:03: > > Hi Robin/David and all, > > > > We realized the feature arc patch series is difficult to understand as > > a new concept. Our objectives are following with feature arc changes > > > > 1. Allow reusability of standard DPDK nodes (defined in lib/nodes/*) > > with out-of-tree applications (like grout). Currently out-of-tree > > graph applications are duplicating standard nodes but not reusing > > the standard ones which are available. In the long term, we would > > like to mature standard DPDK nodes with flexibility of hooking them > > to out-of-tree application nodes. > > It would be ideal if the in-built nodes could be reused. When we started > working on grout, I tried multiple approaches where I could reuse these > nodes, but all failed. The nodes public API seems tailored for app/graph > but does not fit well with other control plane implementations. > > One of the main issues I had is that the ethdev_rx and ethdev_tx nodes > are cloned per rxq / txq associated with a graph worker. The rte_node > API requires that every clone has a unique name. This in turn makes hot > plugging of DPDK ports very complex, if not impossible. Agreed. I guess hot plugging of DPDK ports was not the objective when initial changes went in. But we can add hot-plugging functionality without affecting performance > > For example, with the in-built nodes, it is not possible to change the > number of ports or their number of RX queues without destroying the > whole graph and creating a new one from scratch. Coincidentally, I have also encountered these technical issues while writing an out-of-tree application [1]. I had internal discussions with @Jerin Jacob and other graph maintainers to fix these shortcomings. If you want, we can collaborate on fixing these issues For [port, rq] pair mapping to worker core, I have an alternate design [2] which currently stops worker cores. It can be enhanced by RCU based scheme for an ideal DPDK implementation [1]: https://marvellembeddedprocessors.github.io/dao/guides/applications/secgw-graph.html [2]: https://github.com/MarvellEmbeddedProcessors/dao/blob/dao-devel/app/secgw-graph/nodes/rxtx/ethdev-rx.c#L27 > > Also, the current implementation of "ip{4,6}-rewrite" handles writing > ethernet header data. This would prevent it from using this node for an > IP-in-IP tunnel interface as we did in grout. For IP-in-IP, a separate rewrite node would be required which computes checksum etc. but not add rewrite data. > > Do you think we could change the in-built nodes to enforce OSI layer > separation of concerns? It would make them much more flexible. Yes. We are also in agreement to make RFC compliant optimized in-built nodes with such flexibility in place. > It may > cause a slight drop of performance because you'd be splitting processing > in two different nodes. But I think flexibility is more important. > Otherwise, the in-built nodes can only be used for very specific > use-cases. > > Finally, I would like to improve the rte_node API to allow defining and > enforcing per-packet metadata that every node expects as input. The > current in-built nodes rely on mbuf dynamic fields for this but this > means you only have 9x32 bits available. And using all of these may > break some drivers (ixgbe) that rely on dynfields to work. Have you > considered using mbuf private data for this? IMO, "node_mbuf_priv_t" would be ideal for most of the use-cases as it fits in second 64B cache line. With mbuf private data, fast path have to access another cache line per packet which may not be efficient from performance PoV. But we can discuss in more detail about it. Although, I thought of adding "sw_if_index" (which is not same as port_id) to accommodate IP-in-IP like software interfaces > > > > > 2. Flexibility to enable/disable sub-graphs per interface based on the > > runtime configuration updates. Protocol sub-graphs can be > > selectively enabled for few (or all interfaces) at runtime > > > > 3. More than one sub-graphs/features can be enabled on an interface. > > So a packet has to follow a sequential ordering node path on worker > > cores. Packets may need to move from one sub-graph to another > > sub-graph per interface > > > > 4. Last but not least, an optimized implementation which does not (or > > minimally) stop worker cores for any control plane runtime updates. > > Any performance regression should also be avoided > > > > I am planning to create a draft presentation on feature arc which > > I can share, when ready, to discuss. If needed, I can also plan to > > present that in one of the DPDK community meetings. Their we can also > > discuss if there are any alternatives of achieving above objectives > > Looking forward to this. Sure. Will share ppt asap > > Thanks! > ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH v2 0/5] add feature arc in rte_graph 2024-09-07 7:31 [RFC PATCH 0/3] add feature arc in rte_graph Nitin Saxena ` (3 preceding siblings ...) 2024-10-08 8:04 ` [RFC PATCH 0/3] add feature arc in rte_graph David Marchand @ 2024-10-08 13:30 ` Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 1/5] graph: add feature arc support Nitin Saxena ` (5 more replies) 4 siblings, 6 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on whether feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Changes in v2: - Added unit tests for feature arc - Fixed issues found in testing - Added new public APIs rte_graph_feature_arc_feature_to_node(), rte_graph_feature_arc_feature_to_name(), rte_graph_feature_arc_num_features() - Added programming guide for feature arc - Added release notes for feature arc Nitin Saxena (5): graph: add feature arc support graph: add feature arc option in graph create graph: add IPv4 output feature arc test/graph_feature_arc: add functional tests docs: add programming guide for feature arc app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1415 +++++++++++++++++++ doc/guides/prog_guide/graph_lib.rst | 289 ++++ doc/guides/prog_guide/img/feature_arc-1.jpg | Bin 0 -> 48984 bytes doc/guides/prog_guide/img/feature_arc-2.jpg | Bin 0 -> 113287 bytes doc/guides/prog_guide/img/feature_arc-3.jpg | Bin 0 -> 93408 bytes doc/guides/rel_notes/release_24_11.rst | 11 +- lib/graph/graph.c | 1 + lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++ lib/graph/graph_populate.c | 7 +- lib/graph/graph_private.h | 3 + lib/graph/meson.build | 2 + lib/graph/node.c | 2 + lib/graph/rte_graph.h | 3 + lib/graph/rte_graph_feature_arc.h | 429 ++++++ lib/graph/rte_graph_feature_arc_worker.h | 672 +++++++++ lib/graph/version.map | 20 + lib/node/ip4_rewrite.c | 476 +++++-- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 21 files changed, 4493 insertions(+), 99 deletions(-) create mode 100644 app/test/test_graph_feature_arc.c create mode 100644 doc/guides/prog_guide/img/feature_arc-1.jpg create mode 100644 doc/guides/prog_guide/img/feature_arc-2.jpg create mode 100644 doc/guides/prog_guide/img/feature_arc-3.jpg create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH v2 1/5] graph: add feature arc support 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena @ 2024-10-08 13:30 ` Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 2/5] graph: add feature arc option in graph create Nitin Saxena ` (4 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena add feature arc to allow dynamic steering of packets across graph nodes based on protocol features enabled on incoming or outgoing interface Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 11 +- lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++++++++ lib/graph/meson.build | 2 + lib/graph/rte_graph_feature_arc.h | 429 ++++++++ lib/graph/rte_graph_feature_arc_worker.h | 672 ++++++++++++ lib/graph/version.map | 20 + 6 files changed, 2356 insertions(+), 1 deletion(-) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index 0ff70d9057..24852aa8e0 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -23,7 +23,6 @@ DPDK Release 24.11 New Features ------------ - .. This section should contain new features added in this release. Sample format: @@ -55,6 +54,16 @@ New Features Also, make sure to start the actual text at the margin. ======================================================= +* **Added feature arc abstraction in graph library.** + + Feature arc abstraction helps ``rte_graph`` based applications to steer + packets across different node path(s) based on the features (or protocols) + enabled on interfaces. Different feature node paths can be enabled/disabled + at runtime on some or on all interfaces. This abstraction also help + applications to hook their ``custom nodes`` in standard DPDK node paths + without any code changes in the later. + + * Added ``ip4-output`` feature arc processing in ``ip4_rewrite`` node. Removed Items ------------- diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c new file mode 100644 index 0000000000..ff99f7b26a --- /dev/null +++ b/lib/graph/graph_feature_arc.c @@ -0,0 +1,1223 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "graph_private.h" +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> + +#define ARC_PASSIVE_LIST(arc) (arc->active_feature_list ^ 0x1) + +#define rte_graph_uint_cast(x) ((unsigned int)x) +#define feat_dbg graph_dbg + +static rte_graph_feature_arc_main_t *__rte_graph_feature_arc_main; + +/* Make sure fast path cache line is compact */ +_Static_assert((offsetof(struct rte_graph_feature_arc, slow_path_variables) + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) + <= RTE_CACHE_LINE_SIZE, + "Fast path feature arc variables exceed cache line size"); + +#define connect_graph_nodes(node1, node2, edge, arc_name) \ + __connect_graph_nodes(node1, node2, edge, arc_name, __LINE__) + +#define FEAT_COND_ERR(cond, fmt, ...) \ + do { \ + if (cond) \ + graph_err(fmt, ##__VA_ARGS__); \ + } while (0) + +/* + * lookup feature name and get control path node_list as well as feature index + * at which it is inserted + */ +static int +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = UINT32_MAX; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + fi++; + } + return -1; +} + +/* Lookup used only during rte_graph_feature_add() */ +static int +feature_add_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + /* Update slot where new feature can be added */ + if (slot) + *slot = fi; + fi++; + } + + return -1; +} + +/* Get control path node info from provided input feature_index */ +static int +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t feature_index, + struct rte_graph_feature_node_list **ppfinfo, + const int do_sanity_check) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + if (!ppfinfo) + return -1; + + *ppfinfo = NULL; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + /* Check sanity */ + if (do_sanity_check) + if (finfo->node_index != index) + RTE_VERIFY(0); + if (index == feature_index) { + *ppfinfo = finfo; + return 0; + } + index++; + } + return -1; +} + +/* prepare feature arc after addition of all features */ +static void +prepare_feature_arc_before_first_enable(struct rte_graph_feature_arc *arc) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + arc->active_feature_list = 0; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + finfo->node_index = index; + feat_dbg("\t%s prepare: %s added to list at index: %u", arc->feature_arc_name, + finfo->feature_node->name, index); + index++; + } +} + +/* feature arc lookup in array */ +static int +feature_arc_lookup(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (arc == (rte_graph_feature_arc_get(dm->feature_arcs[iter]))) + return 0; + } + return -1; +} + +/* Check valid values for known fields in arc to make sure arc is sane */ +static int check_feature_arc_sanity(rte_graph_feature_arc_t _arc, int iter) +{ +#ifdef FEATURE_ARC_DEBUG + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + RTE_VERIFY(arc->feature_arc_main == __rte_graph_feature_arc_main); + RTE_VERIFY(arc->feature_arc_index == iter); + + RTE_VERIFY(arc->feature_list[0]->indexed_by_features = arc->features[0]); + RTE_VERIFY(arc->feature_list[1]->indexed_by_features = arc->features[1]); + + RTE_VERIFY(arc->active_feature_list < 2); +#else + RTE_SET_USED(_arc); + RTE_SET_USED(iter); +#endif + return 0; +} + +/* Perform sanity on all arc if any corruption occurred */ +static int do_sanity_all_arcs(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!dm) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (check_feature_arc_sanity(dm->feature_arcs[iter], iter)) + return -1; + } + return 0; +} + +/* get existing edge from parent_node -> child_node */ +static int +get_existing_edge(const char *arc_name, struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t i, count = 0; + + RTE_SET_USED(arc_name); + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +/* create or retrieve already existing edge from parent_node -> child_node */ +static int +__connect_graph_nodes(struct rte_node_register *parent_node, struct rte_node_register *child_node, + rte_edge_t *_edge, char *arc_name, int lineno) +{ + const char *next_node = NULL; + rte_edge_t edge; + + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { + feat_dbg("\t%s/%d: %s[%u]: \"%s\", edge reused", arc_name, lineno, + parent_node->name, edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; + } + + /* Node to be added */ + next_node = child_node->name; + + edge = rte_node_edge_update(parent_node->id, RTE_EDGE_ID_INVALID, &next_node, 1); + + if (edge == RTE_EDGE_ID_INVALID) { + graph_err("edge invalid"); + return -1; + } + edge = rte_node_edge_count(parent_node->id) - 1; + + feat_dbg("\t%s/%d: %s[%u]: \"%s\", new edge added", arc_name, lineno, parent_node->name, + edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; +} + +/* feature arc initialization */ +static int +feature_arc_main_init(rte_graph_feature_arc_main_t **pfl, uint32_t max_feature_arcs) +{ + rte_graph_feature_arc_main_t *pm = NULL; + uint32_t i; + size_t sz; + + if (!pfl) + return -1; + + sz = sizeof(rte_graph_feature_arc_main_t) + + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); + + pm = rte_malloc("rte_graph_feature_arc_main", sz, 0); + if (!pm) + return -1; + + memset(pm, 0, sz); + + for (i = 0; i < max_feature_arcs; i++) + pm->feature_arcs[i] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + pm->max_feature_arcs = max_feature_arcs; + + *pfl = pm; + + return 0; +} + +/* feature arc initialization, public API */ +int +rte_graph_feature_arc_init(int max_feature_arcs) +{ + if (!max_feature_arcs) + return -1; + + if (__rte_graph_feature_arc_main) + return -1; + + return feature_arc_main_init(&__rte_graph_feature_arc_main, max_feature_arcs); +} + +/* reset feature list before switching to passive list */ +static void +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t list_index) +{ + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + uint32_t i, j; + + list = arc->feature_list[list_index]; + feat = arc->features[list_index]; + + /*Initialize variables*/ + memset(feat, 0, arc->feature_size * arc->max_features); + memset(list, 0, arc->feature_list_size); + + /* Initialize feature and feature_data */ + for (i = 0; i < arc->max_features; i++) { + feat = __rte_graph_feature_get(arc, i, list_index); + feat->this_feature_index = i; + + for (j = 0; j < arc->max_indexes; j++) { + fdata = rte_graph_feature_data_get(arc, feat, j); + fdata->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + fdata->next_edge = UINT16_MAX; + fdata->user_data = UINT32_MAX; + } + } + + for (i = 0; i < arc->max_indexes; i++) + list->first_enabled_feature_by_index[i] = RTE_GRAPH_FEATURE_INVALID; +} + +static int +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char *flist_name, + rte_graph_feature_list_t **pplist, + struct rte_graph_feature **ppfeature, uint32_t list_index) +{ + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + size_t list_size, feat_size, fdata_size; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + + list_size = sizeof(struct rte_graph_feature_list) + + (sizeof(list->first_enabled_feature_by_index[0]) * arc->max_indexes); + + list_size = RTE_ALIGN_CEIL(list_size, RTE_CACHE_LINE_SIZE); + + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); + if (!list) + return -ENOMEM; + + memset(list, 0, list_size); + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); + + /* Let one feature and its associated data per index capture complete + * cache lines + */ + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + fdata_size, + RTE_CACHE_LINE_SIZE); + + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, "feat"); + + feat = rte_malloc(fname, feat_size * arc->max_features, RTE_CACHE_LINE_SIZE); + if (!feat) { + rte_free(list); + return -ENOMEM; + } + arc->feature_size = feat_size; + arc->feature_data_size = fdata_size; + arc->feature_list_size = list_size; + + /* Initialize list */ + list->indexed_by_features = feat; + *pplist = list; + *ppfeature = feat; + + feature_arc_list_reset(arc, list_index); + + return 0; +} + +/* free resources allocated in feature_arc_list_init() */ +static void +feature_arc_list_destroy(struct rte_graph_feature_arc *arc, int list_index) +{ + rte_graph_feature_list_t *list = NULL; + + list = arc->feature_list[list_index]; + + rte_free(list->indexed_by_features); + + arc->features[list_index] = NULL; + + rte_free(list); + + arc->feature_list[list_index] = NULL; +} + +int +rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, rte_graph_feature_arc_t *_arc) +{ + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_arc_main_t *dfm = NULL; + struct rte_graph_feature_arc *arc = NULL; + struct rte_graph_feature *df = NULL; + uint32_t iter, j, arc_index; + size_t sz; + + if (!_arc) + SET_ERR_JMP(EINVAL, err, "%s: Invalid _arc", feature_arc_name); + + if (max_features < 2) + SET_ERR_JMP(EINVAL, err, "%s: max_features must be greater than 1", + feature_arc_name); + + if (!start_node) + SET_ERR_JMP(EINVAL, err, "%s: start_node cannot be NULL", + feature_arc_name); + + if (!feature_arc_name) + SET_ERR_JMP(EINVAL, err, "%s: feature_arc name cannot be NULL", + feature_arc_name); + + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) + SET_ERR_JMP(EAGAIN, err, "%s: number of features cannot be greater than 64", + feature_arc_name); + + /* + * Application hasn't called rte_graph_feature_arc_init(). Initialize with + * default values + */ + if (!__rte_graph_feature_arc_main) { + if (rte_graph_feature_arc_init((int)RTE_GRAPH_FEATURE_ARC_MAX) < 0) { + graph_err("rte_graph_feature_arc_init() failed"); + return -1; + } + } + + /* If name is not unique */ + if (!rte_graph_feature_arc_lookup_by_name(feature_arc_name, NULL)) + SET_ERR_JMP(EINVAL, err, "%s: feature arc name already exists", + feature_arc_name); + + dfm = __rte_graph_feature_arc_main; + + /* threshold check */ + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) + SET_ERR_JMP(EAGAIN, err, "%s: max number (%u) of feature arcs reached", + feature_arc_name, dfm->max_feature_arcs); + + /* Find the free slot for feature arc */ + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { + if (dfm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + break; + } + arc_index = iter; + + if (arc_index >= dfm->max_feature_arcs) { + graph_err("No free slot found for num_feature_arc"); + return -1; + } + + /* This should not happen */ + RTE_VERIFY(dfm->feature_arcs[arc_index] == RTE_GRAPH_FEATURE_ARC_INITIALIZER); + + /* size of feature arc + feature_bit_mask_by_index */ + sz = RTE_ALIGN_CEIL(sizeof(*arc) + (sizeof(uint64_t) * max_indexes), RTE_CACHE_LINE_SIZE); + + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); + + if (!arc) { + graph_err("malloc failed for feature_arc_create()"); + return -1; + } + + memset(arc, 0, sz); + + /* Initialize rte_graph port group fixed variables */ + STAILQ_INIT(&arc->all_features); + strncpy(arc->feature_arc_name, feature_arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + arc->feature_arc_main = (void *)dfm; + arc->start_node = start_node; + arc->max_features = max_features; + arc->max_indexes = max_indexes; + arc->feature_arc_index = arc_index; + + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc->features[0], 0) < 0) { + rte_free(arc); + graph_err("feature_arc_list_init(0) failed"); + return -1; + } + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc->features[1], 1) < 0) { + feature_arc_list_destroy(arc, 0); + rte_free(arc); + graph_err("feature_arc_list_init(1) failed"); + return -1; + } + + for (iter = 0; iter < arc->max_features; iter++) { + df = rte_graph_feature_get(arc, iter); + for (j = 0; j < arc->max_indexes; j++) { + gfd = rte_graph_feature_data_get(arc, df, j); + gfd->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + } + } + dfm->feature_arcs[arc->feature_arc_index] = (rte_graph_feature_arc_t)arc; + dfm->num_feature_arcs++; + + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + + do_sanity_all_arcs(); + + feat_dbg("Feature arc %s[%p] created with max_features: %u and indexes: %u", + feature_arc_name, (void *)arc, max_features, max_indexes); + return 0; + +err: + return -rte_errno; +} + +int +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *_runs_after, const char *runs_before) +{ + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo = NULL; + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; + char feature_name[3*RTE_GRAPH_FEATURE_ARC_NAMELEN]; + const char *runs_after = NULL; + uint32_t num_feature = 0; + uint32_t slot, add_flag; + rte_edge_t edge = -1; + + /* sanity */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + graph_err("feature arc not created: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (feature_arc_lookup(_arc)) { + graph_err("invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (arc->runtime_enabled_features) { + graph_err("adding features after enabling any one of them is not supported"); + return -1; + } + + if ((_runs_after != NULL) && (runs_before != NULL) && + (_runs_after == runs_before)) { + graph_err("runs_after and runs_before are same '%s:%s]", _runs_after, + runs_before); + return -1; + } + + if (!feature_node) { + graph_err("feature_node: %p invalid", feature_node); + return -1; + } + + arc = rte_graph_feature_arc_get(_arc); + + if (feature_node->id == RTE_NODE_ID_INVALID) { + graph_err("Invalid node: %s", feature_node->name); + return -1; + } + + if (!feature_add_lookup(arc, feature_node->name, &finfo, &slot)) { + graph_err("%s feature already added", feature_node->name); + return -1; + } + + if (slot >= arc->max_features) { + graph_err("%s: Max features %u added to feature arc", + arc->feature_arc_name, slot); + return -1; + } + + if (strstr(feature_node->name, arc->start_node->name)) { + graph_err("Feature %s cannot point to itself: %s", feature_node->name, + arc->start_node->name); + return -1; + } + + feat_dbg("%s: adding feature node: %s at feature index: %u", arc->feature_arc_name, + feature_node->name, slot); + + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc->feature_arc_name)) { + graph_err("unable to connect %s -> %s", arc->start_node->name, feature_node->name); + return -1; + } + + snprintf(feature_name, sizeof(feature_name), "%s-%s-finfo", + arc->feature_arc_name, feature_node->name); + + finfo = rte_malloc(feature_name, sizeof(*finfo), 0); + if (!finfo) { + graph_err("%s/%s: rte_malloc failed", arc->feature_arc_name, feature_node->name); + return -1; + } + + memset(finfo, 0, sizeof(*finfo)); + + finfo->feature_arc = (void *)arc; + finfo->feature_node = feature_node; + finfo->edge_to_this_feature = edge; + arc->runtime_enabled_features = 0; + + /* + * if no constraints given and provided feature is not the first feature, + * explicitly set "runs_after" as last_feature. Handles the case: + * + * add(f1, NULL, NULL); + * add(f2, NULL, NULL); + */ + num_feature = rte_graph_feature_arc_num_features(_arc); + if (!_runs_after && !runs_before && num_feature) + runs_after = rte_graph_feature_arc_feature_to_name(_arc, num_feature - 1); + else + runs_after = _runs_after; + + /* Check for before and after constraints */ + if (runs_before) { + /* runs_before sanity */ + if (feature_lookup(arc, runs_before, &before_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid before feature name: %s", runs_before); + + if (!before_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_before %s does not exist", runs_before); + + /* + * Starting from 0 to runs_before, continue connecting edges + */ + add_flag = 1; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (!add_flag) + /* Nodes after seeing "runs_before", finfo connects to temp*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, + NULL, arc->feature_arc_name); + /* + * As soon as we see runs_before. stop adding edges + */ + if (!strncmp(temp->feature_node->name, runs_before, + RTE_GRAPH_NAMESIZE)) { + if (!connect_graph_nodes(finfo->feature_node, temp->feature_node, + &edge, arc->feature_arc_name)) + add_flag = 0; + } + + if (add_flag) + /* Nodes before seeing "run_before" are connected to finfo */ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + } + } + + if (runs_after) { + if (feature_lookup(arc, runs_after, &after_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid after feature_name %s", runs_after); + + if (!after_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_after %s does not exist", runs_after); + + /* Starting from runs_after to end continue connecting edges */ + add_flag = 0; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (add_flag) + /* We have already seen runs_after now */ + /* Add all features as next node to current feature*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, NULL, + arc->feature_arc_name); + else + /* Connect initial nodes to newly added node*/ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + + /* as soon as we see runs_after. start adding edges + * from next iteration + */ + if (!strncmp(temp->feature_node->name, runs_after, RTE_GRAPH_NAMESIZE)) + add_flag = 1; + } + + /* add feature next to runs_after */ + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, next_feature); + } else { + if (before_finfo) { + /* add finfo before "before_finfo" element in the list */ + after_finfo = NULL; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (before_finfo == temp) { + if (after_finfo) + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, + finfo, next_feature); + else + STAILQ_INSERT_HEAD(&arc->all_features, finfo, + next_feature); + + return 0; + } + after_finfo = temp; + } + } else { + /* Very first feature just needs to be added to list */ + STAILQ_INSERT_TAIL(&arc->all_features, finfo, next_feature); + } + } + + return 0; + +finfo_free: + rte_free(finfo); + + return -1; +} + +int +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot; + + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { + *feat = (rte_graph_feature_t) slot; + return 0; + } + + return -1; +} + +int +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int is_enable_disable, bool emit_logs) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature *gf = NULL; + uint32_t slot; + + /* validate _arc */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + FEAT_COND_ERR(emit_logs, "invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -EINVAL; + } + + /* validate index */ + if (index >= arc->max_indexes) { + FEAT_COND_ERR(emit_logs, "%s: Invalid provided index: %u >= %u configured", + arc->feature_arc_name, index, arc->max_indexes); + return -1; + } + + /* validate feature_name is already added or not */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) { + FEAT_COND_ERR(emit_logs, "%s: No feature %s added", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + if (!finfo) { + FEAT_COND_ERR(emit_logs, "%s: No feature: %s found", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + /* slot should be in valid range */ + if (slot >= arc->max_features) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid free slot %u(max=%u) for feature", + arc->feature_arc_name, feature_name, slot, arc->max_features); + return -EINVAL; + } + + /* slot should be in range of 0 - 63 */ + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid slot: %u", arc->feature_arc_name, + feature_name, slot); + return -EINVAL; + } + + if (finfo->node_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s/%s: lookup slot mismatch for finfo idx: %u and lookup slot: %u", + arc->feature_arc_name, feature_name, finfo->node_index, slot); + return -1; + } + + /* Get feature from active list */ + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(arc)); + if (gf->this_feature_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s: %s rcvd feature_idx: %u does not match with saved: %u", + arc->feature_arc_name, feature_name, slot, gf->this_feature_index); + return -1; + } + + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & + RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s already enabled on index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + if (!is_enable_disable && !arc->runtime_enabled_features) { + FEAT_COND_ERR(emit_logs, "%s: No feature enabled to disable", + arc->feature_arc_name); + return -1; + } + + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s not enabled in bitmask for index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + return 0; +} + +/* + * Before switch to passive list, user_data needs to be copied from active list to passive list + */ +static void +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t dest_list_index, + uint16_t src_list_index) +{ + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; + struct rte_graph_feature *sgf = NULL, *dgf = NULL; + uint32_t i, j; + + for (i = 0; i < arc->max_features; i++) { + sgf = __rte_graph_feature_get(arc, i, src_list_index); + dgf = __rte_graph_feature_get(arc, i, dest_list_index); + for (j = 0; j < arc->max_indexes; j++) { + sgfd = rte_graph_feature_data_get(arc, sgf, j); + dgfd = rte_graph_feature_data_get(arc, dgf, j); + dgfd->user_data = sgfd->user_data; + } + } +} +/* + * Fill fast path information like + * - next_edge + * - next_enabled_feature + */ +static void +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t list_index) +{ + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; + uint32_t fi = UINT32_MAX, di = UINT32_MAX, prev_fi = UINT32_MAX; + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; + rte_graph_feature_list_t *flist = NULL; + rte_edge_t edge = UINT16_MAX; + uint64_t bitmask = 0; + + flist = arc->feature_list[list_index]; + + for (di = 0; di < arc->max_indexes; di++) { + bitmask = arc->feature_bit_mask_by_index[di]; + prev_fi = RTE_GRAPH_FEATURE_INVALID; + /* for each feature set for index, set fast path data */ + while (rte_bsf64_safe(bitmask, &fi)) { + gf = __rte_graph_feature_get(arc, fi, list_index); + gfd = rte_graph_feature_data_get(arc, gf, di); + RTE_VERIFY(!feature_arc_node_info_lookup(arc, fi, &finfo, 1)); + + /* If previous feature_index was valid in last loop */ + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { + prev_gf = __rte_graph_feature_get(arc, prev_fi, list_index); + prev_gfd = rte_graph_feature_data_get(arc, prev_gf, di); + /* + * Get edge of previous feature node connecting + * to this feature node + */ + RTE_VERIFY(!feature_arc_node_info_lookup(arc, prev_fi, + &prev_finfo, 1)); + if (!get_existing_edge(arc->feature_arc_name, + prev_finfo->feature_node, + finfo->feature_node, &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (%u->%u)%s[%u] = %s", + arc->feature_arc_name, list_index, di, + prev_gfd->user_data, prev_fi, fi, + prev_finfo->feature_node->name, + edge, finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* + * Fill current feature as next enabled + * feature to previous one + */ + prev_gfd->next_enabled_feature = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* On first feature edge of the node to be added */ + if (fi == rte_bsf64(arc->feature_bit_mask_by_index[di])) { + if (!get_existing_edge(arc->feature_arc_name, arc->start_node, + finfo->feature_node, + &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (->%u)%s[%u]=%s", + arc->feature_arc_name, list_index, di, + gfd->user_data, fi, + arc->start_node->name, edge, + finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* Set first feature set array for index*/ + flist->first_enabled_feature_by_index[di] = + (rte_graph_feature_t)fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* Clear current feature index */ + bitmask &= ~RTE_BIT64(fi); + } + } +} + +int +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const + char *feature_name, int32_t user_data) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_rt_list_t passive_list; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Enabling feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (!arc->runtime_enabled_features) + prepare_feature_arc_before_first_enable(arc); + + if (rte_graph_feature_validate(_arc, index, feature_name, 1, true)) + return -1; + + /** This should not fail as validate() has passed */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) + RTE_VERIFY(0); + + passive_list = ARC_PASSIVE_LIST(arc); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); + + /* Set current user-data */ + gfd->user_data = user_data; + + /* Set bitmask in control path bitmask */ + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + refill_feature_fastpath_data(arc, passive_list); + + /* If first time feature getting enabled */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[arc->active_feature_list], + rte_memory_order_relaxed); + + /* On very first feature enable instance */ + if (!finfo->ref_count) + bitmask |= RTE_BIT64(slot); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], + bitmask, rte_memory_order_relaxed); + + /* Slow path updates */ + arc->runtime_enabled_features++; + + /* Increase feature node info reference count */ + finfo->ref_count++; + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After enable, switched active feature list to %u", + arc->feature_arc_name, feature_name, arc->active_feature_list); + + return 0; +} + +int +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + rte_graph_feature_rt_list_t passive_list; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Disable feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (rte_graph_feature_validate(_arc, index, feature_name, 0, true)) + return -1; + + if (feature_lookup(arc, feature_name, &finfo, &slot)) + return -1; + + passive_list = ARC_PASSIVE_LIST(arc); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); + + /* Reset current user-data */ + gfd->user_data = ~0; + + refill_feature_fastpath_data(arc, passive_list); + + finfo->ref_count--; + arc->runtime_enabled_features--; + + /* If no feature enabled, reset feature in u64 fast path bitmask */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[arc->active_feature_list], + rte_memory_order_relaxed); + + /* When last feature is disabled */ + if (!finfo->ref_count) + bitmask &= ~(RTE_BIT64(slot)); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], bitmask, + rte_memory_order_relaxed); + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After disable, switched active feature list to %u", + arc->feature_arc_name, feature_name, arc->active_feature_list); + + return 0; +} + +int +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_node_list *node_info = NULL; + + while (!STAILQ_EMPTY(&arc->all_features)) { + node_info = STAILQ_FIRST(&arc->all_features); + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); + rte_free(node_info); + } + feature_arc_list_destroy(arc, 0); + feature_arc_list_destroy(arc, 1); + + dm->feature_arcs[arc->feature_arc_index] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + rte_free(arc); + + do_sanity_all_arcs(); + + return 0; +} + +int +rte_graph_feature_arc_cleanup(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm->feature_arcs[iter]); + } + rte_free(dm); + + __rte_graph_feature_arc_main = NULL; + + return 0; +} + +int +rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_arc *arc = NULL; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + if (_arc) + *_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); + + if ((strstr(arc->feature_arc_name, arc_name)) && + (strlen(arc->feature_arc_name) == strlen(arc_name))) { + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + return 0; + } + } + + return -1; +} + +uint32_t +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + return arc->runtime_enabled_features; +} + +uint32_t +rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t count = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) + count++; + + return count; +} + +char * +rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node->name; + + return NULL; +} + +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node; + + return NULL; + +} diff --git a/lib/graph/meson.build b/lib/graph/meson.build index 0cb15442ab..d916176fb7 100644 --- a/lib/graph/meson.build +++ b/lib/graph/meson.build @@ -14,11 +14,13 @@ sources = files( 'graph_debug.c', 'graph_stats.c', 'graph_populate.c', + 'graph_feature_arc.c', 'graph_pcap.c', 'rte_graph_worker.c', 'rte_graph_model_mcore_dispatch.c', ) headers = files('rte_graph.h', 'rte_graph_worker.h') +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') indirect_headers += files( 'rte_graph_model_mcore_dispatch.h', 'rte_graph_model_rtc.h', diff --git a/lib/graph/rte_graph_feature_arc.h b/lib/graph/rte_graph_feature_arc.h new file mode 100644 index 0000000000..d1c52bb3fb --- /dev/null +++ b/lib/graph/rte_graph_feature_arc.h @@ -0,0 +1,429 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ +#define _RTE_GRAPH_FEATURE_ARC_H_ + +#include <assert.h> +#include <errno.h> +#include <signal.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <rte_common.h> +#include <rte_compat.h> +#include <rte_debug.h> +#include <rte_graph.h> +#include <rte_graph_worker.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @file + * + * rte_graph_feature_arc.h + * + * Define APIs and structures/variables with respect to feature arc + * + * - Feature arc(s) + * - Feature(s) + * + * A feature arc represents an ordered list of features/protocol-nodes at a + * given networking layer. Feature arc provides a high level abstraction to + * connect various *rte_graph* nodes, designated as *feature nodes*, and + * allowing steering of packets across these feature nodes fast path processing + * in a generic manner. In a typical network stack, often a protocol or feature + * must be first enabled on a given interface, before any packet is steered + * towards it for feature processing. For eg: incoming IPv4 packets are sent to + * routing sub-system only after a valid IPv4 address is assigned to the + * received interface. In other words, often packets needs to be steered across + * features not based on the packet content but based on whether a feature is + * enable or disable on a given incoming/outgoing interface. Feature arc + * provides mechanism to enable/disable feature(s) on each interface at runtime + * and allow seamless packet steering across runtime enabled feature nodes in + * fast path. + * + * Feature arc also provides a way to steer packets from standard nodes to + * custom/user-defined *feature nodes* without any change in standard node's + * fast path functions + * + * On a given interface multiple feature(s) might be enabled in a particular + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" + * features may be enabled on "eth0" interface in "L3-output" feature arc. + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" + * interface in same "L3-output" feature arc. + * + * When multiple features are present in a given feature arc, its imperative + * to allow each feature processing in a particular sequential order. For + * instance, in "L3-input" feature arc it may be required to run "IPsec + * input" feature first, for packet decryption, before "ip-lookup". So a + * sequential order must be maintained among features present in a feature arc. + * + * Features are enabled/disabled multiple times at runtime to some or all + * available interfaces present in the system. Enable/disabling features on one + * interface is independent of other interface. + * + * A given feature might consume packet (if it's configured to consume) or may + * forward it to next enabled feature. For instance, "IPsec input" feature may + * consume/drop all packets with "Protect" policy action while all packets with + * policy action as "Bypass" may be forwarded to next enabled feature (with in + * same feature arc) + * + * This library facilitates rte graph based applications to steer packets in + * fast path to different feature nodes with-in a feature arc and support all + * functionalities described above + * + * In order to use feature-arc APIs, applications needs to do following in + * control path: + * - Initialize feature arc library via rte_graph_feature_arc_init() + * - Create feature arc via rte_graph_feature_arc_create() + * - *Before calling rte_graph_create()*, features must be added to feature-arc + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding + * features in a sequential order with "runs_after" and "runs_before" + * constraints. + * - Post rte_graph_create(), features can be enabled/disabled at runtime on + * any interface via rte_graph_feature_enable()/rte_graph_feature_disable() + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() + * + * In fast path, APIs are provided to steer packets towards feature path from + * - start_node (provided as an argument to rte_graph_feature_arc_create()) + * - feature nodes (which are added via rte_graph_feature_add()) + * + * For typical steering of packets across feature nodes, application required + * to know "rte_edges" which are saved in feature data object. Feature data + * object is unique for every interface per feature with in a feature arc. + * + * When steering packets from start_node to feature node: + * - rte_graph_feature_arc_first_feature_get() provides first enabled feature. + * - Next rte_edge from start_node to first enabled feature can be obtained via + * rte_graph_feature_arc_feature_set() + * + * rte_mbuf can carry [current feature, index] from start_node of an arc to other + * feature nodes + * + * In feature node, application can get 32-bit user_data + * via_rte_graph_feature_user_data_get() which is provided in + * rte_graph_feature_enable(). User data can hold feature specific cookie like + * IPsec policy database index (if more than one are supported) + * + * If feature node is not consuming packet, next enabled feature and next + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() + * + * It is application responsibility to ensure that at-least *last feature*(or sink + * feature) must be enabled from where packet can exit feature-arc path, if + * *NO* intermediate feature is consuming the packet and it has reached till + * the end of feature arc path + * + * It is recommended that all features *MUST* be added to feature arc before calling + * `rte_graph_create()`. Addition of features after `rte_graph_create()` may + * not work functionally. Although,rte_graph_feature_enable()/rte_graph_feature_disable() + * should be called after `rte_graph_create()` in control plane. + * + * Synchronization among cores + * --------------------------- + * Subsequent calls to rte_graph_feature_enable() is allowed while worker cores + * are processing in rte_graph_walk() loop. However, for + * rte_graph_feature_disable() application must use RCU based synchronization + */ + +/** Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER ((rte_graph_feature_arc_t)UINT64_MAX) + +/** Max number of feature arcs which can be created */ +#define RTE_GRAPH_FEATURE_ARC_MAX 64 + +/** Max number of features supported in a given feature arc */ +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 + +/** Length of feature arc name */ +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE + +/** @internal */ +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) + +/**< Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_INVALID rte_graph_feature_cast(UINT8_MAX) + +/** rte_graph feature arc object */ +typedef uint64_t rte_graph_feature_arc_t; + +/** rte_graph feature object */ +typedef uint8_t rte_graph_feature_t; + +/** runtime active feature list index with in feature arc*/ +typedef uint16_t rte_graph_feature_rt_list_t; + +/** per feature arc monotonically increasing counter to synchronize fast path APIs */ +typedef uint16_t rte_graph_feature_counter_t; + +/** + * Initialize feature arc subsystem + * + * @param max_feature_arcs + * Maximum number of feature arcs required to be supported + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_init(int max_feature_arcs); + +/** + * Create a feature arc + * + * @param feature_arc_name + * Feature arc name with max length of @ref RTE_GRAPH_FEATURE_ARC_NAMELEN + * @param max_features + * Maximum number of features to be supported in this feature arc + * @param max_indexes + * Maximum number of interfaces/ports/indexes to be supported + * @param start_node + * Base node where this feature arc's features are checked in fast path + * @param[out] _arc + * Feature arc object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, + rte_graph_feature_arc_t *_arc); + +/** + * Get feature arc object with name + * + * @param arc_name + * Feature arc name provided to successful @ref rte_graph_feature_arc_create + * @param[out] _arc + * Feature arc object returned. Valid only when API returns SUCCESS + * + * @return + * 0: Success + * <0: Failure. + */ +__rte_experimental +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc); + +/** + * Add a feature to already created feature arc. For instance + * + * 1. Add first feature node: "ipv4-input" to input arc + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); + * + * 2. Add "ipsec-input" feature node after "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", NULL); + * + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", NULL, "ipv4-input"); + * + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4-input", "ipsec-input"); + * + * @param _arc + * Feature arc handle returned from @ref rte_graph_feature_arc_create() + * @param feature_node + * Graph node representing feature. On success, feature_node is next_node of + * feature_arc->start_node + * @param runs_after + * Add this feature_node after already added "runs_after". Creates + * start_node -> runs_after -> this_feature sequence + * @param runs_before + * Add this feature_node before already added "runs_before". Creates + * start_node -> this_feature -> runs_before sequence + * + * <I> Must be called before rte_graph_create() </I> + * <I> rte_graph_feature_add() is not allowed after call to + * rte_graph_feature_enable() so all features must be added before they can be + * enabled </I> + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *runs_after, const char *runs_before); + +/** + * Enable feature within a feature arc + * + * Must be called after @b rte_graph_create(). + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param user_data + * Application specific data which is retrieved in fast path + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int32_t user_data); + +/** + * Validate whether subsequent enable/disable feature would succeed or not. + * API is thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param is_enable_disable + * If 1, validate whether subsequent @ref rte_graph_feature_enable would pass or not + * If 0, validate whether subsequent @ref rte_graph_feature_disable would pass or not + * @param emit_logs + * If passed true, emit error logs when failure is returned + * If passed false, do not emit error logs when failure is returned + * + * @return + * 0: Subsequent enable/disable API would pass + * <0: Subsequent enable/disable API would not pass + */ +__rte_experimental +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, int is_enable_disable, bool emit_logs); + +/** + * Disable already enabled feature within a feature arc + * + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name); + +/** + * Get rte_graph_feature_t object from feature name + * + * @param arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param feature_name + * Feature name provided to @ref rte_graph_feature_add + * @param[out] feature + * Feature object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_lookup(rte_graph_feature_arc_t arc, const char *feature_name, + rte_graph_feature_t *feature); + +/** + * Delete feature_arc object + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); + +/** + * Cleanup all feature arcs + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_cleanup(void); + +/** + * Slow path API to know how many features are added (NOT enabled) within a + * feature arc + * + * @param _arc + * Feature arc object + * + * @return: Number of added features to arc + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to know how many features are currently enabled within a + * feature arc across all indexes. If a single feature is enabled on all interfaces, + * this API would return "number_of_interfaces" as count (but not "1") + * + * @param _arc + * Feature arc object + * + * @return: Number of enabled features across all indexes + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to get feature node name from rte_graph_feature_t object + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: Name of the feature node + */ +__rte_experimental +char *rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + +/** + * Slow path API to get corresponding struct rte_node_register * from + * rte_graph_feature_t + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: struct rte_node_register * of feature node on SUCCESS else NULL + */ +__rte_experimental +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/lib/graph/rte_graph_feature_arc_worker.h b/lib/graph/rte_graph_feature_arc_worker.h new file mode 100644 index 0000000000..5c76cb1151 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc_worker.h @@ -0,0 +1,672 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ + +#include <stddef.h> +#include <rte_graph_feature_arc.h> +#include <rte_bitops.h> + +/** + * @file + * + * rte_graph_feature_arc_worker.h + * + * Defines fast path structure + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/** @internal + * + * Slow path feature node info list + */ +struct rte_graph_feature_node_list { + /** Next feature */ + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; + + /** node representing feature */ + struct rte_node_register *feature_node; + + /** How many indexes/interfaces using this feature */ + int32_t ref_count; + + /* node_index in list (after feature_enable())*/ + uint32_t node_index; + + /** Back pointer to feature arc */ + void *feature_arc; + + /** rte_edge_t to this feature node from feature_arc->start_node */ + rte_edge_t edge_to_this_feature; +}; + +/** + * Feature data object: + * + * Feature data stores information to steer packets for: + * - a feature with in feature arc + * - Index i.e. Port/Interface index + * + * Each feature data object holds + * - User data of current feature retrieved via rte_graph_feature_user_data_get() + * - next_edge is used in two conditions when packet to be steered from + * -- start_node to first enabled feature on an interface index + * -- current feature node to next enabled feature on an interface index + * - next_enabled_feature on interface index, if current feature is not + * consuming packet + * + * While user_data corresponds to current enabled feature node, while + * next_edge and next_enabled_feature corresponds to be next enabled feature + * node on an interface index + * + * First enabled feature on interface index can be retrieved via: + * - rte_graph_feature_first_feature_get() if arc's start_node is trying to + * steer packet from start_node to first enabled feature on interface index + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_next_feature_get() if current node is not arc's + * start_node. Input to rte_graph_feature_next_feature_get() is current + * enabled feature and interface index + */ +typedef struct __rte_packed rte_graph_feature_data { + /** edge from current node to next enabled feature */ + rte_edge_t next_edge; + + union { + uint16_t reserved; + struct { + /** next enabled feature on index from current feature */ + rte_graph_feature_t next_enabled_feature; + }; + }; + + /** user_data set by application in rte_graph_feature_enable() for + * - current feature + * - interface index + */ + int32_t user_data; +} rte_graph_feature_data_t; + +/** + * Feature object + * + * Feature object holds feature data object for every index/interface within + * feature + * + * Within a given arc and interface index, first feature object can be + * retrieved in arc's start_node via: + * - rte_graph_feature_arc_first_feature_get() + * + * Feature data information can be retrieved for first feature in start node via + * - rte_graph_feature_arc_feature_set() + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_arc_next_feature_get() + * + * Typically application stores rte_graph_feature_t object in rte_mbuf. + * rte_graph_feature_t can be translated to (struct rte_graph_feature *) via + * rte_graph_feature_get() in fast path. Further if needed, feature data for an + * index within a feature can be retrieved via rte_graph_feature_data_get() + */ +struct __rte_cache_aligned rte_graph_feature { + /** feature index or rte_graph_feature_t */ + uint16_t this_feature_index; + + /* + * Array of size arc->feature_data_size + * + * <----------------- Feature --------------------------> + * [data-index-0][data-index-1]...[data-index-max_index-1] + * + * sizeof(feature_data_by_index[0] == sizeof(rte_graph_feature_data_t) + * + */ + uint8_t feature_data_by_index[]; +}; + +/** + * Feature list object + * + * Feature list is required to decouple fast path APIs with control path APIs. + * + * There are two feature lists: active, passive + * + * While fast path APIs always work on active list, control plane updates + * passive lists and atomically switch passive list to active list to make it + * available to fast path APIs + * + * Each feature node in start of it's fast path function, must grab active list from + * arc via + * - rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * + * Retrieved list must be provided to other feature arc fast path APIs so that + * any control plane changes of active list should not impact current node + * execution iteration. Active list change would be reflected to current node + * in next iteration + * + * With active/passive list mechanism and integrating RCU APIs in graph worker + * loop, application can update features at runtime without stopping fast path + * cores. A RCU synchronization is required when a feature needs to be disabled via + * rte_graph_feature_disable(). On enabling a feature, RCU synchronization may + * not be required + * + */ +typedef struct __rte_cache_aligned rte_graph_feature_list { + /** + * fast path array holding per_feature data. + * Duplicate entry as feature-arc also hold this pointer + * arc->features[] + * + *<-------------feature-0 ---------><---------feature-1 -------------->... + *[index-0][index-1]...[max_index-1]<-ALIGN->[index-0][index-1] ...[max_index-1]... + */ + struct rte_graph_feature *indexed_by_features; + /* + * fast path array holding first enabled feature per index + * (Required in start_node. In non start_node, mbuf can hold next enabled + * feature) + */ + rte_graph_feature_t first_enabled_feature_by_index[]; +} rte_graph_feature_list_t; + +/** + * rte_graph Feature arc object + * + * Feature arc object holds control plane and fast path information for all + * features and all interface index information for steering packets across + * feature nodes + * + * Within a feature arc, only RTE_GRAPH_FEATURE_MAX_PER_ARC features can be + * added. If more features needs to be added, another feature arc can be + * created + * + * Application gets rte_graph_feature_arc_t object via + * - rte_graph_feature_arc_create() OR + * - rte_graph_feature_arc_lookup_by_name() + * + * In fast path, rte_graph_feature_arc_t can be translated to (struct + * rte_graph_feature_arc *) via rte_graph_feature_arc_get(). Later is needed to + * add as an input argument to all fast path feature arc APIs + */ +struct __rte_cache_aligned rte_graph_feature_arc { + /* First 64B is fast path variables */ + RTE_MARKER fast_path_variables; + + /** runtime active feature list */ + rte_graph_feature_rt_list_t active_feature_list; + + /** Actual Size of feature_list object */ + uint16_t feature_list_size; + + /** + * Size each feature in fastpath. + * Required to navigate from feature to another feature in fast path + */ + uint16_t feature_size; + + /** + * Size of all feature data for an index + * Required to navigate through various feature data within a feature + * in fast path + */ + uint16_t feature_data_size; + + /** + * Quick fast path bitmask indicating if any feature enabled or not on + * any of the indexes. Helps in optimally process packets for the case + * when features are added but not enabled + * + * Separate for active and passive list + */ + uint64_t feature_enable_bitmask[2]; + + /** + * Pointer to both active and passive feature list object + */ + rte_graph_feature_list_t *feature_list[2]; + + /** + * Feature objects for each list + */ + struct rte_graph_feature *features[2]; + + /** index in feature_arc_main */ + uint16_t feature_arc_index; + + uint16_t reserved[3]; + + /** Slow path variables follows*/ + RTE_MARKER slow_path_variables; + + /** feature arc name */ + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** All feature lists */ + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; + + /** control plane counter to track enabled features */ + uint32_t runtime_enabled_features; + + /** Back pointer to feature_arc_main */ + void *feature_arc_main; + + /** Arc's start/base node */ + struct rte_node_register *start_node; + + /** maximum number of features supported by this arc */ + uint32_t max_features; + + /** maximum number of index supported by this arc */ + uint32_t max_indexes; + + /** Slow path bit mask per feature per index */ + uint64_t feature_bit_mask_by_index[]; +}; + +/** + * Feature arc main object + * + * Holds all feature arcs created by application + * + * RTE_GRAPH_FEATURE_ARC_MAX number of feature arcs can be created by + * application via rte_graph_feature_arc_create() + */ +typedef struct feature_arc_main { + /** number of feature arcs created by application */ + uint32_t num_feature_arcs; + + /** max features arcs allowed */ + uint32_t max_feature_arcs; + + /** feature arcs */ + rte_graph_feature_arc_t feature_arcs[]; +} rte_graph_feature_arc_main_t; + +/** @internal Get feature arc pointer from object */ +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc *)arc) + +extern rte_graph_feature_arc_main_t *__feature_arc_main; + +/** + * API to know if feature is valid or not + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_is_valid(rte_graph_feature_t feature) +{ + return (feature != RTE_GRAPH_FEATURE_INVALID); +} + +/** + * Get rte_graph_feature object with no checks + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * @param feature_list + * active feature list retrieved from rte_graph_feature_arc_has_any_feature() + * or rte_graph_feature_arc_has_feature() + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature, + const rte_graph_feature_rt_list_t feature_list) +{ + return ((struct rte_graph_feature *)(((uint8_t *)arc->features[feature_list]) + + (feature * arc->feature_size))); +} + +/** + * Get rte_graph_feature object for a given interface/index from feature arc + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature) +{ + if (unlikely(feature >= arc->max_features)) + RTE_VERIFY(0); + + if (likely(rte_graph_feature_is_valid(feature))) + return __rte_graph_feature_get(arc, feature, arc->active_feature_list); + + return NULL; +} + +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + RTE_SET_USED(arc); + return ((rte_graph_feature_data_t *)(((uint8_t *)feature->feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Get rte_graph feature data object for a index in feature + * + * @param arc + * feature arc + * @param feature + * Pointer to feature object + * @param index + * Index of feature maintained in slow path linked list + * + * @return + * Valid feature data + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + if (likely(index < arc->max_indexes)) + return __rte_graph_feature_data_get(arc, feature, index); + + RTE_VERIFY(0); +} + +/** + * Fast path API to check if any feature enabled on a feature arc + * Typically from arc->start_node process function + * + * @param arc + * Feature arc object + * @param[out] plist + * Pointer to runtime active feature list which needs to be provided to other + * fast path APIs + * + * @return + * 0: If no feature enabled + * Non-Zero: Bitmask of features enabled. plist is valid + * + */ +__rte_experimental +static __rte_always_inline uint64_t +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_rt_list_t *plist) +{ + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Fast path API to check if provided feature is enabled on any interface/index + * or not + * + * @param arc + * Feature arc object + * @param feature + * Input rte_graph_feature_t that needs to be checked + * @param[out] plist + * Returns active list to caller which needs to be provided to other fast path + * APIs + * + * @return + * 1: If input [feature] is enabled in arc + * 0: If input [feature] is not enabled in arc + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_t feature, + rte_graph_feature_rt_list_t *plist) +{ + uint64_t bitmask = RTE_BIT64(feature); + + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (bitmask & rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Prefetch feature arc fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) +{ + rte_prefetch0((void *)&arc->fast_path_variables); +} + +/** + * Prefetch feature related fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature) +{ + /* feature cache line */ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, list)); +} + +/** + * Prefetch feature data upfront. Perform sanity + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object returned from @ref + * rte_graph_feature_arc_first_feature_get() + * @param index + * Interface/index + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, uint32_t index) +{ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)((uint8_t *)arc->features[list] + + offsetof(struct rte_graph_feature, feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Fast path API to get first enabled feature on interface index + * Typically required in arc->start_node so that from returned feature, + * feature-data can be retrieved to steer packets + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. + * + * @return + * 1. Success. If first feature field is enabled and returned [feature] is valid + * 0. Failure. If first feature field is disabled in arc + * + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + + *feature = feature_list->first_enabled_feature_by_index[index]; + + return rte_graph_feature_is_valid(*feature); +} + +/** + * Fast path API to get next enabled feature on interface index with provided + * input feature + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. API sets next enabled feature on [index] + * from provided input feature. Valid only if API returns Success + * @param[out] next_edge + * Edge from current feature to next feature. Valid only if next feature is valid + * + * @return + * 1. Success. first feature field is enabled/valid + * 0. Failure. first feature field is disabled/invalid + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature, + rte_edge_t *next_edge) +{ + rte_graph_feature_data_t *feature_data = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(*feature))) { + f = __rte_graph_feature_get(arc, *feature, list); + feature_data = rte_graph_feature_data_get(arc, f, index); + *feature = feature_data->next_enabled_feature; + *next_edge = feature_data->next_edge; + return rte_graph_feature_is_valid(*feature); + } + + return 0; +} + +/** + * Set fields with respect to first enabled feature in an arc and return edge + * Typically returned feature and interface index must be saved in rte_mbuf + * structure to pass this information to next feature node + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param index + * Index (of interface) + * @param[out] gf + * Pointer to rte_graph_feature_t. Valid if API returns Success + * @param[out] edge + * Edge to steer packet from arc->start_node to first enabled feature. Valid + * only if API returns Success + * + * @return + * 0: If valid feature is enabled and set by API in *gf + * 1: If valid feature is NOT enabled + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_t +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *gf, + rte_edge_t *edge) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + struct rte_graph_feature_data *feature_data = NULL; + struct rte_graph_feature *feature = NULL; + rte_graph_feature_t f; + + f = feature_list->first_enabled_feature_by_index[index]; + + if (unlikely(rte_graph_feature_is_valid(f))) { + feature = __rte_graph_feature_get(arc, f, list); + feature_data = rte_graph_feature_data_get(arc, feature, index); + *gf = f; + *edge = feature_data->next_edge; + return 0; + } + + return 1; +} + +__rte_experimental +static __rte_always_inline int32_t +__rte_graph_feature_user_data_get(rte_graph_feature_data_t *fdata) +{ + return fdata->user_data; +} + +/** + * Get user data corresponding to current feature set by application in + * rte_graph_feature_enable() + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Feature index + * @param index + * Interface index + * + * @return + * -1: Failure + * Valid user data: Success + */ +__rte_experimental +static __rte_always_inline int32_t +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, + uint32_t index) +{ + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(feature))) { + f = __rte_graph_feature_get(arc, feature, list); + fdata = rte_graph_feature_data_get(arc, f, index); + return __rte_graph_feature_user_data_get(fdata); + } + + return -1; +} +#ifdef __cplusplus +} +#endif +#endif diff --git a/lib/graph/version.map b/lib/graph/version.map index 2c83425ddc..3b7f475afd 100644 --- a/lib/graph/version.map +++ b/lib/graph/version.map @@ -52,3 +52,23 @@ DPDK_25 { local: *; }; + +EXPERIMENTAL { + global: + + # added in 24.11 + rte_graph_feature_arc_init; + rte_graph_feature_arc_create; + rte_graph_feature_arc_lookup_by_name; + rte_graph_feature_add; + rte_graph_feature_enable; + rte_graph_feature_validate; + rte_graph_feature_disable; + rte_graph_feature_lookup; + rte_graph_feature_arc_destroy; + rte_graph_feature_arc_cleanup; + rte_graph_feature_arc_num_enabled_features; + rte_graph_feature_arc_num_features; + rte_graph_feature_arc_feature_to_name; + rte_graph_feature_arc_feature_to_node; +}; -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH v2 2/5] graph: add feature arc option in graph create 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 1/5] graph: add feature arc support Nitin Saxena @ 2024-10-08 13:30 ` Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 3/5] graph: add IPv4 output feature arc Nitin Saxena ` (3 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena, Pavan Nikhilesh Added option in graph create to call feature-specific process node functions. This removes extra overhead for checking feature arc status in nodes where application is not using feature arc processing Signed-off-by: Pavan Nikhilesh <pbhagavatula@marvell.com> Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/graph/graph.c | 1 + lib/graph/graph_populate.c | 7 ++++++- lib/graph/graph_private.h | 3 +++ lib/graph/node.c | 2 ++ lib/graph/rte_graph.h | 3 +++ 5 files changed, 15 insertions(+), 1 deletion(-) diff --git a/lib/graph/graph.c b/lib/graph/graph.c index d5b8c9f918..b0ad3a83ae 100644 --- a/lib/graph/graph.c +++ b/lib/graph/graph.c @@ -455,6 +455,7 @@ rte_graph_create(const char *name, struct rte_graph_param *prm) graph->parent_id = RTE_GRAPH_ID_INVALID; graph->lcore_id = RTE_MAX_LCORE; graph->num_pkt_to_capture = prm->num_pkt_to_capture; + graph->feature_arc_enabled = prm->feature_arc_enable; if (prm->pcap_filename) rte_strscpy(graph->pcap_filename, prm->pcap_filename, RTE_GRAPH_PCAP_FILE_SZ); diff --git a/lib/graph/graph_populate.c b/lib/graph/graph_populate.c index ed596a7711..5d8aa7b903 100644 --- a/lib/graph/graph_populate.c +++ b/lib/graph/graph_populate.c @@ -79,8 +79,13 @@ graph_nodes_populate(struct graph *_graph) if (graph_pcap_is_enable()) { node->process = graph_pcap_dispatch; node->original_process = graph_node->node->process; - } else + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->original_process = graph_node->node->feat_arc_proc; + } else { node->process = graph_node->node->process; + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->process = graph_node->node->feat_arc_proc; + } memcpy(node->name, graph_node->node->name, RTE_GRAPH_NAMESIZE); pid = graph_node->node->parent_id; if (pid != RTE_NODE_ID_INVALID) { /* Cloned node */ diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index d557d55f2d..58ba0abeff 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -56,6 +56,7 @@ struct node { unsigned int lcore_id; /**< Node runs on the Lcore ID used for mcore dispatch model. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Allocated identifier for the node. */ @@ -126,6 +127,8 @@ struct graph { /**< Number of packets to be captured per core. */ char pcap_filename[RTE_GRAPH_PCAP_FILE_SZ]; /**< pcap file name/path. */ + uint8_t feature_arc_enabled; + /**< Graph feature arc. */ STAILQ_HEAD(gnode_list, graph_node) node_list; /**< Nodes in a graph. */ }; diff --git a/lib/graph/node.c b/lib/graph/node.c index 99a9622779..d8fd273543 100644 --- a/lib/graph/node.c +++ b/lib/graph/node.c @@ -90,6 +90,7 @@ __rte_node_register(const struct rte_node_register *reg) goto free; node->flags = reg->flags; node->process = reg->process; + node->feat_arc_proc = reg->feat_arc_proc; node->init = reg->init; node->fini = reg->fini; node->nb_edges = reg->nb_edges; @@ -137,6 +138,7 @@ node_clone(struct node *node, const char *name) /* Clone the source node */ reg->flags = node->flags; reg->process = node->process; + reg->feat_arc_proc = node->feat_arc_proc; reg->init = node->init; reg->fini = node->fini; reg->nb_edges = node->nb_edges; diff --git a/lib/graph/rte_graph.h b/lib/graph/rte_graph.h index ecfec2068a..ebbdbbea48 100644 --- a/lib/graph/rte_graph.h +++ b/lib/graph/rte_graph.h @@ -163,6 +163,8 @@ struct rte_graph_param { uint64_t num_pkt_to_capture; /**< Number of packets to capture. */ char *pcap_filename; /**< Filename in which packets to be captured.*/ + bool feature_arc_enable; /**< Enable Graph feature arc. */ + union { struct { uint64_t rsvd; /**< Reserved for rtc model. */ @@ -470,6 +472,7 @@ struct rte_node_register { uint64_t flags; /**< Node configuration flag. */ #define RTE_NODE_SOURCE_F (1ULL << 0) /**< Node type is source. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Node Identifier. */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH v2 3/5] graph: add IPv4 output feature arc 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 1/5] graph: add feature arc support Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 2/5] graph: add feature arc option in graph create Nitin Saxena @ 2024-10-08 13:30 ` Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 4/5] test/graph_feature_arc: add functional tests Nitin Saxena ` (2 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena add ipv4-output feature arc in ipv4-rewrite node to allow custom/standard nodes(like outbound IPsec policy node) in outgoing forwarding path Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/node/ip4_rewrite.c | 476 +++++++++++++++++++++++++++++------- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 4 files changed, 417 insertions(+), 97 deletions(-) diff --git a/lib/node/ip4_rewrite.c b/lib/node/ip4_rewrite.c index 34a920df5e..5a160335f2 100644 --- a/lib/node/ip4_rewrite.c +++ b/lib/node/ip4_rewrite.c @@ -15,39 +15,156 @@ #include "ip4_rewrite_priv.h" #include "node_private.h" +#define ALL_PKT_MASK 0xf + struct ip4_rewrite_node_ctx { + rte_graph_feature_arc_t output_feature_arc; /* Dynamic offset to mbuf priv1 */ int mbuf_priv1_off; /* Cached next index */ uint16_t next_index; + uint16_t last_tx; }; +typedef struct rewrite_priv_vars { + union { + struct { + rte_xmm_t xmm1; + }; + struct __rte_packed { + uint16_t next0; + uint16_t next1; + uint16_t next2; + uint16_t next3; + uint16_t last_tx_interface; + uint16_t last_if_feature; + uint16_t actual_feat_mask; + uint16_t speculative_feat_mask; + }; + }; +} rewrite_priv_vars_t; + static struct ip4_rewrite_node_main *ip4_rewrite_nm; #define IP4_REWRITE_NODE_LAST_NEXT(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->next_index) +#define IP4_REWRITE_NODE_LAST_TX(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->last_tx) + #define IP4_REWRITE_NODE_PRIV1_OFF(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->mbuf_priv1_off) -static uint16_t -ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, - void **objs, uint16_t nb_objs) +#define IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->output_feature_arc) + +static __rte_always_inline void +prefetch_mbuf_and_dynfield(struct rte_mbuf *mbuf) { + /* prefetch first cache line required for accessing buf_addr */ + rte_prefetch0((void *)mbuf); +} + +static __rte_always_inline void +check_output_feature_x4(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t flist, + rewrite_priv_vars_t *pvar, struct node_mbuf_priv1 *priv0, + struct node_mbuf_priv1 *priv1, struct node_mbuf_priv1 *priv2, + struct node_mbuf_priv1 *priv3) +{ + uint32_t mask = 0; + uint16_t xor = 0; + + /* + * interface edge's start from 1 and not from 0 as "pkt_drop" + * is next node at 0th index + */ + priv0->if_index = pvar->next0 - 1; + priv1->if_index = pvar->next1 - 1; + priv2->if_index = pvar->next2 - 1; + priv3->if_index = pvar->next3 - 1; + + /* Find out if all packets are sent to last_tx_interface */ + xor = pvar->last_tx_interface ^ priv0->if_index; + xor += priv0->if_index ^ priv1->if_index; + xor += priv1->if_index ^ priv2->if_index; + xor += priv2->if_index ^ priv3->if_index; + + if (likely(!xor)) { + /* copy last interface feature and feature mask */ + priv0->current_feature = priv1->current_feature = + priv2->current_feature = priv3->current_feature = + pvar->last_if_feature; + pvar->actual_feat_mask = pvar->speculative_feat_mask; + } else { + /* create a mask for index which does not have feature + * Also override next edge and if feature enabled, get feature + */ + mask = rte_graph_feature_arc_feature_set(arc, flist, priv0->if_index, + &priv0->current_feature, + &pvar->next0); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv1->if_index, + &priv1->current_feature, + &pvar->next1)) << 1); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv2->if_index, + &priv2->current_feature, + &pvar->next2)) << 2); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv3->if_index, + &priv3->current_feature, + &pvar->next3)) << 3); + + /* + * add last tx and last feature regardless even if feature is + * valid or not + */ + pvar->last_tx_interface = priv3->if_index; + pvar->last_if_feature = priv3->current_feature; + /* Set 0xf if invalid feature to last packet, else 0 */ + pvar->speculative_feat_mask = (priv3->current_feature == + RTE_GRAPH_FEATURE_INVALID) ? ALL_PKT_MASK : 0x0; + pvar->actual_feat_mask = mask; + } +} + +static __rte_always_inline uint16_t +__ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs, + const int dyn, const int check_enabled_features, + struct rte_graph_feature_arc *out_feature_arc, + const rte_graph_feature_rt_list_t flist) +{ + struct node_mbuf_priv1 *priv0 = NULL, *priv1 = NULL, *priv2 = NULL, *priv3 = NULL; struct rte_mbuf *mbuf0, *mbuf1, *mbuf2, *mbuf3, **pkts; struct ip4_rewrite_nh_header *nh = ip4_rewrite_nm->nh; - const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); - uint16_t next0, next1, next2, next3, next_index; - struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; uint16_t n_left_from, held = 0, last_spec = 0; + struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; + rewrite_priv_vars_t pvar; + int64_t fd0, fd1, fd2, fd3; + rte_edge_t fix_spec = 0; void *d0, *d1, *d2, *d3; void **to_next, **from; + uint16_t next_index; rte_xmm_t priv01; rte_xmm_t priv23; int i; - /* Speculative next as last next */ + RTE_SET_USED(fd0); + RTE_SET_USED(fd1); + RTE_SET_USED(fd2); + RTE_SET_USED(fd3); + + /* Initialize speculative variables.*/ + + /* Last interface */ + pvar.last_tx_interface = IP4_REWRITE_NODE_LAST_TX(node->ctx); + /*last next from node ctx*/ next_index = IP4_REWRITE_NODE_LAST_NEXT(node->ctx); + pvar.speculative_feat_mask = ALL_PKT_MASK; + pvar.actual_feat_mask = 0; + rte_prefetch0(nh); pkts = (struct rte_mbuf **)objs; @@ -55,20 +172,47 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, n_left_from = nb_objs; for (i = 0; i < 4 && i < n_left_from; i++) - rte_prefetch0(pkts[i]); + prefetch_mbuf_and_dynfield(pkts[i]); /* Get stream for the speculated next node */ to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + + /* prefetch speculative feature and corresponding data */ + if (check_enabled_features) { + /* + * Get first feature enabled, if any, on last_tx_interface + */ + if (unlikely(rte_graph_feature_arc_first_feature_get(out_feature_arc, + flist, + pvar.last_tx_interface, + (rte_graph_feature_t *) + &pvar.last_if_feature))) { + /* prefetch feature cache line */ + rte_graph_feature_arc_feature_prefetch(out_feature_arc, flist, + pvar.last_if_feature); + + /* prefetch feature data cache line */ + rte_graph_feature_arc_data_prefetch(out_feature_arc, flist, + pvar.last_if_feature, + pvar.last_tx_interface); + /* + * Set speculativa_feat mask to indicate, all 4 packets + * going to feature path + */ + pvar.speculative_feat_mask = 0; + } + } + /* Update Ethernet header of pkts */ while (n_left_from >= 4) { if (likely(n_left_from > 7)) { /* Prefetch only next-mbuf struct and priv area. * Data need not be prefetched as we only write. */ - rte_prefetch0(pkts[4]); - rte_prefetch0(pkts[5]); - rte_prefetch0(pkts[6]); - rte_prefetch0(pkts[7]); + prefetch_mbuf_and_dynfield(pkts[4]); + prefetch_mbuf_and_dynfield(pkts[5]); + prefetch_mbuf_and_dynfield(pkts[6]); + prefetch_mbuf_and_dynfield(pkts[7]); } mbuf0 = pkts[0]; @@ -78,66 +222,138 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 4; n_left_from -= 4; + + /* Copy mbuf private data into private variables */ priv01.u64[0] = node_mbuf_priv1(mbuf0, dyn)->u; priv01.u64[1] = node_mbuf_priv1(mbuf1, dyn)->u; priv23.u64[0] = node_mbuf_priv1(mbuf2, dyn)->u; priv23.u64[1] = node_mbuf_priv1(mbuf3, dyn)->u; - /* Increment checksum by one. */ - priv01.u32[1] += rte_cpu_to_be_16(0x0100); - priv01.u32[3] += rte_cpu_to_be_16(0x0100); - priv23.u32[1] += rte_cpu_to_be_16(0x0100); - priv23.u32[3] += rte_cpu_to_be_16(0x0100); - - /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, - nh[priv01.u16[0]].rewrite_len); - - next0 = nh[priv01.u16[0]].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - ip0->time_to_live = priv01.u16[1] - 1; - ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ - d1 = rte_pktmbuf_mtod(mbuf1, void *); - rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, - nh[priv01.u16[4]].rewrite_len); - - next1 = nh[priv01.u16[4]].tx_node; - ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + - sizeof(struct rte_ether_hdr)); - ip1->time_to_live = priv01.u16[5] - 1; - ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ - d2 = rte_pktmbuf_mtod(mbuf2, void *); - rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, - nh[priv23.u16[0]].rewrite_len); - next2 = nh[priv23.u16[0]].tx_node; - ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + - sizeof(struct rte_ether_hdr)); - ip2->time_to_live = priv23.u16[1] - 1; - ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ - d3 = rte_pktmbuf_mtod(mbuf3, void *); - rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, - nh[priv23.u16[4]].rewrite_len); - - next3 = nh[priv23.u16[4]].tx_node; - ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + - sizeof(struct rte_ether_hdr)); - ip3->time_to_live = priv23.u16[5] - 1; - ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + /* Copy next edge from next hop */ + pvar.next0 = nh[priv01.u16[0]].tx_node; + pvar.next1 = nh[priv01.u16[4]].tx_node; + pvar.next2 = nh[priv23.u16[0]].tx_node; + pvar.next3 = nh[priv23.u16[4]].tx_node; + + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + priv1 = node_mbuf_priv1(mbuf1, dyn); + priv2 = node_mbuf_priv1(mbuf2, dyn); + priv3 = node_mbuf_priv1(mbuf3, dyn); + + /* If feature is enabled, override next edge for each mbuf + * and set node_mbuf_priv data appropriately + */ + check_output_feature_x4(out_feature_arc, flist, + &pvar, priv0, priv1, priv2, priv3); + + /* check_output_feature_x4() returns bit mask which indicates + * which packet is not following feature path, hence normal processing + * has to happen on them + */ + if (unlikely(pvar.actual_feat_mask)) { + if (pvar.actual_feat_mask & 0x1) { + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + } + if (pvar.actual_feat_mask & 0x2) { + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + } + if (pvar.actual_feat_mask & 0x4) { + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + } + if (pvar.actual_feat_mask & 0x8) { + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } + } + } else { + /* Case when no feature is enabled */ + + /* Increment checksum by one. */ + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } /* Enqueue four to next node */ - rte_edge_t fix_spec = - ((next_index == next0) && (next0 == next1) && - (next1 == next2) && (next2 == next3)); + fix_spec = next_index ^ pvar.next0; + fix_spec += next_index ^ pvar.next1; + fix_spec += next_index ^ pvar.next2; + fix_spec += next_index ^ pvar.next3; - if (unlikely(fix_spec == 0)) { + if (unlikely(fix_spec != 0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -146,56 +362,56 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, last_spec = 0; /* next0 */ - if (next_index == next0) { + if (next_index == pvar.next0) { to_next[0] = from[0]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next0, + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); } /* next1 */ - if (next_index == next1) { + if (next_index == pvar.next1) { to_next[0] = from[1]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next1, + rte_node_enqueue_x1(graph, node, pvar.next1, from[1]); } /* next2 */ - if (next_index == next2) { + if (next_index == pvar.next2) { to_next[0] = from[2]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next2, + rte_node_enqueue_x1(graph, node, pvar.next2, from[2]); } /* next3 */ - if (next_index == next3) { + if (next_index == pvar.next3) { to_next[0] = from[3]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next3, + rte_node_enqueue_x1(graph, node, pvar.next3, from[3]); } from += 4; /* Change speculation if last two are same */ - if ((next_index != next3) && (next2 == next3)) { + if ((next_index != pvar.next3) && (pvar.next2 == pvar.next3)) { /* Put the current speculated node */ rte_node_next_stream_put(graph, node, next_index, held); held = 0; /* Get next speculated stream */ - next_index = next3; + next_index = pvar.next3; to_next = rte_node_next_stream_get( graph, node, next_index, nb_objs); } @@ -212,20 +428,41 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 1; n_left_from -= 1; - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, - nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); - - next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + - rte_cpu_to_be_16(0x0100); - chksum += chksum >= 0xffff; - ip0->hdr_checksum = chksum; - ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; - - if (unlikely(next_index ^ next0)) { + pvar.next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + if (pvar.next0 != (pvar.last_tx_interface + 1)) { + priv0->if_index = pvar.next0 - 1; + rte_graph_feature_arc_feature_set(out_feature_arc, flist, + priv0->if_index, + &priv0->current_feature, + &pvar.next0); + pvar.last_tx_interface = priv0->if_index; + pvar.last_if_feature = priv0->current_feature; + } else { + /* current mbuf index is same as last_tx_interface */ + priv0->if_index = pvar.last_tx_interface; + priv0->current_feature = pvar.last_if_feature; + } + } + /* Do the needful if either feature arc is disabled OR + * Invalid feature is present + */ + if (!check_enabled_features || + (priv0->current_feature == RTE_GRAPH_FEATURE_INVALID)) { + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, + nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + + rte_cpu_to_be_16(0x0100); + chksum += chksum >= 0xffff; + ip0->hdr_checksum = chksum; + ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; + } + if (unlikely(next_index ^ pvar.next0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -233,13 +470,15 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, held += last_spec; last_spec = 0; - rte_node_enqueue_x1(graph, node, next0, from[0]); + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); from += 1; } else { last_spec += 1; } } + IP4_REWRITE_NODE_LAST_TX(node->ctx) = pvar.last_tx_interface; + /* !!! Home run !!! */ if (likely(last_spec == nb_objs)) { rte_node_next_stream_move(graph, node, next_index); @@ -255,22 +494,78 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, return nb_objs; } +static uint16_t +ip4_rewrite_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = + rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + rte_graph_feature_rt_list_t flist; + + /* If any feature is enabled on this arc */ + if (unlikely(rte_graph_feature_arc_has_any_feature(arc, &flist))) { + if (flist) + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)1); + else + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)0); + } else { + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); + } + return 0; +} + +static uint16_t +ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); +} + static int ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) { + rte_graph_feature_arc_t feature_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; static bool init_once; RTE_SET_USED(graph); RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_node_ctx) > RTE_NODE_CTX_SZ); + RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_nh_header) != RTE_CACHE_LINE_MIN_SIZE); if (!init_once) { node_mbuf_priv1_dynfield_offset = rte_mbuf_dynfield_register( &node_mbuf_priv1_dynfield_desc); if (node_mbuf_priv1_dynfield_offset < 0) return -rte_errno; - init_once = true; + + /* Create ipv4-output feature arc, if not created + */ + if (rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + NULL) < 0) { + if (rte_graph_feature_arc_create(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + RTE_GRAPH_FEATURE_MAX_PER_ARC, + RTE_MAX_ETHPORTS, + ip4_rewrite_node_get(), &feature_arc)) { + return -rte_errno; + } + init_once = true; + } } IP4_REWRITE_NODE_PRIV1_OFF(node->ctx) = node_mbuf_priv1_dynfield_offset; + IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx) = feature_arc; + IP4_REWRITE_NODE_LAST_TX(node->ctx) = UINT16_MAX; node_dbg("ip4_rewrite", "Initialized ip4_rewrite node initialized"); @@ -329,6 +624,7 @@ rte_node_ip4_rewrite_add(uint16_t next_hop, uint8_t *rewrite_data, static struct rte_node_register ip4_rewrite_node = { .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, .name = "ip4_rewrite", /* Default edge i.e '0' is pkt drop */ .nb_edges = 1, diff --git a/lib/node/ip4_rewrite_priv.h b/lib/node/ip4_rewrite_priv.h index 5105ec1d29..52f39601bd 100644 --- a/lib/node/ip4_rewrite_priv.h +++ b/lib/node/ip4_rewrite_priv.h @@ -5,9 +5,11 @@ #define __INCLUDE_IP4_REWRITE_PRIV_H__ #include <rte_common.h> +#include <rte_graph_feature_arc.h> #define RTE_GRAPH_IP4_REWRITE_MAX_NH 64 -#define RTE_GRAPH_IP4_REWRITE_MAX_LEN 56 +#define RTE_GRAPH_IP4_REWRITE_MAX_LEN (sizeof(struct rte_ether_hdr) + \ + (2 * sizeof(struct rte_vlan_hdr))) /** * @internal @@ -15,11 +17,9 @@ * Ipv4 rewrite next hop header data structure. Used to store port specific * rewrite data. */ -struct ip4_rewrite_nh_header { - uint16_t rewrite_len; /**< Header rewrite length. */ +struct __rte_cache_aligned ip4_rewrite_nh_header { uint16_t tx_node; /**< Tx node next index identifier. */ - uint16_t enabled; /**< NH enable flag */ - uint16_t rsvd; + uint16_t rewrite_len; /**< Header rewrite length. */ union { struct { struct rte_ether_addr dst; @@ -30,8 +30,13 @@ struct ip4_rewrite_nh_header { uint8_t rewrite_data[RTE_GRAPH_IP4_REWRITE_MAX_LEN]; /**< Generic rewrite data */ }; + /* used in control path */ + uint8_t enabled; /**< NH enable flag */ }; +_Static_assert(sizeof(struct ip4_rewrite_nh_header) <= (size_t)RTE_CACHE_LINE_SIZE, + "ip4_rewrite_nh_header size must be less or equal to cache line"); + /** * @internal * diff --git a/lib/node/node_private.h b/lib/node/node_private.h index 1de7306792..25db04a9a6 100644 --- a/lib/node/node_private.h +++ b/lib/node/node_private.h @@ -12,6 +12,9 @@ #include <rte_mbuf.h> #include <rte_mbuf_dyn.h> +#include <rte_graph_worker_common.h> +#include <rte_graph_feature_arc_worker.h> + extern int rte_node_logtype; #define RTE_LOGTYPE_NODE rte_node_logtype @@ -29,15 +32,28 @@ extern int rte_node_logtype; */ struct node_mbuf_priv1 { union { - /* IP4/IP6 rewrite */ + /** + * IP4/IP6 rewrite + * only used to pass lookup data from + * ip4-lookup to ip4-rewrite + */ struct { uint16_t nh; uint16_t ttl; uint32_t cksum; }; - uint64_t u; }; + /** + * Feature arc data + */ + struct { + /** interface index */ + uint16_t if_index; + /** feature that current mbuf holds */ + rte_graph_feature_t current_feature; + uint8_t rsvd; + }; }; static const struct rte_mbuf_dynfield node_mbuf_priv1_dynfield_desc = { diff --git a/lib/node/rte_node_ip4_api.h b/lib/node/rte_node_ip4_api.h index 24f8ec843a..0de06f7fc7 100644 --- a/lib/node/rte_node_ip4_api.h +++ b/lib/node/rte_node_ip4_api.h @@ -23,6 +23,7 @@ extern "C" { #include <rte_compat.h> #include <rte_graph.h> +#include <rte_graph_feature_arc_worker.h> /** * IP4 lookup next nodes. @@ -67,6 +68,8 @@ struct rte_node_ip4_reassembly_cfg { /**< Node identifier to configure. */ }; +#define RTE_IP4_OUTPUT_FEATURE_ARC_NAME "ipv4-output" + /** * Add ipv4 route to lookup table. * -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH v2 4/5] test/graph_feature_arc: add functional tests 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena ` (2 preceding siblings ...) 2024-10-08 13:30 ` [RFC PATCH v2 3/5] graph: add IPv4 output feature arc Nitin Saxena @ 2024-10-08 13:30 ` Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 5/5] docs: add programming guide for feature arc Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena Added functional unit test case for verifying feature arc control plane and fast path APIs How to run: $ echo "graph_feature_arc_autotest" | ./bin/dpdk-test Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1415 +++++++++++++++++++++++++++++ 2 files changed, 1416 insertions(+) create mode 100644 app/test/test_graph_feature_arc.c diff --git a/app/test/meson.build b/app/test/meson.build index e29258e6ec..740fa1bfb4 100644 --- a/app/test/meson.build +++ b/app/test/meson.build @@ -90,6 +90,7 @@ source_file_deps = { 'test_func_reentrancy.c': ['hash', 'lpm'], 'test_graph.c': ['graph'], 'test_graph_perf.c': ['graph'], + 'test_graph_feature_arc.c': ['graph'], 'test_hash.c': ['net', 'hash'], 'test_hash_functions.c': ['hash'], 'test_hash_multiwriter.c': ['hash'], diff --git a/app/test/test_graph_feature_arc.c b/app/test/test_graph_feature_arc.c new file mode 100644 index 0000000000..e185fcd393 --- /dev/null +++ b/app/test/test_graph_feature_arc.c @@ -0,0 +1,1415 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "test.h" + +#include <assert.h> +#include <inttypes.h> +#include <signal.h> +#include <stdalign.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> + +#include <rte_errno.h> + +#ifndef RTE_EXEC_ENV_WINDOWS +#include <rte_graph.h> +#include <rte_graph_worker.h> +#include <rte_mbuf.h> +#include <rte_mbuf_dyn.h> +#include <rte_random.h> +#include <rte_graph_feature_arc.h> +#include <rte_graph_feature_arc_worker.h> + +#define MBUFF_SIZE 512 +#define TEST_ARC1_NAME "arc1" +#define TEST_ARC2_NAME "arc2" +#define MAX_INDEXES 10 +#define MAX_FEATURES 5 + +#define SOURCE1 "test_node_arc_source1" +#define INPUT_STATIC "test_node_arc_input_static" +#define OUTPUT_STATIC "test_node_arc_output_static" +#define PKT_FREE_STATIC "test_node_arc_pkt_free_static" +#define ARC1_FEATURE1 "test_node_arc1_feature1" +#define ARC1_FEATURE2 "test_node_arc1_feature2" +#define ARC2_FEATURE1 "test_node_arc2_feature1" +#define ARC2_FEATURE2 "test_node_arc2_feature2" +#define ARC2_FEATURE3 "test_node_arc2_feature3" +#define DUMMY1_STATIC "test_node_arc_dummy1_static" +#define DUMMY2_STATIC "test_node_arc_dummy2_static" + +/* (Node index, Node Name, feature user data base */ +#define FOREACH_TEST_NODE_ARC { \ + R(0, SOURCE1, 64) \ + R(1, INPUT_STATIC, 128) \ + R(2, OUTPUT_STATIC, 256) \ + R(3, PKT_FREE_STATIC, 512) \ + R(4, ARC1_FEATURE1, 1024) \ + R(5, ARC1_FEATURE2, 2048) \ + R(6, ARC2_FEATURE1, 4096) \ + R(7, ARC2_FEATURE2, 8192) \ + R(8, ARC2_FEATURE3, 16384) \ + R(9, DUMMY1_STATIC, 32768) \ + R(10, DUMMY2_STATIC, 65536) \ + } + +/** + * ARC1: Feature arc on ingress interface + * ARC2: Feature arc on egress interface + * XX_static: Static nodes + * XX_featureX: Feature X on arc + * + * -----> ARC1_FEATURE1 + * | | | + * | | v + * | | ARC1_FEATURE2 + * | | | + * | v v + * SOURCE1 ->-----> INPUT_STATIC --> OUTPUT_STATIC -----> PKT_FREE_STATIC + * | | | ^ ^ ^ + * | | | | | | + * | | --> ARC2_FEATURE1 | | + * | | ^ ^ | | + * | | | | | | + * | ----------c-> ARC2_FEATURE2 | + * | | ^ | + * | | | | + * ----------> ARC2_FEATURE3 ------- + */ +const char *node_names_feature_arc[] = { + SOURCE1, INPUT_STATIC, OUTPUT_STATIC, PKT_FREE_STATIC, + ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE2, ARC2_FEATURE3, + DUMMY1_STATIC, DUMMY2_STATIC +}; + +#define MAX_NODES RTE_DIM(node_names_feature_arc) + +/* Function declaraions */ +static uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static int +common_node_init(const struct rte_graph *graph, struct rte_node *node); + +typedef struct test_node_priv { + /* index fom 0 - MAX_NODES -1 */ + uint8_t node_index; + + /* feature */ + rte_graph_feature_t feature; + + /* rte_graph node id */ + uint32_t node_id; + + rte_graph_feature_arc_t arc; +} test_node_priv_t; + +typedef struct { + rte_graph_feature_t feature; + uint16_t egress_interface; + uint16_t ingress_interface; +} graph_dynfield_t; + +static int graph_dynfield_offset = -1; +static rte_graph_feature_arc_t arcs[RTE_GRAPH_FEATURE_ARC_MAX + 128]; +static struct rte_mbuf mbuf[MAX_NODES + 1][MBUFF_SIZE]; +static void *mbuf_p[MAX_NODES + 1][MBUFF_SIZE]; +static rte_graph_t graph_id = RTE_GRAPH_ID_INVALID; + +const char *node_patterns_feature_arc[] = { + "test_node_arc*" +}; + +static inline graph_dynfield_t * +graph_field(struct rte_mbuf *mbuf) +{ + return RTE_MBUF_DYNFIELD(mbuf, graph_dynfield_offset, graph_dynfield_t *); +} + +static int32_t +compute_unique_user_data(const char *parent, const char *child, uint32_t interface_index) +{ + uint32_t user_data = interface_index; + + RTE_SET_USED(parent); +#define R(idx, node, node_cookie) { \ + if (!strcmp(child, node)) { \ + user_data += node_cookie; \ + } \ + } + + FOREACH_TEST_NODE_ARC +#undef R + + return user_data; +} + +static int +get_edge(struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t count, i; + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +int +common_node_init(const struct rte_graph *graph, struct rte_node *node) +{ + test_node_priv_t *priv = (test_node_priv_t *)node->ctx; + + RTE_SET_USED(graph); + + priv->node_id = node->id; + priv->feature = RTE_GRAPH_FEATURE_INVALID; + priv->arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + +#define R(idx, _name, user_data) { \ + if (!strcmp(node->name, _name)) { \ + priv->node_index = idx; \ + } \ + } + FOREACH_TEST_NODE_ARC +#undef R + + return 0; +} + +uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register source1 = { + .name = SOURCE1, + .process = source1_fn, + .flags = RTE_NODE_SOURCE_F, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {INPUT_STATIC, DUMMY1_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(source1); + +uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register input = { + .name = INPUT_STATIC, + .process = input_fn, + .feat_arc_proc = input_fa_fn, + .nb_edges = 2, + .init = common_node_init, + .next_nodes = {OUTPUT_STATIC, DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(input); + +uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register output = { + .name = OUTPUT_STATIC, + .process = output_fn, + .feat_arc_proc = output_fa_fn, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC, PKT_FREE_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(output); + +uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register pkt_free = { + .name = PKT_FREE_STATIC, + .process = pkt_free_fn, + .feat_arc_proc = pkt_free_fa_fn, + .nb_edges = 1, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(pkt_free); + +uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy1 = { + .name = DUMMY1_STATIC, + .process = dummy1_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(dummy1); + +uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy2 = { + .name = DUMMY2_STATIC, + .process = dummy2_fn, + .nb_edges = 5, + .init = common_node_init, + .next_nodes = { ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, + ARC2_FEATURE2, ARC2_FEATURE3}, +}; +RTE_NODE_REGISTER(dummy2); + +uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature1 = { + .name = ARC1_FEATURE1, + .process = arc1_feature1_fn, + .feat_arc_proc = arc1_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature1); + +uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature2 = { + .name = ARC1_FEATURE2, + .process = arc1_feature2_fn, + .feat_arc_proc = arc1_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature2); + +uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature1 = { + .name = ARC2_FEATURE1, + .process = arc2_feature1_fn, + .feat_arc_proc = arc2_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature1); + +uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature2 = { + .name = ARC2_FEATURE2, + .process = arc2_feature2_fn, + .feat_arc_proc = arc2_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature2); + +uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature3 = { + .name = ARC2_FEATURE3, + .process = arc2_feature3_fn, + .feat_arc_proc = arc2_feature3_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature3); + +static int +create_graph(void) +{ + struct rte_graph_param gconf = { + .socket_id = SOCKET_ID_ANY, + .nb_node_patterns = 1, + .node_patterns = node_patterns_feature_arc, + }; + + graph_id = rte_graph_create("worker0", &gconf); + if (graph_id == RTE_GRAPH_ID_INVALID) { + printf("Graph creation failed with error = %d\n", rte_errno); + return TEST_FAILED; + } + + return TEST_SUCCESS; +} + +static int +__test_create_feature_arc(rte_graph_feature_arc_t *arcs, int max_arcs) +{ + rte_graph_feature_arc_t arc; + const char *sample_arc_name = "sample_arc"; + char arc_name[256]; + int n_arcs; + + /* Create max number of feature arcs first */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + if (rte_graph_feature_arc_create(arc_name, MAX_FEATURES, + MAX_INDEXES, &dummy1, &arcs[n_arcs])) { + printf("Feature arc creation failed for %u\n", n_arcs); + return TEST_FAILED; + } + } + /* Verify feature arc created more than max_arcs must fail */ + if (!rte_graph_feature_arc_create("negative_test_create_arc", MAX_FEATURES, + MAX_INDEXES, &dummy2, &arc)) { + printf("Feature arc creation success for more than max configured: %u\n", n_arcs); + return TEST_FAILED; + } + /* Make sure lookup passes for all feature arcs */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + if (!rte_graph_feature_arc_lookup_by_name(arc_name, &arc)) { + if (arc != arcs[n_arcs]) { + printf("%s: Feature arc lookup mismatch for arc [%p, exp: %p]\n", + arc_name, (void *)arc, (void *)arcs[n_arcs]); + return TEST_FAILED; + } + } else { + printf("Feature arc lookup %s failed after creation\n", arc_name); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_create(void) +{ + int ret = 0, i; + + /* Create arcs with RTE_GRAPH_FEATURE_ARC_MAX */ + ret = __test_create_feature_arc(arcs, RTE_GRAPH_FEATURE_ARC_MAX); + if (ret) { + printf("Feature arc creation test failed for RTE_GRAPH_FEATURE_ARC_MAX arcs\n"); + return TEST_FAILED; + } + /* destroy all arcs via cleanup API*/ + ret = rte_graph_feature_arc_cleanup(); + if (ret) { + printf("Feature arc cleanup failed\n"); + return TEST_FAILED; + } + +#define NUM_FEAT_ARCS 128 + /* create 128 dummy feature arcs */ + ret = rte_graph_feature_arc_init(NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc init failed for NUM_FEAT_ARCS"); + return TEST_FAILED; + } + ret = __test_create_feature_arc(arcs, NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc creation test failed for NUM_FEAT_ARCS\n"); + return TEST_FAILED; + } + /* destroy all of them*/ + for (i = 0; i < NUM_FEAT_ARCS; i++) { + if (rte_graph_feature_arc_destroy(arcs[i])) { + printf("Feature arc destroy failed for %u\n", i); + return TEST_FAILED; + } + } + rte_graph_feature_arc_cleanup(); + + /* Create two arcs as per test plan */ + /* First arc start/source node is node: SOURCE1 */ + if (rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[0])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + /* Duplicate name should fail */ + if (!rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[1])) { + printf("Duplicate feature arc %s creation is not caught\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Second arc start/source node is node: OUTPUT_STATIC */ + if (rte_graph_feature_arc_create(TEST_ARC2_NAME, MAX_FEATURES, + MAX_INDEXES, &output, &arcs[1])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_features_add(void) +{ + rte_graph_feature_t temp; + + /* First feature to SOURCE1 start node -> ARC1_FEATURE1 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature1, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1); + return TEST_FAILED; + } + /* Second feature to SOURCE1 -> ARC1_FEATURE2 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature2, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + /* adding statically connected INPUT_STATIC as a last feature */ + if (rte_graph_feature_add(arcs[0], &input, ARC1_FEATURE2, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC1_NAME, INPUT_STATIC, ARC1_FEATURE2); + return TEST_FAILED; + } + /* First feature to OUTPUT_STATIC start node -> ARC2_FEATURE3 */ + if (rte_graph_feature_add(arcs[1], &arc2_feature3, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Second feature to OUTPUT_STATIC -> ARC2_FEATURE1 and before feature to + * ARC2_FEATURE3 + */ + if (rte_graph_feature_add(arcs[1], &arc2_feature1, NULL, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Add PKT_FREE node as last feature, next to arc2_feature3 */ + if (rte_graph_feature_add(arcs[1], &pkt_free, ARC2_FEATURE3, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Adding feature ARC2_FEATURE2 between ARC2_FEATURE1 and ARC2_FEATURE3. */ + if (rte_graph_feature_add(arcs[1], &arc2_feature2, ARC2_FEATURE1, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s between [%s - %s]\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Now check feature sequencing is correct for both ARCS */ + + /* arc1_featur1 must be first feature to arcs[0] */ + if (!strstr(ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc1_feature2 must be second feature to arcs[0] */ + if (!strstr(ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* Make sure INPUT_STATIC is the last feature in arcs[0] */ + temp = rte_graph_feature_arc_num_features(arcs[0]); + if (!strstr(INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC1_NAME, INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_featur1 must be first feature to arcs[1] */ + if (!strstr(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc2_feature2 must be second feature to arcs[1] */ + if (!strstr(ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_feature3 must be third feature to arcs[1] */ + if (!strstr(ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(2)))) { + printf("%s: %s is not the third feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(2))); + return TEST_FAILED; + } + + /* Make sure PKT_FREE is the last feature in arcs[1] */ + temp = rte_graph_feature_arc_num_features(arcs[1]); + if (!strstr(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + if (get_edge(&arc2_feature1, &pkt_free, NULL)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, PKT_FREE_STATIC); + return TEST_FAILED; + } + + return create_graph(); +} + +static int +test_graph_feature_arc_first_feature_enable(void) +{ + uint32_t n_indexes, n_features, count = 0; + rte_graph_feature_rt_list_t feature_list, temp = 0; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + char *feature_name = NULL; + int32_t user_data; + rte_edge_t edge = ~0; + + arc = rte_graph_feature_arc_get(arcs[0]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 0, + * On interface 1, enable feature 1 and so on so forth + * + * later verify first feature on every interface index is unique + * and check [rte_edge, user_data] retrieved via fast path APIs + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = n_indexes % 3 /* 3 features added to arc1 */; + feature_name = rte_graph_feature_arc_feature_to_name(arcs[0], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_validate(arcs[0], n_indexes, feature_name, 1, true)) { + printf("%s: Feature validate failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* negative test case. enable feature on invalid index */ + if (!n_indexes && !rte_graph_feature_enable(arcs[0], MAX_INDEXES, feature_name, + (int32_t)user_data)) { + printf("%s: Feature %s should not be enabled on invalid index\n", + TEST_ARC1_NAME, feature_name); + return TEST_FAILED; + } + if (rte_graph_feature_enable(arcs[0], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + if (temp == feature_list) { + printf("%s: Activer feature list not switched from %u -> %u\n", + TEST_ARC1_NAME, temp, feature_list); + return TEST_FAILED; + } + temp = feature_list; + if ((count + 1) != rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Number of enabled mismatches [found: %u, exp: %u]\n", + TEST_ARC1_NAME, + rte_graph_feature_arc_num_enabled_features(arcs[0]), + count + 1); + return TEST_FAILED; + } + count++; + } + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Negative test case */ + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, 1); + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC2_FEATURE1, user_data)) { + printf("%s: Invalid feature %s is enabled on index 1\n", + TEST_ARC1_NAME, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Duplicate enable */ + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC1_FEATURE2, user_data)) { + printf("%s: Duplicate feature %s shouldn't be enabled again on index 1\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC1_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc1_feature1; + else if (1 == (n_indexes % 3)) + child = &arc1_feature2; + else + child = &input; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC1_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +verify_feature_sequencing(struct rte_graph_feature_arc *arc) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_t feature; + uint32_t n_indexes; + rte_edge_t edge = ~0; + int32_t user_data; + + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: feature_list can't be obtained\n", + arc->feature_arc_name); + return TEST_FAILED; + } + /* Verify next features on interface 0 and interface 1*/ + for (n_indexes = 0; n_indexes < 2; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: 0\n", + arc->feature_arc_name); + return TEST_FAILED; + } + parent = arc->start_node; + child = rte_graph_feature_arc_feature_to_node(arcs[1], feature); + /* until fast path API reaches last feature i.e pkt_free */ + while (child != &pkt_free) { + fdata = rte_graph_feature_data_get(arc, + rte_graph_feature_get(arc, feature), + n_indexes); + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + arc->feature_arc_name, parent->name, child->name); + return TEST_FAILED; + } + user_data = compute_unique_user_data(parent->name, child->name, n_indexes); + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != user_data) { + printf("%s: Udata mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->user_data, user_data); + return TEST_FAILED; + } + + feature = fdata->next_enabled_feature; + + parent = child; + child = rte_graph_feature_arc_feature_to_node(arcs[1], + fdata->next_enabled_feature); + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_enable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + uint32_t n_indexes, n_features; + rte_graph_feature_t feature; + char *feature_name = NULL; + rte_edge_t edge = ~0; + int32_t user_data; + + arc = rte_graph_feature_arc_get(arcs[1]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[1])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 2, skip feature 1 for later + * On interface 1, enable feature 3 + * On interface 2, enable pkt_free feature + * On interface 3, continue as interface 0 + * + * later enable next feature sequence for interface 0 from feature2 -> pkt_free + * later enable next feature sequence for interface 1 from feature3 -> pkt_free + * + * also later enable feature-1 and see first feature changes for all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = (n_indexes % 3) + 1; /* feature2 to pkt_free are 3 features */ + feature_name = rte_graph_feature_arc_feature_to_name(arcs[1], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + } + /* Retrieve latest feature_list */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + /* verify first feature */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc2_feature2; + else if (1 == (n_indexes % 3)) + child = &arc2_feature3; + else + child = &pkt_free; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + /* add next_features now + * On interface 0, enable feature-3 and pkt_free + * On interface 1, enable pkt_free + * Skip interface 2 + * On interface 3, same as interface 0 + * On interface 4, same as interface 1 + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (0 == (n_indexes % 3)) { + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE3, + compute_unique_user_data(ARC2_FEATURE2, + ARC2_FEATURE3, + n_indexes))) { + printf("%s: Feature enable failed for %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + /* pkt_free on interface-0, 1, 3, 4 and so on */ + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_enable(arcs[1], n_indexes, PKT_FREE_STATIC, + compute_unique_user_data(ARC2_FEATURE3, + PKT_FREE_STATIC, + n_indexes))) { + printf("%s: Feature enable failed %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + /* Enable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE1, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: None first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature != rte_graph_feature_cast(0)) { + printf("%s: First feature mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(0)); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_first_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* Disable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE1)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE1, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature == rte_graph_feature_cast(0)) { + printf("%s: First feature not disabled on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(1)); + return TEST_FAILED; + } + if (!strncmp(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(ARC2_FEATURE1))) { + printf("%s: First feature mismatch on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + ARC2_FEATURE2); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* + * On interface 0, disable feature 2, keep feature3 and pkt_free enabled + * On interface 1, skip interface 1 where feature3 and pkt_free are enabled + * skip interface 2 as only pkt_free is enabled + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!(n_indexes % 3)) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE2)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, n_indexes); + return TEST_FAILED; + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + } + + /** + * Disable feature 3 on all interface 0 and 1 and check first feature + * is pkt_free on all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE3)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + } + /* Make sure pkt_free is first feature for all indexes */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (strncmp(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(PKT_FREE_STATIC))) { + printf("%s: %s is not first feature found on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + PKT_FREE_STATIC); + return TEST_FAILED; + } + } + + /* Disable PKT_FREE_STATIC from all indexes with no feature enabled on any interface */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, PKT_FREE_STATIC)) { + printf("%s: Feat disable failed for %s on index %u\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + /* Make sure no feature is enabled now on any interface */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: Index: %u should not have first feature enabled\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_destroy(void) +{ + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC1_NAME, &arc)) { + printf("Feature arc lookup failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (arc != arcs[0]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC1_NAME, (void *)arc, (void *)arcs[0]); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC2_NAME, &arc)) { + printf("Feature arc lookup success after destroy for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (arc != arcs[1]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC2_NAME, (void *)arc, (void *)arcs[1]); + return TEST_FAILED; + } + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +graph_feature_arc_setup(void) +{ + unsigned long i, j; + + static const struct rte_mbuf_dynfield graph_dynfield_desc = { + .name = "test_graph_dynfield", + .size = sizeof(graph_dynfield_t), + .align = alignof(graph_dynfield_t), + }; + + graph_dynfield_offset = + rte_mbuf_dynfield_register(&graph_dynfield_desc); + if (graph_dynfield_offset < 0) { + printf("Cannot register mbuf field\n"); + return TEST_FAILED; + } + + for (i = 0; i <= MAX_NODES; i++) { + for (j = 0; j < MBUFF_SIZE; j++) + mbuf_p[i][j] = &mbuf[i][j]; + } + + return TEST_SUCCESS; + +} + +static void +graph_feature_arc_teardown(void) +{ + if (graph_id != RTE_GRAPH_ID_INVALID) + rte_graph_destroy(graph_id); + + rte_graph_feature_arc_cleanup(); +} + +static struct unit_test_suite graph_feature_arc_testsuite = { + .suite_name = "Graph Feature arc library test suite", + .setup = graph_feature_arc_setup, + .teardown = graph_feature_arc_teardown, + .unit_test_cases = { + TEST_CASE(test_graph_feature_arc_create), + TEST_CASE(test_graph_feature_arc_features_add), + TEST_CASE(test_graph_feature_arc_first_feature_enable), + TEST_CASE(test_graph_feature_arc_next_feature_enable), + TEST_CASE(test_graph_feature_arc_first_feature_disable), + TEST_CASE(test_graph_feature_arc_next_feature_disable), + TEST_CASE(test_graph_feature_arc_destroy), + TEST_CASES_END(), /**< NULL terminate unit test array */ + }, +}; + +static int +graph_feature_arc_autotest_fn(void) +{ + return unit_test_suite_runner(&graph_feature_arc_testsuite); +} + +REGISTER_FAST_TEST(graph_feature_arc_autotest, true, true, graph_feature_arc_autotest_fn); +#endif /* !RTE_EXEC_ENV_WINDOWS */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [RFC PATCH v2 5/5] docs: add programming guide for feature arc 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena ` (3 preceding siblings ...) 2024-10-08 13:30 ` [RFC PATCH v2 4/5] test/graph_feature_arc: add functional tests Nitin Saxena @ 2024-10-08 13:30 ` Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-08 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan Cc: dev, Nitin Saxena Updated graph library guide with feature arc Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/prog_guide/graph_lib.rst | 289 ++++++++++++++++++++ doc/guides/prog_guide/img/feature_arc-1.jpg | Bin 0 -> 48984 bytes doc/guides/prog_guide/img/feature_arc-2.jpg | Bin 0 -> 113287 bytes doc/guides/prog_guide/img/feature_arc-3.jpg | Bin 0 -> 93408 bytes 4 files changed, 289 insertions(+) create mode 100644 doc/guides/prog_guide/img/feature_arc-1.jpg create mode 100644 doc/guides/prog_guide/img/feature_arc-2.jpg create mode 100644 doc/guides/prog_guide/img/feature_arc-3.jpg diff --git a/doc/guides/prog_guide/graph_lib.rst b/doc/guides/prog_guide/graph_lib.rst index ad09bdfe26..45c7695c80 100644 --- a/doc/guides/prog_guide/graph_lib.rst +++ b/doc/guides/prog_guide/graph_lib.rst @@ -547,3 +547,292 @@ on success packet is enqueued to ``udp4_input`` node. Hash lookup is performed in ``udp4_input`` node with registered destination port and destination port in UDP packet , on success packet is handed to ``udp_user_node``. + +Feature Arc +----------- +`Feature arc` represents an ordered list of `protocols/features` at a given +networking layer. It is a high level abstraction to connect various `feature` +nodes in `rte_graph` instance and allows seamless packets steering based on the +sequence of enabled features at runtime on each interface. + +`Features` (or feature nodes) are nodes which handle partial or complete +protocol processing in a given direction. For instance, `ipv4-rewrite` and +`IPv4 IPsec encryption` are outbound features while `ipv4-lookup` and `IPv4 +IPsec decryption` are inbound features. Further, `ipv4-rewrite` and `IPv4 +IPsec encryption` can collectively represent a `feature arc` towards egress +direction with ordering constraints that `IPv4 IPsec encryption` must be +performed before `ipv4-rewrite`. Similarly, `IPv4 IPsec decryption` and +`ipv4-lookup` can represent a `feature arc` in an ingress direction. Both of +these `feature arc` can co-exist at an IPv4 layer in egress and ingress +direction respectively. + +A `feature` can be represented by a single node or collection of multiple nodes +performing feature processing collectively. + +.. figure:: img/feature_arc-1.jpg + :alt: feature-arc-1 + :width: 350px + :align: center + + Feature Arc overview + +Each `feature arc` is associated with a `Start` node from which all features in +a feature arc are connected. A `start` node itself is not a `feature` node but +it is where `first enabled feature` is checked in fast path. In above figure, +`Node-A` represents a `start node`. There may be a `Sink` node as well which +is child node for every feature in an arc. 'Sink` node is responsible of +consuming those packets which are not consumed by intermediate enabled features +between `start` and `sink` node. `Sink` node, if present, is the last enabled +feature in a feature arc. A `feature` node statically connected to `start` node +must also be added via feature arc API, `rte_graph_feature_add()``. Here `Node-B` +acts as a `sink` node which is statically linked to `Node A`. `Feature` nodes +are connected via `rte_graph_feature_add()` which takes care of connecting +all `feature` nodes with each other and start node. + +.. code-block:: bash + :linenos: + :emphasize-lines: 8 + :caption: Node-B statically linked to Node-A + + static struct rte_node_register node_A_node = { + .process = node_A_process_func, + ... + ... + .name = "Node-A", + .next_nodes = + { + [0] = "Node-B", + }, + .nb_edges = 1, + }; + +When multiple features are enabled on an interface, it may be required to steer +packets across `features` in a given order. For instance, if `Feature 1` and +`Feature 2` both are enabled on an interface ``X``, it may be required to send +packets to `Feature-1` before `Feature-2`. Such ordering constrainsts +can be easily expressed with `feature arc`. In this case, `Feature 1` is called as +``First Feature`` and `Feature 2` is called as ``Next Feature`` to `Feature 1`. + +.. figure:: img/feature_arc-2.jpg + :alt: feature-arc-2 + :width: 600px + :align: center + + First and Next features and their ordering + +In similar manner, even application specific ``custom features`` can be hooked +to standard nodes. It is to be noted that this `custom feature` hooking to +`feature arc` aware node does not require any code changes. + +It may be obvious by now that `features` enabled on one interface does not +affect packets on other interfaces. In above example, if no feature is +enabled on an interface ``X``, packets destined to interface ``X`` would be +directly sent to `Node-B` from `Node-A`. + +.. figure:: img/feature_arc-3.jpg + :alt: feature-arc-3 + :width: 450px + :align: center + + Feature-2 consumed/non-consumed packet path + +When a `Feature-X` node receives packets via feature arc, it may decide whether +to ``consume packet`` or send to `next enabled feature`. A node can consume +packet by freeing it, sending it on wire or enqueuing it to hardware queue. If a +packet is not consumed by a `Feature-X` node, it may send to `next enabled +feature` on an interface. In above figure, `Feature-2` nodes are represented to +consume packets. Classic example for a node performing consume and non-consume +operation on packets would be IPsec policy node where all packets with +``protect`` actions are consumed while remanining packets with ``bypass`` +actions are sent to next enabled feature. + +In fast path feature node may require to lookup local data structures for each +interface. For example, retrieving policy database per interface for IPsec +processing. ``rte_graph_feature_enable`` API allows to set application +specific cookie per feature per interface. `Feature data` object maintains this +cookie in fast path for each interface. + +`Feature arc design` allows to enable subsequent features in a control plane +without stopping workers which are accessing feature arc's fast path APIs in +``rte_graph_walk()`` context. However for disabling features require RCU like +scheme for synchronization. + +Programming model +~~~~~~~~~~~~~~~~~ +Feature Arc Objects +^^^^^^^^^^^^^^^^^^^ +Control plane and fast path APIs deals with following objects: + +Feature arc +*********** +``rte_graph_feature_arc_t`` is a handle to feature arc which is created via +``rte_graph_feature_arc_create()``. It is a `uint64_t` size object which can be +saved in feature node's context. This object can be translated to fast path +feature arc object ``struct rte_graph_feature_arc`` which is an input +argument to all fast path APIs. Control plane APIs majorly takes +`rte_graph_feature_arc_t` object as an input. + +Feature List +************ +Each feature arc holds two feature lists: `active` and `passive`. While worker +cores uses `active` list, control plane APIs uses `passive` list for +enabling/disabling a feature on any interface with in a arc. After successful +feature enable/disable, ``rte_graph_feature_enable()``/ +``rte_graph_feature_disable()`` atomically switches passive list to active list +and vice-versa. Most of the fast path APIs takes active list as an argument +(``rte_graph_feature_rt_list_t``), which feature node can obtain in start of +it's `process_func()` via ``rte_graph_feature_arc_has_any_feature()`` (in `start` +node) or ``rte_graph_feature_arc_has_feature()`` (in next feature nodes). + +Each feature list holds RTE_GRAPH_MAX_FEATURES number of features and +associated feature data for every interface index + +Feature +******** +Feature is a data structure which holds `feature data` object for every +interface. It is represented via ``rte_graph_feature_t`` which is a `uint8_t` +size object. Fast path internal structure ``struct rte_graph_feature`` can be +obtained from ``rte_graph_feature_t`` via ``rte_graph_feature_get()`` API. + +In `start` node `rte_graph_feature_arc_first_feature_get()` can be used to get +first enabled `rte_graph_feature_t` object for an interface. `rte_edge` from +`start` node to first enabled feature is provided by +``rte_graph_feature_arc_feature_set()`` API. + +In `feature nodes`, next enabled feature is obtained by providing current feature +as an input to ``rte_graph_feature_arc_next_feature_get()`` API. + +Feature data +************ +Feature data object is maintained per feature per interface which holds +following information in fast path + +- ``rte_edge_t`` to send packet to next enabled feature +- ``Next enabled feature`` on current interface +- ``User_data`` per feature per interface set by application via `rte_graph_feature_enable()` + +Enabling Feature Arc processing +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +By default, feature arc processing is disabled in `rte_graph_create()`. To +enable feature arc processing in fast path, `rte_graph_create()` API should be +invoked with `feature_arc_enable` flag set as `true` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Enabling feature are processing in rte_graph_create() + + struct rte_graph_param graph_conf; + + graph_conf.feature_arc_enable = true; + struct rte_graph *graph = rte_graph_create("graph_name", &graph_conf); + +Further as an optimization technique, `rte_graph_walk()` would call newly added +``feat_arc_proc()`` node callback function (if non-NULL) instead of +``preocess_func`` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Feature arc specific node callback function + + static struct rte_node_register ip4_rewrite_node = { + .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, + .name = "ip4_rewrite", + ... + ... + }; + +If `feat_arc_proc` is not provided in node registration, `process_func` would +be called by `rte_graph_walk()` + +Sample Usage +^^^^^^^^^^^^ +.. code-block:: bash + :linenos: + :emphasize-lines: 29,38,49, 53,68,69,74 + :caption: Feature arc sample usage + + #define MAX_FEATURES 10 + #define MAX_INDEXES 5 + + static uint16_t + feature2_feature_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* features may be enabled */ + } + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc is disabled in rte_graph_create() */ + } + + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc may be enabled or disabled as this process_func() would + * be called for the case when feature arc is enabled in rte_graph_create() + * and also the case when it is disabled + */ + } + + static struct rte_node_register feature2_node = { + .process = feature2_node_process, + .feat_arc_proc = feature2_feature_node_process, + .name = "feature2", + .init = feature2_init_func, + ... + ... + }; + + static struct rte_node_register feature1_node = { + .process = feature1_node_process, + .feat_arc_proc = NULL, + .name = "feature1", + ... + ... + }; + + int worker_cb(void *_em) + { + rte_graph_feature_arc_t arc; + uint32_t user_data; + + rte_graph_feature_arc_lookup_by_name("sample_arc", &arc); + user_data = 0x1234; + + /* enable feature2 on interface index 4 */ + rte_graph_feature_enable(arc, 4 /* interface index */, "feature2", user_data); + + while(1) { + rte_graph_walk); + } + } + + int main(void) + { + struct rte_graph_param graph_conf; + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_create("sample_arc", MAX_FEATURES, MAX_INDEXES, &arc)) + return -1; + + rte_graph_feature_add(arc, "feature1", NULL, NULL); + rte_graph_feature_add(arc, "feature2", "feature1" /* add feature2 after feature 1*/, NULL); + + /* create graph*/ + ... + ... + graph_conf.feature_arc_enable = true; + + struct rte_graph *graph = rte_graph_create("sample_graph", &graph_conf); + + rte_eal_mp_remote_launch(worker_cb, arg, CALL_MAIN); + } + + + diff --git a/doc/guides/prog_guide/img/feature_arc-1.jpg b/doc/guides/prog_guide/img/feature_arc-1.jpg new file mode 100644 index 0000000000000000000000000000000000000000..7103286b2e75c49b5f7dc77b2f8b0c61d8c5302d GIT binary patch literal 48984 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 bQsDS?UG901$3M@V{$64D|NG;pKa>9pi3Yx6 literal 0 HcmV?d00001 diff --git a/doc/guides/prog_guide/img/feature_arc-2.jpg b/doc/guides/prog_guide/img/feature_arc-2.jpg new file mode 100644 index 0000000000000000000000000000000000000000..1a3e3dd0a62028bae7090edc6f536a73910b58da GIT binary patch literal 113287 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%0A&(of@<u3Kum!PcL48@V}bek+Wpy;Qtlp8-3#m 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�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 literal 0 HcmV?d00001 diff --git a/doc/guides/prog_guide/img/feature_arc-3.jpg b/doc/guides/prog_guide/img/feature_arc-3.jpg new file mode 100644 index 0000000000000000000000000000000000000000..36b754bbd4072dbf558fce3db69aea436d34757e GIT binary patch literal 93408 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 NzYt1cf1LmSO#nJ?14IA- literal 0 HcmV?d00001 -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v3 0/5] add feature arc in rte_graph 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena ` (4 preceding siblings ...) 2024-10-08 13:30 ` [RFC PATCH v2 5/5] docs: add programming guide for feature arc Nitin Saxena @ 2024-10-09 13:29 ` Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 1/5] graph: add feature arc support Nitin Saxena ` (7 more replies) 5 siblings, 8 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-09 13:29 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on whether feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Changes in v3: - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix compilation on 32-bit machine - Updated images in .png format - Added ABI change section in release notes - Fixed DPDK CI failures Changes in v2: - Added unit tests for feature arc - Fixed issues found in testing - Added new public APIs rte_graph_feature_arc_feature_to_node(), rte_graph_feature_arc_feature_to_name(), rte_graph_feature_arc_num_features() - Added programming guide for feature arc - Added release notes for feature arc Nitin Saxena (5): graph: add feature arc support graph: add feature arc option in graph create graph: add IPv4 output feature arc test/graph_feature_arc: add functional tests docs: add programming guide for feature arc app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ doc/guides/prog_guide/graph_lib.rst | 288 ++++ doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes doc/guides/rel_notes/release_24_11.rst | 13 + lib/graph/graph.c | 1 + lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++ lib/graph/graph_populate.c | 7 +- lib/graph/graph_private.h | 3 + lib/graph/meson.build | 2 + lib/graph/node.c | 2 + lib/graph/rte_graph.h | 3 + lib/graph/rte_graph_feature_arc.h | 431 ++++++ lib/graph/rte_graph_feature_arc_worker.h | 674 +++++++++ lib/graph/version.map | 20 + lib/node/ip4_rewrite.c | 476 +++++-- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 21 files changed, 4494 insertions(+), 98 deletions(-) create mode 100644 app/test/test_graph_feature_arc.c create mode 100644 doc/guides/prog_guide/img/feature_arc-1.png create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v3 1/5] graph: add feature arc support 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena @ 2024-10-09 13:29 ` Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 2/5] graph: add feature arc option in graph create Nitin Saxena ` (6 subsequent siblings) 7 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-09 13:29 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena add feature arc to allow dynamic steering of packets across graph nodes based on protocol features enabled on incoming or outgoing interface Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 10 + lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++++++++ lib/graph/meson.build | 2 + lib/graph/rte_graph_feature_arc.h | 431 ++++++++ lib/graph/rte_graph_feature_arc_worker.h | 674 ++++++++++++ lib/graph/version.map | 20 + 6 files changed, 2360 insertions(+) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index e0a9aa55a1..d6d64518e0 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -67,6 +67,16 @@ New Features The new statistics are useful for debugging and profiling. +* **Added feature arc abstraction in graph library.** + + Feature arc abstraction helps ``rte_graph`` based applications to steer + packets across different node path(s) based on the features (or protocols) + enabled on interfaces. Different feature node paths can be enabled/disabled + at runtime on some or on all interfaces. This abstraction also help + applications to hook their ``custom nodes`` in standard DPDK node paths + without any code changes in the later. + + * Added ``ip4-output`` feature arc processing in ``ip4_rewrite`` node. Removed Items ------------- diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c new file mode 100644 index 0000000000..ff99f7b26a --- /dev/null +++ b/lib/graph/graph_feature_arc.c @@ -0,0 +1,1223 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "graph_private.h" +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> + +#define ARC_PASSIVE_LIST(arc) (arc->active_feature_list ^ 0x1) + +#define rte_graph_uint_cast(x) ((unsigned int)x) +#define feat_dbg graph_dbg + +static rte_graph_feature_arc_main_t *__rte_graph_feature_arc_main; + +/* Make sure fast path cache line is compact */ +_Static_assert((offsetof(struct rte_graph_feature_arc, slow_path_variables) + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) + <= RTE_CACHE_LINE_SIZE, + "Fast path feature arc variables exceed cache line size"); + +#define connect_graph_nodes(node1, node2, edge, arc_name) \ + __connect_graph_nodes(node1, node2, edge, arc_name, __LINE__) + +#define FEAT_COND_ERR(cond, fmt, ...) \ + do { \ + if (cond) \ + graph_err(fmt, ##__VA_ARGS__); \ + } while (0) + +/* + * lookup feature name and get control path node_list as well as feature index + * at which it is inserted + */ +static int +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = UINT32_MAX; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + fi++; + } + return -1; +} + +/* Lookup used only during rte_graph_feature_add() */ +static int +feature_add_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + /* Update slot where new feature can be added */ + if (slot) + *slot = fi; + fi++; + } + + return -1; +} + +/* Get control path node info from provided input feature_index */ +static int +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t feature_index, + struct rte_graph_feature_node_list **ppfinfo, + const int do_sanity_check) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + if (!ppfinfo) + return -1; + + *ppfinfo = NULL; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + /* Check sanity */ + if (do_sanity_check) + if (finfo->node_index != index) + RTE_VERIFY(0); + if (index == feature_index) { + *ppfinfo = finfo; + return 0; + } + index++; + } + return -1; +} + +/* prepare feature arc after addition of all features */ +static void +prepare_feature_arc_before_first_enable(struct rte_graph_feature_arc *arc) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + arc->active_feature_list = 0; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + finfo->node_index = index; + feat_dbg("\t%s prepare: %s added to list at index: %u", arc->feature_arc_name, + finfo->feature_node->name, index); + index++; + } +} + +/* feature arc lookup in array */ +static int +feature_arc_lookup(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (arc == (rte_graph_feature_arc_get(dm->feature_arcs[iter]))) + return 0; + } + return -1; +} + +/* Check valid values for known fields in arc to make sure arc is sane */ +static int check_feature_arc_sanity(rte_graph_feature_arc_t _arc, int iter) +{ +#ifdef FEATURE_ARC_DEBUG + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + RTE_VERIFY(arc->feature_arc_main == __rte_graph_feature_arc_main); + RTE_VERIFY(arc->feature_arc_index == iter); + + RTE_VERIFY(arc->feature_list[0]->indexed_by_features = arc->features[0]); + RTE_VERIFY(arc->feature_list[1]->indexed_by_features = arc->features[1]); + + RTE_VERIFY(arc->active_feature_list < 2); +#else + RTE_SET_USED(_arc); + RTE_SET_USED(iter); +#endif + return 0; +} + +/* Perform sanity on all arc if any corruption occurred */ +static int do_sanity_all_arcs(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!dm) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (check_feature_arc_sanity(dm->feature_arcs[iter], iter)) + return -1; + } + return 0; +} + +/* get existing edge from parent_node -> child_node */ +static int +get_existing_edge(const char *arc_name, struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t i, count = 0; + + RTE_SET_USED(arc_name); + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +/* create or retrieve already existing edge from parent_node -> child_node */ +static int +__connect_graph_nodes(struct rte_node_register *parent_node, struct rte_node_register *child_node, + rte_edge_t *_edge, char *arc_name, int lineno) +{ + const char *next_node = NULL; + rte_edge_t edge; + + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { + feat_dbg("\t%s/%d: %s[%u]: \"%s\", edge reused", arc_name, lineno, + parent_node->name, edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; + } + + /* Node to be added */ + next_node = child_node->name; + + edge = rte_node_edge_update(parent_node->id, RTE_EDGE_ID_INVALID, &next_node, 1); + + if (edge == RTE_EDGE_ID_INVALID) { + graph_err("edge invalid"); + return -1; + } + edge = rte_node_edge_count(parent_node->id) - 1; + + feat_dbg("\t%s/%d: %s[%u]: \"%s\", new edge added", arc_name, lineno, parent_node->name, + edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; +} + +/* feature arc initialization */ +static int +feature_arc_main_init(rte_graph_feature_arc_main_t **pfl, uint32_t max_feature_arcs) +{ + rte_graph_feature_arc_main_t *pm = NULL; + uint32_t i; + size_t sz; + + if (!pfl) + return -1; + + sz = sizeof(rte_graph_feature_arc_main_t) + + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); + + pm = rte_malloc("rte_graph_feature_arc_main", sz, 0); + if (!pm) + return -1; + + memset(pm, 0, sz); + + for (i = 0; i < max_feature_arcs; i++) + pm->feature_arcs[i] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + pm->max_feature_arcs = max_feature_arcs; + + *pfl = pm; + + return 0; +} + +/* feature arc initialization, public API */ +int +rte_graph_feature_arc_init(int max_feature_arcs) +{ + if (!max_feature_arcs) + return -1; + + if (__rte_graph_feature_arc_main) + return -1; + + return feature_arc_main_init(&__rte_graph_feature_arc_main, max_feature_arcs); +} + +/* reset feature list before switching to passive list */ +static void +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t list_index) +{ + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + uint32_t i, j; + + list = arc->feature_list[list_index]; + feat = arc->features[list_index]; + + /*Initialize variables*/ + memset(feat, 0, arc->feature_size * arc->max_features); + memset(list, 0, arc->feature_list_size); + + /* Initialize feature and feature_data */ + for (i = 0; i < arc->max_features; i++) { + feat = __rte_graph_feature_get(arc, i, list_index); + feat->this_feature_index = i; + + for (j = 0; j < arc->max_indexes; j++) { + fdata = rte_graph_feature_data_get(arc, feat, j); + fdata->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + fdata->next_edge = UINT16_MAX; + fdata->user_data = UINT32_MAX; + } + } + + for (i = 0; i < arc->max_indexes; i++) + list->first_enabled_feature_by_index[i] = RTE_GRAPH_FEATURE_INVALID; +} + +static int +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char *flist_name, + rte_graph_feature_list_t **pplist, + struct rte_graph_feature **ppfeature, uint32_t list_index) +{ + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + size_t list_size, feat_size, fdata_size; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + + list_size = sizeof(struct rte_graph_feature_list) + + (sizeof(list->first_enabled_feature_by_index[0]) * arc->max_indexes); + + list_size = RTE_ALIGN_CEIL(list_size, RTE_CACHE_LINE_SIZE); + + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); + if (!list) + return -ENOMEM; + + memset(list, 0, list_size); + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); + + /* Let one feature and its associated data per index capture complete + * cache lines + */ + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + fdata_size, + RTE_CACHE_LINE_SIZE); + + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, "feat"); + + feat = rte_malloc(fname, feat_size * arc->max_features, RTE_CACHE_LINE_SIZE); + if (!feat) { + rte_free(list); + return -ENOMEM; + } + arc->feature_size = feat_size; + arc->feature_data_size = fdata_size; + arc->feature_list_size = list_size; + + /* Initialize list */ + list->indexed_by_features = feat; + *pplist = list; + *ppfeature = feat; + + feature_arc_list_reset(arc, list_index); + + return 0; +} + +/* free resources allocated in feature_arc_list_init() */ +static void +feature_arc_list_destroy(struct rte_graph_feature_arc *arc, int list_index) +{ + rte_graph_feature_list_t *list = NULL; + + list = arc->feature_list[list_index]; + + rte_free(list->indexed_by_features); + + arc->features[list_index] = NULL; + + rte_free(list); + + arc->feature_list[list_index] = NULL; +} + +int +rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, rte_graph_feature_arc_t *_arc) +{ + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_arc_main_t *dfm = NULL; + struct rte_graph_feature_arc *arc = NULL; + struct rte_graph_feature *df = NULL; + uint32_t iter, j, arc_index; + size_t sz; + + if (!_arc) + SET_ERR_JMP(EINVAL, err, "%s: Invalid _arc", feature_arc_name); + + if (max_features < 2) + SET_ERR_JMP(EINVAL, err, "%s: max_features must be greater than 1", + feature_arc_name); + + if (!start_node) + SET_ERR_JMP(EINVAL, err, "%s: start_node cannot be NULL", + feature_arc_name); + + if (!feature_arc_name) + SET_ERR_JMP(EINVAL, err, "%s: feature_arc name cannot be NULL", + feature_arc_name); + + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) + SET_ERR_JMP(EAGAIN, err, "%s: number of features cannot be greater than 64", + feature_arc_name); + + /* + * Application hasn't called rte_graph_feature_arc_init(). Initialize with + * default values + */ + if (!__rte_graph_feature_arc_main) { + if (rte_graph_feature_arc_init((int)RTE_GRAPH_FEATURE_ARC_MAX) < 0) { + graph_err("rte_graph_feature_arc_init() failed"); + return -1; + } + } + + /* If name is not unique */ + if (!rte_graph_feature_arc_lookup_by_name(feature_arc_name, NULL)) + SET_ERR_JMP(EINVAL, err, "%s: feature arc name already exists", + feature_arc_name); + + dfm = __rte_graph_feature_arc_main; + + /* threshold check */ + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) + SET_ERR_JMP(EAGAIN, err, "%s: max number (%u) of feature arcs reached", + feature_arc_name, dfm->max_feature_arcs); + + /* Find the free slot for feature arc */ + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { + if (dfm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + break; + } + arc_index = iter; + + if (arc_index >= dfm->max_feature_arcs) { + graph_err("No free slot found for num_feature_arc"); + return -1; + } + + /* This should not happen */ + RTE_VERIFY(dfm->feature_arcs[arc_index] == RTE_GRAPH_FEATURE_ARC_INITIALIZER); + + /* size of feature arc + feature_bit_mask_by_index */ + sz = RTE_ALIGN_CEIL(sizeof(*arc) + (sizeof(uint64_t) * max_indexes), RTE_CACHE_LINE_SIZE); + + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); + + if (!arc) { + graph_err("malloc failed for feature_arc_create()"); + return -1; + } + + memset(arc, 0, sz); + + /* Initialize rte_graph port group fixed variables */ + STAILQ_INIT(&arc->all_features); + strncpy(arc->feature_arc_name, feature_arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + arc->feature_arc_main = (void *)dfm; + arc->start_node = start_node; + arc->max_features = max_features; + arc->max_indexes = max_indexes; + arc->feature_arc_index = arc_index; + + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc->features[0], 0) < 0) { + rte_free(arc); + graph_err("feature_arc_list_init(0) failed"); + return -1; + } + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc->features[1], 1) < 0) { + feature_arc_list_destroy(arc, 0); + rte_free(arc); + graph_err("feature_arc_list_init(1) failed"); + return -1; + } + + for (iter = 0; iter < arc->max_features; iter++) { + df = rte_graph_feature_get(arc, iter); + for (j = 0; j < arc->max_indexes; j++) { + gfd = rte_graph_feature_data_get(arc, df, j); + gfd->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + } + } + dfm->feature_arcs[arc->feature_arc_index] = (rte_graph_feature_arc_t)arc; + dfm->num_feature_arcs++; + + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + + do_sanity_all_arcs(); + + feat_dbg("Feature arc %s[%p] created with max_features: %u and indexes: %u", + feature_arc_name, (void *)arc, max_features, max_indexes); + return 0; + +err: + return -rte_errno; +} + +int +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *_runs_after, const char *runs_before) +{ + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo = NULL; + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; + char feature_name[3*RTE_GRAPH_FEATURE_ARC_NAMELEN]; + const char *runs_after = NULL; + uint32_t num_feature = 0; + uint32_t slot, add_flag; + rte_edge_t edge = -1; + + /* sanity */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + graph_err("feature arc not created: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (feature_arc_lookup(_arc)) { + graph_err("invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (arc->runtime_enabled_features) { + graph_err("adding features after enabling any one of them is not supported"); + return -1; + } + + if ((_runs_after != NULL) && (runs_before != NULL) && + (_runs_after == runs_before)) { + graph_err("runs_after and runs_before are same '%s:%s]", _runs_after, + runs_before); + return -1; + } + + if (!feature_node) { + graph_err("feature_node: %p invalid", feature_node); + return -1; + } + + arc = rte_graph_feature_arc_get(_arc); + + if (feature_node->id == RTE_NODE_ID_INVALID) { + graph_err("Invalid node: %s", feature_node->name); + return -1; + } + + if (!feature_add_lookup(arc, feature_node->name, &finfo, &slot)) { + graph_err("%s feature already added", feature_node->name); + return -1; + } + + if (slot >= arc->max_features) { + graph_err("%s: Max features %u added to feature arc", + arc->feature_arc_name, slot); + return -1; + } + + if (strstr(feature_node->name, arc->start_node->name)) { + graph_err("Feature %s cannot point to itself: %s", feature_node->name, + arc->start_node->name); + return -1; + } + + feat_dbg("%s: adding feature node: %s at feature index: %u", arc->feature_arc_name, + feature_node->name, slot); + + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc->feature_arc_name)) { + graph_err("unable to connect %s -> %s", arc->start_node->name, feature_node->name); + return -1; + } + + snprintf(feature_name, sizeof(feature_name), "%s-%s-finfo", + arc->feature_arc_name, feature_node->name); + + finfo = rte_malloc(feature_name, sizeof(*finfo), 0); + if (!finfo) { + graph_err("%s/%s: rte_malloc failed", arc->feature_arc_name, feature_node->name); + return -1; + } + + memset(finfo, 0, sizeof(*finfo)); + + finfo->feature_arc = (void *)arc; + finfo->feature_node = feature_node; + finfo->edge_to_this_feature = edge; + arc->runtime_enabled_features = 0; + + /* + * if no constraints given and provided feature is not the first feature, + * explicitly set "runs_after" as last_feature. Handles the case: + * + * add(f1, NULL, NULL); + * add(f2, NULL, NULL); + */ + num_feature = rte_graph_feature_arc_num_features(_arc); + if (!_runs_after && !runs_before && num_feature) + runs_after = rte_graph_feature_arc_feature_to_name(_arc, num_feature - 1); + else + runs_after = _runs_after; + + /* Check for before and after constraints */ + if (runs_before) { + /* runs_before sanity */ + if (feature_lookup(arc, runs_before, &before_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid before feature name: %s", runs_before); + + if (!before_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_before %s does not exist", runs_before); + + /* + * Starting from 0 to runs_before, continue connecting edges + */ + add_flag = 1; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (!add_flag) + /* Nodes after seeing "runs_before", finfo connects to temp*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, + NULL, arc->feature_arc_name); + /* + * As soon as we see runs_before. stop adding edges + */ + if (!strncmp(temp->feature_node->name, runs_before, + RTE_GRAPH_NAMESIZE)) { + if (!connect_graph_nodes(finfo->feature_node, temp->feature_node, + &edge, arc->feature_arc_name)) + add_flag = 0; + } + + if (add_flag) + /* Nodes before seeing "run_before" are connected to finfo */ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + } + } + + if (runs_after) { + if (feature_lookup(arc, runs_after, &after_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid after feature_name %s", runs_after); + + if (!after_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_after %s does not exist", runs_after); + + /* Starting from runs_after to end continue connecting edges */ + add_flag = 0; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (add_flag) + /* We have already seen runs_after now */ + /* Add all features as next node to current feature*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, NULL, + arc->feature_arc_name); + else + /* Connect initial nodes to newly added node*/ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + + /* as soon as we see runs_after. start adding edges + * from next iteration + */ + if (!strncmp(temp->feature_node->name, runs_after, RTE_GRAPH_NAMESIZE)) + add_flag = 1; + } + + /* add feature next to runs_after */ + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, next_feature); + } else { + if (before_finfo) { + /* add finfo before "before_finfo" element in the list */ + after_finfo = NULL; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (before_finfo == temp) { + if (after_finfo) + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, + finfo, next_feature); + else + STAILQ_INSERT_HEAD(&arc->all_features, finfo, + next_feature); + + return 0; + } + after_finfo = temp; + } + } else { + /* Very first feature just needs to be added to list */ + STAILQ_INSERT_TAIL(&arc->all_features, finfo, next_feature); + } + } + + return 0; + +finfo_free: + rte_free(finfo); + + return -1; +} + +int +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot; + + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { + *feat = (rte_graph_feature_t) slot; + return 0; + } + + return -1; +} + +int +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int is_enable_disable, bool emit_logs) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature *gf = NULL; + uint32_t slot; + + /* validate _arc */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + FEAT_COND_ERR(emit_logs, "invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -EINVAL; + } + + /* validate index */ + if (index >= arc->max_indexes) { + FEAT_COND_ERR(emit_logs, "%s: Invalid provided index: %u >= %u configured", + arc->feature_arc_name, index, arc->max_indexes); + return -1; + } + + /* validate feature_name is already added or not */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) { + FEAT_COND_ERR(emit_logs, "%s: No feature %s added", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + if (!finfo) { + FEAT_COND_ERR(emit_logs, "%s: No feature: %s found", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + /* slot should be in valid range */ + if (slot >= arc->max_features) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid free slot %u(max=%u) for feature", + arc->feature_arc_name, feature_name, slot, arc->max_features); + return -EINVAL; + } + + /* slot should be in range of 0 - 63 */ + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid slot: %u", arc->feature_arc_name, + feature_name, slot); + return -EINVAL; + } + + if (finfo->node_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s/%s: lookup slot mismatch for finfo idx: %u and lookup slot: %u", + arc->feature_arc_name, feature_name, finfo->node_index, slot); + return -1; + } + + /* Get feature from active list */ + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(arc)); + if (gf->this_feature_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s: %s rcvd feature_idx: %u does not match with saved: %u", + arc->feature_arc_name, feature_name, slot, gf->this_feature_index); + return -1; + } + + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & + RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s already enabled on index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + if (!is_enable_disable && !arc->runtime_enabled_features) { + FEAT_COND_ERR(emit_logs, "%s: No feature enabled to disable", + arc->feature_arc_name); + return -1; + } + + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s not enabled in bitmask for index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + return 0; +} + +/* + * Before switch to passive list, user_data needs to be copied from active list to passive list + */ +static void +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t dest_list_index, + uint16_t src_list_index) +{ + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; + struct rte_graph_feature *sgf = NULL, *dgf = NULL; + uint32_t i, j; + + for (i = 0; i < arc->max_features; i++) { + sgf = __rte_graph_feature_get(arc, i, src_list_index); + dgf = __rte_graph_feature_get(arc, i, dest_list_index); + for (j = 0; j < arc->max_indexes; j++) { + sgfd = rte_graph_feature_data_get(arc, sgf, j); + dgfd = rte_graph_feature_data_get(arc, dgf, j); + dgfd->user_data = sgfd->user_data; + } + } +} +/* + * Fill fast path information like + * - next_edge + * - next_enabled_feature + */ +static void +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t list_index) +{ + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; + uint32_t fi = UINT32_MAX, di = UINT32_MAX, prev_fi = UINT32_MAX; + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; + rte_graph_feature_list_t *flist = NULL; + rte_edge_t edge = UINT16_MAX; + uint64_t bitmask = 0; + + flist = arc->feature_list[list_index]; + + for (di = 0; di < arc->max_indexes; di++) { + bitmask = arc->feature_bit_mask_by_index[di]; + prev_fi = RTE_GRAPH_FEATURE_INVALID; + /* for each feature set for index, set fast path data */ + while (rte_bsf64_safe(bitmask, &fi)) { + gf = __rte_graph_feature_get(arc, fi, list_index); + gfd = rte_graph_feature_data_get(arc, gf, di); + RTE_VERIFY(!feature_arc_node_info_lookup(arc, fi, &finfo, 1)); + + /* If previous feature_index was valid in last loop */ + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { + prev_gf = __rte_graph_feature_get(arc, prev_fi, list_index); + prev_gfd = rte_graph_feature_data_get(arc, prev_gf, di); + /* + * Get edge of previous feature node connecting + * to this feature node + */ + RTE_VERIFY(!feature_arc_node_info_lookup(arc, prev_fi, + &prev_finfo, 1)); + if (!get_existing_edge(arc->feature_arc_name, + prev_finfo->feature_node, + finfo->feature_node, &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (%u->%u)%s[%u] = %s", + arc->feature_arc_name, list_index, di, + prev_gfd->user_data, prev_fi, fi, + prev_finfo->feature_node->name, + edge, finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* + * Fill current feature as next enabled + * feature to previous one + */ + prev_gfd->next_enabled_feature = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* On first feature edge of the node to be added */ + if (fi == rte_bsf64(arc->feature_bit_mask_by_index[di])) { + if (!get_existing_edge(arc->feature_arc_name, arc->start_node, + finfo->feature_node, + &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (->%u)%s[%u]=%s", + arc->feature_arc_name, list_index, di, + gfd->user_data, fi, + arc->start_node->name, edge, + finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* Set first feature set array for index*/ + flist->first_enabled_feature_by_index[di] = + (rte_graph_feature_t)fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* Clear current feature index */ + bitmask &= ~RTE_BIT64(fi); + } + } +} + +int +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const + char *feature_name, int32_t user_data) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_rt_list_t passive_list; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Enabling feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (!arc->runtime_enabled_features) + prepare_feature_arc_before_first_enable(arc); + + if (rte_graph_feature_validate(_arc, index, feature_name, 1, true)) + return -1; + + /** This should not fail as validate() has passed */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) + RTE_VERIFY(0); + + passive_list = ARC_PASSIVE_LIST(arc); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); + + /* Set current user-data */ + gfd->user_data = user_data; + + /* Set bitmask in control path bitmask */ + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + refill_feature_fastpath_data(arc, passive_list); + + /* If first time feature getting enabled */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[arc->active_feature_list], + rte_memory_order_relaxed); + + /* On very first feature enable instance */ + if (!finfo->ref_count) + bitmask |= RTE_BIT64(slot); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], + bitmask, rte_memory_order_relaxed); + + /* Slow path updates */ + arc->runtime_enabled_features++; + + /* Increase feature node info reference count */ + finfo->ref_count++; + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After enable, switched active feature list to %u", + arc->feature_arc_name, feature_name, arc->active_feature_list); + + return 0; +} + +int +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + rte_graph_feature_rt_list_t passive_list; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Disable feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (rte_graph_feature_validate(_arc, index, feature_name, 0, true)) + return -1; + + if (feature_lookup(arc, feature_name, &finfo, &slot)) + return -1; + + passive_list = ARC_PASSIVE_LIST(arc); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, arc->active_feature_list); + + /* Reset current user-data */ + gfd->user_data = ~0; + + refill_feature_fastpath_data(arc, passive_list); + + finfo->ref_count--; + arc->runtime_enabled_features--; + + /* If no feature enabled, reset feature in u64 fast path bitmask */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[arc->active_feature_list], + rte_memory_order_relaxed); + + /* When last feature is disabled */ + if (!finfo->ref_count) + bitmask &= ~(RTE_BIT64(slot)); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], bitmask, + rte_memory_order_relaxed); + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After disable, switched active feature list to %u", + arc->feature_arc_name, feature_name, arc->active_feature_list); + + return 0; +} + +int +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_node_list *node_info = NULL; + + while (!STAILQ_EMPTY(&arc->all_features)) { + node_info = STAILQ_FIRST(&arc->all_features); + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); + rte_free(node_info); + } + feature_arc_list_destroy(arc, 0); + feature_arc_list_destroy(arc, 1); + + dm->feature_arcs[arc->feature_arc_index] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + rte_free(arc); + + do_sanity_all_arcs(); + + return 0; +} + +int +rte_graph_feature_arc_cleanup(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm->feature_arcs[iter]); + } + rte_free(dm); + + __rte_graph_feature_arc_main = NULL; + + return 0; +} + +int +rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_arc *arc = NULL; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + if (_arc) + *_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); + + if ((strstr(arc->feature_arc_name, arc_name)) && + (strlen(arc->feature_arc_name) == strlen(arc_name))) { + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + return 0; + } + } + + return -1; +} + +uint32_t +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + return arc->runtime_enabled_features; +} + +uint32_t +rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t count = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) + count++; + + return count; +} + +char * +rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node->name; + + return NULL; +} + +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node; + + return NULL; + +} diff --git a/lib/graph/meson.build b/lib/graph/meson.build index 0cb15442ab..d916176fb7 100644 --- a/lib/graph/meson.build +++ b/lib/graph/meson.build @@ -14,11 +14,13 @@ sources = files( 'graph_debug.c', 'graph_stats.c', 'graph_populate.c', + 'graph_feature_arc.c', 'graph_pcap.c', 'rte_graph_worker.c', 'rte_graph_model_mcore_dispatch.c', ) headers = files('rte_graph.h', 'rte_graph_worker.h') +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') indirect_headers += files( 'rte_graph_model_mcore_dispatch.h', 'rte_graph_model_rtc.h', diff --git a/lib/graph/rte_graph_feature_arc.h b/lib/graph/rte_graph_feature_arc.h new file mode 100644 index 0000000000..1615f8e1c8 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc.h @@ -0,0 +1,431 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ +#define _RTE_GRAPH_FEATURE_ARC_H_ + +#include <assert.h> +#include <errno.h> +#include <signal.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <rte_common.h> +#include <rte_compat.h> +#include <rte_debug.h> +#include <rte_graph.h> +#include <rte_graph_worker.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @file + * + * rte_graph_feature_arc.h + * + * Define APIs and structures/variables with respect to feature arc + * + * - Feature arc(s) + * - Feature(s) + * + * A feature arc represents an ordered list of features/protocol-nodes at a + * given networking layer. Feature arc provides a high level abstraction to + * connect various *rte_graph* nodes, designated as *feature nodes*, and + * allowing steering of packets across these feature nodes fast path processing + * in a generic manner. In a typical network stack, often a protocol or feature + * must be first enabled on a given interface, before any packet is steered + * towards it for feature processing. For eg: incoming IPv4 packets are sent to + * routing sub-system only after a valid IPv4 address is assigned to the + * received interface. In other words, often packets needs to be steered across + * features not based on the packet content but based on whether a feature is + * enable or disable on a given incoming/outgoing interface. Feature arc + * provides mechanism to enable/disable feature(s) on each interface at runtime + * and allow seamless packet steering across runtime enabled feature nodes in + * fast path. + * + * Feature arc also provides a way to steer packets from standard nodes to + * custom/user-defined *feature nodes* without any change in standard node's + * fast path functions + * + * On a given interface multiple feature(s) might be enabled in a particular + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" + * features may be enabled on "eth0" interface in "L3-output" feature arc. + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" + * interface in same "L3-output" feature arc. + * + * When multiple features are present in a given feature arc, its imperative + * to allow each feature processing in a particular sequential order. For + * instance, in "L3-input" feature arc it may be required to run "IPsec + * input" feature first, for packet decryption, before "ip-lookup". So a + * sequential order must be maintained among features present in a feature arc. + * + * Features are enabled/disabled multiple times at runtime to some or all + * available interfaces present in the system. Enable/disabling features on one + * interface is independent of other interface. + * + * A given feature might consume packet (if it's configured to consume) or may + * forward it to next enabled feature. For instance, "IPsec input" feature may + * consume/drop all packets with "Protect" policy action while all packets with + * policy action as "Bypass" may be forwarded to next enabled feature (with in + * same feature arc) + * + * This library facilitates rte graph based applications to steer packets in + * fast path to different feature nodes with-in a feature arc and support all + * functionalities described above + * + * In order to use feature-arc APIs, applications needs to do following in + * control path: + * - Initialize feature arc library via rte_graph_feature_arc_init() + * - Create feature arc via rte_graph_feature_arc_create() + * - *Before calling rte_graph_create()*, features must be added to feature-arc + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding + * features in a sequential order with "runs_after" and "runs_before" + * constraints. + * - Post rte_graph_create(), features can be enabled/disabled at runtime on + * any interface via rte_graph_feature_enable()/rte_graph_feature_disable() + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() + * + * In fast path, APIs are provided to steer packets towards feature path from + * - start_node (provided as an argument to rte_graph_feature_arc_create()) + * - feature nodes (which are added via rte_graph_feature_add()) + * + * For typical steering of packets across feature nodes, application required + * to know "rte_edges" which are saved in feature data object. Feature data + * object is unique for every interface per feature with in a feature arc. + * + * When steering packets from start_node to feature node: + * - rte_graph_feature_arc_first_feature_get() provides first enabled feature. + * - Next rte_edge from start_node to first enabled feature can be obtained via + * rte_graph_feature_arc_feature_set() + * + * rte_mbuf can carry [current feature, interface index] from start_node of an + * arc to other feature nodes + * + * At the time of feature enable(rte_graph_feature_enable), application can set + * 32-bit unique user_data specific to feature per interface. In fast path + * user_data can be retrieved via rte_graph_feature_user_data_get(). User data + * can hold application specific cookie like IPsec policy database index, FIB + * table index etc. + * + * If feature node is not consuming packet, next enabled feature and next + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() + * + * It is application responsibility to ensure that at-least *last feature*(or + * sink feature) must be enabled from where packet can exit feature-arc path, + * if *NO* intermediate feature is consuming the packet and it has reached till + * the end of feature arc path + * + * It is recommended that all features *MUST* be added to feature arc before + * calling `rte_graph_create()`. Addition of features after + * `rte_graph_create()` may not work functionally. + * Although,rte_graph_feature_enable()/rte_graph_feature_disable() should be + * called after `rte_graph_create()` in control plane. + * + * Synchronization among cores + * --------------------------- + * Subsequent calls to rte_graph_feature_enable() is allowed while worker cores + * are processing in rte_graph_walk() loop. However, for + * rte_graph_feature_disable() application must use RCU based synchronization + */ + +/** Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER ((rte_graph_feature_arc_t)UINT64_MAX) + +/** Max number of feature arcs which can be created */ +#define RTE_GRAPH_FEATURE_ARC_MAX 64 + +/** Max number of features supported in a given feature arc */ +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 + +/** Length of feature arc name */ +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE + +/** @internal */ +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) + +/**< Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_INVALID rte_graph_feature_cast(UINT8_MAX) + +/** rte_graph feature arc object */ +typedef uintptr_t rte_graph_feature_arc_t; + +/** rte_graph feature object */ +typedef uint8_t rte_graph_feature_t; + +/** runtime active feature list index with in feature arc*/ +typedef uint16_t rte_graph_feature_rt_list_t; + +/** per feature arc monotonically increasing counter to synchronize fast path APIs */ +typedef uint16_t rte_graph_feature_counter_t; + +/** + * Initialize feature arc subsystem + * + * @param max_feature_arcs + * Maximum number of feature arcs required to be supported + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_init(int max_feature_arcs); + +/** + * Create a feature arc + * + * @param feature_arc_name + * Feature arc name with max length of @ref RTE_GRAPH_FEATURE_ARC_NAMELEN + * @param max_features + * Maximum number of features to be supported in this feature arc + * @param max_indexes + * Maximum number of interfaces/ports/indexes to be supported + * @param start_node + * Base node where this feature arc's features are checked in fast path + * @param[out] _arc + * Feature arc object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, + rte_graph_feature_arc_t *_arc); + +/** + * Get feature arc object with name + * + * @param arc_name + * Feature arc name provided to successful @ref rte_graph_feature_arc_create + * @param[out] _arc + * Feature arc object returned. Valid only when API returns SUCCESS + * + * @return + * 0: Success + * <0: Failure. + */ +__rte_experimental +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc); + +/** + * Add a feature to already created feature arc. For instance + * + * 1. Add first feature node: "ipv4-input" to input arc + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); + * + * 2. Add "ipsec-input" feature node after "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", NULL); + * + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", NULL, "ipv4-input"); + * + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4-input", "ipsec-input"); + * + * @param _arc + * Feature arc handle returned from @ref rte_graph_feature_arc_create() + * @param feature_node + * Graph node representing feature. On success, feature_node is next_node of + * feature_arc->start_node + * @param runs_after + * Add this feature_node after already added "runs_after". Creates + * start_node -> runs_after -> this_feature sequence + * @param runs_before + * Add this feature_node before already added "runs_before". Creates + * start_node -> this_feature -> runs_before sequence + * + * <I> Must be called before rte_graph_create() </I> + * <I> rte_graph_feature_add() is not allowed after call to + * rte_graph_feature_enable() so all features must be added before they can be + * enabled </I> + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *runs_after, const char *runs_before); + +/** + * Enable feature within a feature arc + * + * Must be called after @b rte_graph_create(). + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param user_data + * Application specific data which is retrieved in fast path + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int32_t user_data); + +/** + * Validate whether subsequent enable/disable feature would succeed or not. + * API is thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param is_enable_disable + * If 1, validate whether subsequent @ref rte_graph_feature_enable would pass or not + * If 0, validate whether subsequent @ref rte_graph_feature_disable would pass or not + * @param emit_logs + * If passed true, emit error logs when failure is returned + * If passed false, do not emit error logs when failure is returned + * + * @return + * 0: Subsequent enable/disable API would pass + * <0: Subsequent enable/disable API would not pass + */ +__rte_experimental +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, int is_enable_disable, bool emit_logs); + +/** + * Disable already enabled feature within a feature arc + * + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name); + +/** + * Get rte_graph_feature_t object from feature name + * + * @param arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param feature_name + * Feature name provided to @ref rte_graph_feature_add + * @param[out] feature + * Feature object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_lookup(rte_graph_feature_arc_t arc, const char *feature_name, + rte_graph_feature_t *feature); + +/** + * Delete feature_arc object + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); + +/** + * Cleanup all feature arcs + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_cleanup(void); + +/** + * Slow path API to know how many features are added (NOT enabled) within a + * feature arc + * + * @param _arc + * Feature arc object + * + * @return: Number of added features to arc + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to know how many features are currently enabled within a + * feature arc across all indexes. If a single feature is enabled on all interfaces, + * this API would return "number_of_interfaces" as count (but not "1") + * + * @param _arc + * Feature arc object + * + * @return: Number of enabled features across all indexes + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to get feature node name from rte_graph_feature_t object + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: Name of the feature node + */ +__rte_experimental +char *rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + +/** + * Slow path API to get corresponding struct rte_node_register * from + * rte_graph_feature_t + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: struct rte_node_register * of feature node on SUCCESS else NULL + */ +__rte_experimental +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/lib/graph/rte_graph_feature_arc_worker.h b/lib/graph/rte_graph_feature_arc_worker.h new file mode 100644 index 0000000000..ca12b3ffd0 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc_worker.h @@ -0,0 +1,674 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ + +#include <stddef.h> +#include <rte_graph_feature_arc.h> +#include <rte_bitops.h> + +/** + * @file + * + * rte_graph_feature_arc_worker.h + * + * Defines fast path structure + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/** @internal + * + * Slow path feature node info list + */ +struct rte_graph_feature_node_list { + /** Next feature */ + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; + + /** node representing feature */ + struct rte_node_register *feature_node; + + /** How many indexes/interfaces using this feature */ + int32_t ref_count; + + /* node_index in list (after feature_enable())*/ + uint32_t node_index; + + /** Back pointer to feature arc */ + void *feature_arc; + + /** rte_edge_t to this feature node from feature_arc->start_node */ + rte_edge_t edge_to_this_feature; +}; + +/** + * Feature data object: + * + * Feature data stores information to steer packets for: + * - a feature with in feature arc + * - Index i.e. Port/Interface index + * + * Each feature data object holds + * - User data of current feature retrieved via rte_graph_feature_user_data_get() + * - next_edge is used in two conditions when packet to be steered from + * -- start_node to first enabled feature on an interface index + * -- current feature node to next enabled feature on an interface index + * - next_enabled_feature on interface index, if current feature is not + * consuming packet + * + * While user_data corresponds to current enabled feature node however + * next_edge and next_enabled_feature corresponds to next enabled feature + * node on an interface index + * + * First enabled feature on interface index can be retrieved via: + * - rte_graph_feature_first_feature_get() if arc's start_node is trying to + * steer packet from start_node to first enabled feature on interface index + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_next_feature_get() if current node is not arc's + * start_node. Input to rte_graph_feature_next_feature_get() is current + * enabled feature and interface index + */ +typedef struct __rte_packed rte_graph_feature_data { + /** edge from current node to next enabled feature */ + rte_edge_t next_edge; + + union { + uint16_t reserved; + struct { + /** next enabled feature on index from current feature */ + rte_graph_feature_t next_enabled_feature; + }; + }; + + /** user_data set by application in rte_graph_feature_enable() for + * - current feature + * - interface index + */ + int32_t user_data; +} rte_graph_feature_data_t; + +/** + * Feature object + * + * Feature object holds feature data object for every index/interface within + * feature + * + * Within a given arc and interface index, first feature object can be + * retrieved in arc's start_node via: + * - rte_graph_feature_arc_first_feature_get() + * + * Feature data information can be retrieved for first feature in start node via + * - rte_graph_feature_arc_feature_set() + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_arc_next_feature_get() + * + * Typically application stores rte_graph_feature_t object in rte_mbuf. + * rte_graph_feature_t can be translated to (struct rte_graph_feature *) via + * rte_graph_feature_get() in fast path. Further if needed, feature data for an + * index within a feature can be retrieved via rte_graph_feature_data_get() + */ +struct __rte_cache_aligned rte_graph_feature { + /** feature index or rte_graph_feature_t */ + uint16_t this_feature_index; + + /* + * Array of size arc->feature_data_size + * + * <----------------- Feature --------------------------> + * [data-index-0][data-index-1]...[data-index-max_index-1] + * + * sizeof(feature_data_by_index[0] == sizeof(rte_graph_feature_data_t) + * + */ + uint8_t feature_data_by_index[]; +}; + +/** + * Feature list object + * + * Feature list is required to decouple fast path APIs with control path APIs. + * + * There are two feature lists: active, passive + * Passive list is duplicate of active list in terms of memory. + * + * While fast path APIs always work on active list but control plane work on + * passive list. When control plane needs to enable/disable any feature, it + * populates passive list afresh and atomically switch passive list to active + * list to make it available for fast path APIs + * + * Each feature node in start of its fast path function, must grab active list from + * arc via + * - rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * + * Retrieved list must be provided to other feature arc fast path APIs so that + * any control plane changes of active list should not impact current node + * execution iteration. Active list change would be reflected to current node + * in next iteration + * + * With active/passive lists and RCU mechanism in graph worker + * loop, application can update features at runtime without stopping fast path + * cores. A RCU synchronization is required when a feature needs to be + * disabled via rte_graph_feature_disable(). On enabling a feature, RCU + * synchronization may not be required + * + */ +typedef struct __rte_cache_aligned rte_graph_feature_list { + /** + * fast path array holding per_feature data. + * Duplicate entry as feature-arc also hold this pointer + * arc->features[] + * + *<-------------feature-0 ---------><---------feature-1 -------------->... + *[index-0][index-1]...[max_index-1]<-ALIGN->[index-0][index-1] ...[max_index-1]... + */ + struct rte_graph_feature *indexed_by_features; + /* + * fast path array holding first enabled feature per index + * (Required in start_node. In non start_node, mbuf can hold next enabled + * feature) + */ + rte_graph_feature_t first_enabled_feature_by_index[]; +} rte_graph_feature_list_t; + +/** + * rte_graph Feature arc object + * + * Feature arc object holds control plane and fast path information for all + * features and all interface index information for steering packets across + * feature nodes + * + * Within a feature arc, only RTE_GRAPH_FEATURE_MAX_PER_ARC features can be + * added. If more features needs to be added, another feature arc can be + * created + * + * Application gets rte_graph_feature_arc_t object via + * - rte_graph_feature_arc_create() OR + * - rte_graph_feature_arc_lookup_by_name() + * + * In fast path, rte_graph_feature_arc_t can be translated to (struct + * rte_graph_feature_arc *) via rte_graph_feature_arc_get(). Later is needed to + * add as an input argument to all fast path feature arc APIs + */ +struct __rte_cache_aligned rte_graph_feature_arc { + /* First 64B is fast path variables */ + RTE_MARKER fast_path_variables; + + /** runtime active feature list */ + rte_graph_feature_rt_list_t active_feature_list; + + /** Actual Size of feature_list object */ + uint16_t feature_list_size; + + /** + * Size each feature in fastpath. + * Required to navigate from feature to another feature in fast path + */ + uint16_t feature_size; + + /** + * Size of all feature data for an index + * Required to navigate through various feature data within a feature + * in fast path + */ + uint16_t feature_data_size; + + /** + * Quick fast path bitmask indicating if any feature enabled or not on + * any of the indexes. Helps in optimally process packets for the case + * when features are added but not enabled + * + * Separate for active and passive list + */ + uint64_t feature_enable_bitmask[2]; + + /** + * Pointer to both active and passive feature list object + */ + rte_graph_feature_list_t *feature_list[2]; + + /** + * Feature objects for each list + */ + struct rte_graph_feature *features[2]; + + /** index in feature_arc_main */ + uint16_t feature_arc_index; + + uint16_t reserved[3]; + + /** Slow path variables follows*/ + RTE_MARKER slow_path_variables; + + /** feature arc name */ + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** All feature lists */ + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; + + /** control plane counter to track enabled features */ + uint32_t runtime_enabled_features; + + /** Back pointer to feature_arc_main */ + void *feature_arc_main; + + /** Arc's start/base node */ + struct rte_node_register *start_node; + + /** maximum number of features supported by this arc */ + uint32_t max_features; + + /** maximum number of index supported by this arc */ + uint32_t max_indexes; + + /** Slow path bit mask per feature per index */ + uint64_t feature_bit_mask_by_index[]; +}; + +/** + * Feature arc main object + * + * Holds all feature arcs created by application + * + * RTE_GRAPH_FEATURE_ARC_MAX number of feature arcs can be created by + * application via rte_graph_feature_arc_create() + */ +typedef struct feature_arc_main { + /** number of feature arcs created by application */ + uint32_t num_feature_arcs; + + /** max features arcs allowed */ + uint32_t max_feature_arcs; + + /** feature arcs */ + rte_graph_feature_arc_t feature_arcs[]; +} rte_graph_feature_arc_main_t; + +/** @internal Get feature arc pointer from object */ +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc *)arc) + +extern rte_graph_feature_arc_main_t *__feature_arc_main; + +/** + * API to know if feature is valid or not + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_is_valid(rte_graph_feature_t feature) +{ + return (feature != RTE_GRAPH_FEATURE_INVALID); +} + +/** + * Get rte_graph_feature object with no checks + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * @param feature_list + * active feature list retrieved from rte_graph_feature_arc_has_any_feature() + * or rte_graph_feature_arc_has_feature() + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature, + const rte_graph_feature_rt_list_t feature_list) +{ + return ((struct rte_graph_feature *)(((uint8_t *)arc->features[feature_list]) + + (feature * arc->feature_size))); +} + +/** + * Get rte_graph_feature object for a given interface/index from feature arc + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature) +{ + if (unlikely(feature >= arc->max_features)) + RTE_VERIFY(0); + + if (likely(rte_graph_feature_is_valid(feature))) + return __rte_graph_feature_get(arc, feature, arc->active_feature_list); + + return NULL; +} + +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + RTE_SET_USED(arc); + return ((rte_graph_feature_data_t *)(((uint8_t *)feature->feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Get rte_graph feature data object for a index in feature + * + * @param arc + * feature arc + * @param feature + * Pointer to feature object + * @param index + * Index of feature maintained in slow path linked list + * + * @return + * Valid feature data + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + if (likely(index < arc->max_indexes)) + return __rte_graph_feature_data_get(arc, feature, index); + + RTE_VERIFY(0); +} + +/** + * Fast path API to check if any feature enabled on a feature arc + * Typically from arc->start_node process function + * + * @param arc + * Feature arc object + * @param[out] plist + * Pointer to runtime active feature list which needs to be provided to other + * fast path APIs + * + * @return + * 0: If no feature enabled + * Non-Zero: Bitmask of features enabled. plist is valid + * + */ +__rte_experimental +static __rte_always_inline uint64_t +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_rt_list_t *plist) +{ + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Fast path API to check if provided feature is enabled on any interface/index + * or not + * + * @param arc + * Feature arc object + * @param feature + * Input rte_graph_feature_t that needs to be checked + * @param[out] plist + * Returns active list to caller which needs to be provided to other fast path + * APIs + * + * @return + * 1: If input [feature] is enabled in arc + * 0: If input [feature] is not enabled in arc + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_t feature, + rte_graph_feature_rt_list_t *plist) +{ + uint64_t bitmask = RTE_BIT64(feature); + + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (bitmask & rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Prefetch feature arc fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) +{ + rte_prefetch0((void *)&arc->fast_path_variables); +} + +/** + * Prefetch feature related fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature) +{ + /* feature cache line */ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, list)); +} + +/** + * Prefetch feature data upfront. Perform sanity + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object returned from @ref + * rte_graph_feature_arc_first_feature_get() + * @param index + * Interface/index + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, uint32_t index) +{ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)((uint8_t *)arc->features[list] + + offsetof(struct rte_graph_feature, feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Fast path API to get first enabled feature on interface index + * Typically required in arc->start_node so that from returned feature, + * feature-data can be retrieved to steer packets + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. + * + * @return + * 1. Success. If first feature field is enabled and returned [feature] is valid + * 0. Failure. If first feature field is disabled in arc + * + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + + *feature = feature_list->first_enabled_feature_by_index[index]; + + return rte_graph_feature_is_valid(*feature); +} + +/** + * Fast path API to get next enabled feature on interface index with provided + * input feature + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. API sets next enabled feature on [index] + * from provided input feature. Valid only if API returns Success + * @param[out] next_edge + * Edge from current feature to next feature. Valid only if next feature is valid + * + * @return + * 1. Success. first feature field is enabled/valid + * 0. Failure. first feature field is disabled/invalid + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature, + rte_edge_t *next_edge) +{ + rte_graph_feature_data_t *feature_data = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(*feature))) { + f = __rte_graph_feature_get(arc, *feature, list); + feature_data = rte_graph_feature_data_get(arc, f, index); + *feature = feature_data->next_enabled_feature; + *next_edge = feature_data->next_edge; + return rte_graph_feature_is_valid(*feature); + } + + return 0; +} + +/** + * Set fields with respect to first enabled feature in an arc and return edge + * Typically returned feature and interface index must be saved in rte_mbuf + * structure to pass this information to next feature node + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param index + * Index (of interface) + * @param[out] gf + * Pointer to rte_graph_feature_t. Valid if API returns Success + * @param[out] edge + * Edge to steer packet from arc->start_node to first enabled feature. Valid + * only if API returns Success + * + * @return + * 0: If valid feature is enabled and set by API in *gf + * 1: If valid feature is NOT enabled + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_t +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *gf, + rte_edge_t *edge) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + struct rte_graph_feature_data *feature_data = NULL; + struct rte_graph_feature *feature = NULL; + rte_graph_feature_t f; + + f = feature_list->first_enabled_feature_by_index[index]; + + if (unlikely(rte_graph_feature_is_valid(f))) { + feature = __rte_graph_feature_get(arc, f, list); + feature_data = rte_graph_feature_data_get(arc, feature, index); + *gf = f; + *edge = feature_data->next_edge; + return 0; + } + + return 1; +} + +__rte_experimental +static __rte_always_inline int32_t +__rte_graph_feature_user_data_get(rte_graph_feature_data_t *fdata) +{ + return fdata->user_data; +} + +/** + * Get user data corresponding to current feature set by application in + * rte_graph_feature_enable() + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Feature index + * @param index + * Interface index + * + * @return + * -1: Failure + * Valid user data: Success + */ +__rte_experimental +static __rte_always_inline int32_t +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, + uint32_t index) +{ + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(feature))) { + f = __rte_graph_feature_get(arc, feature, list); + fdata = rte_graph_feature_data_get(arc, f, index); + return __rte_graph_feature_user_data_get(fdata); + } + + return -1; +} +#ifdef __cplusplus +} +#endif +#endif diff --git a/lib/graph/version.map b/lib/graph/version.map index 2c83425ddc..3b7f475afd 100644 --- a/lib/graph/version.map +++ b/lib/graph/version.map @@ -52,3 +52,23 @@ DPDK_25 { local: *; }; + +EXPERIMENTAL { + global: + + # added in 24.11 + rte_graph_feature_arc_init; + rte_graph_feature_arc_create; + rte_graph_feature_arc_lookup_by_name; + rte_graph_feature_add; + rte_graph_feature_enable; + rte_graph_feature_validate; + rte_graph_feature_disable; + rte_graph_feature_lookup; + rte_graph_feature_arc_destroy; + rte_graph_feature_arc_cleanup; + rte_graph_feature_arc_num_enabled_features; + rte_graph_feature_arc_num_features; + rte_graph_feature_arc_feature_to_name; + rte_graph_feature_arc_feature_to_node; +}; -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v3 2/5] graph: add feature arc option in graph create 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 1/5] graph: add feature arc support Nitin Saxena @ 2024-10-09 13:29 ` Nitin Saxena 2024-10-09 13:30 ` [PATCH v3 3/5] graph: add IPv4 output feature arc Nitin Saxena ` (5 subsequent siblings) 7 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-09 13:29 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena, Pavan Nikhilesh Added option in graph create to call feature-specific process node functions. This removes extra overhead for checking feature arc status in nodes where application is not using feature arc processing Signed-off-by: Pavan Nikhilesh <pbhagavatula@marvell.com> Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 3 +++ lib/graph/graph.c | 1 + lib/graph/graph_populate.c | 7 ++++++- lib/graph/graph_private.h | 3 +++ lib/graph/node.c | 2 ++ lib/graph/rte_graph.h | 3 +++ 6 files changed, 18 insertions(+), 1 deletion(-) diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index d6d64518e0..5195badf54 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -122,6 +122,9 @@ ABI Changes Also, make sure to start the actual text at the margin. ======================================================= +* graph: Added new `feature_arc_enable` parameter in `struct rte_graph_param` to + allow `rte_graph_walk()` call `feat_arc_proc` node callback function, if it + is provided in node registration Known Issues ------------ diff --git a/lib/graph/graph.c b/lib/graph/graph.c index d5b8c9f918..b0ad3a83ae 100644 --- a/lib/graph/graph.c +++ b/lib/graph/graph.c @@ -455,6 +455,7 @@ rte_graph_create(const char *name, struct rte_graph_param *prm) graph->parent_id = RTE_GRAPH_ID_INVALID; graph->lcore_id = RTE_MAX_LCORE; graph->num_pkt_to_capture = prm->num_pkt_to_capture; + graph->feature_arc_enabled = prm->feature_arc_enable; if (prm->pcap_filename) rte_strscpy(graph->pcap_filename, prm->pcap_filename, RTE_GRAPH_PCAP_FILE_SZ); diff --git a/lib/graph/graph_populate.c b/lib/graph/graph_populate.c index ed596a7711..5d8aa7b903 100644 --- a/lib/graph/graph_populate.c +++ b/lib/graph/graph_populate.c @@ -79,8 +79,13 @@ graph_nodes_populate(struct graph *_graph) if (graph_pcap_is_enable()) { node->process = graph_pcap_dispatch; node->original_process = graph_node->node->process; - } else + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->original_process = graph_node->node->feat_arc_proc; + } else { node->process = graph_node->node->process; + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->process = graph_node->node->feat_arc_proc; + } memcpy(node->name, graph_node->node->name, RTE_GRAPH_NAMESIZE); pid = graph_node->node->parent_id; if (pid != RTE_NODE_ID_INVALID) { /* Cloned node */ diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index d557d55f2d..58ba0abeff 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -56,6 +56,7 @@ struct node { unsigned int lcore_id; /**< Node runs on the Lcore ID used for mcore dispatch model. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Allocated identifier for the node. */ @@ -126,6 +127,8 @@ struct graph { /**< Number of packets to be captured per core. */ char pcap_filename[RTE_GRAPH_PCAP_FILE_SZ]; /**< pcap file name/path. */ + uint8_t feature_arc_enabled; + /**< Graph feature arc. */ STAILQ_HEAD(gnode_list, graph_node) node_list; /**< Nodes in a graph. */ }; diff --git a/lib/graph/node.c b/lib/graph/node.c index 99a9622779..d8fd273543 100644 --- a/lib/graph/node.c +++ b/lib/graph/node.c @@ -90,6 +90,7 @@ __rte_node_register(const struct rte_node_register *reg) goto free; node->flags = reg->flags; node->process = reg->process; + node->feat_arc_proc = reg->feat_arc_proc; node->init = reg->init; node->fini = reg->fini; node->nb_edges = reg->nb_edges; @@ -137,6 +138,7 @@ node_clone(struct node *node, const char *name) /* Clone the source node */ reg->flags = node->flags; reg->process = node->process; + reg->feat_arc_proc = node->feat_arc_proc; reg->init = node->init; reg->fini = node->fini; reg->nb_edges = node->nb_edges; diff --git a/lib/graph/rte_graph.h b/lib/graph/rte_graph.h index ecfec2068a..1a3bd7e1ba 100644 --- a/lib/graph/rte_graph.h +++ b/lib/graph/rte_graph.h @@ -172,6 +172,8 @@ struct rte_graph_param { uint32_t mp_capacity; /**< Capacity of memory pool for dispatch model. */ } dispatch; }; + + bool feature_arc_enable; /**< Enable Graph feature arc. */ }; /** @@ -470,6 +472,7 @@ struct rte_node_register { uint64_t flags; /**< Node configuration flag. */ #define RTE_NODE_SOURCE_F (1ULL << 0) /**< Node type is source. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Node Identifier. */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v3 3/5] graph: add IPv4 output feature arc 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 1/5] graph: add feature arc support Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 2/5] graph: add feature arc option in graph create Nitin Saxena @ 2024-10-09 13:30 ` Nitin Saxena 2024-10-09 13:30 ` [PATCH v3 4/5] test/graph_feature_arc: add functional tests Nitin Saxena ` (4 subsequent siblings) 7 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-09 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena add ipv4-output feature arc in ipv4-rewrite node to allow custom/standard nodes(like outbound IPsec policy node) in outgoing forwarding path Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/node/ip4_rewrite.c | 476 +++++++++++++++++++++++++++++------- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 4 files changed, 417 insertions(+), 97 deletions(-) diff --git a/lib/node/ip4_rewrite.c b/lib/node/ip4_rewrite.c index 34a920df5e..5a160335f2 100644 --- a/lib/node/ip4_rewrite.c +++ b/lib/node/ip4_rewrite.c @@ -15,39 +15,156 @@ #include "ip4_rewrite_priv.h" #include "node_private.h" +#define ALL_PKT_MASK 0xf + struct ip4_rewrite_node_ctx { + rte_graph_feature_arc_t output_feature_arc; /* Dynamic offset to mbuf priv1 */ int mbuf_priv1_off; /* Cached next index */ uint16_t next_index; + uint16_t last_tx; }; +typedef struct rewrite_priv_vars { + union { + struct { + rte_xmm_t xmm1; + }; + struct __rte_packed { + uint16_t next0; + uint16_t next1; + uint16_t next2; + uint16_t next3; + uint16_t last_tx_interface; + uint16_t last_if_feature; + uint16_t actual_feat_mask; + uint16_t speculative_feat_mask; + }; + }; +} rewrite_priv_vars_t; + static struct ip4_rewrite_node_main *ip4_rewrite_nm; #define IP4_REWRITE_NODE_LAST_NEXT(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->next_index) +#define IP4_REWRITE_NODE_LAST_TX(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->last_tx) + #define IP4_REWRITE_NODE_PRIV1_OFF(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->mbuf_priv1_off) -static uint16_t -ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, - void **objs, uint16_t nb_objs) +#define IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->output_feature_arc) + +static __rte_always_inline void +prefetch_mbuf_and_dynfield(struct rte_mbuf *mbuf) { + /* prefetch first cache line required for accessing buf_addr */ + rte_prefetch0((void *)mbuf); +} + +static __rte_always_inline void +check_output_feature_x4(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t flist, + rewrite_priv_vars_t *pvar, struct node_mbuf_priv1 *priv0, + struct node_mbuf_priv1 *priv1, struct node_mbuf_priv1 *priv2, + struct node_mbuf_priv1 *priv3) +{ + uint32_t mask = 0; + uint16_t xor = 0; + + /* + * interface edge's start from 1 and not from 0 as "pkt_drop" + * is next node at 0th index + */ + priv0->if_index = pvar->next0 - 1; + priv1->if_index = pvar->next1 - 1; + priv2->if_index = pvar->next2 - 1; + priv3->if_index = pvar->next3 - 1; + + /* Find out if all packets are sent to last_tx_interface */ + xor = pvar->last_tx_interface ^ priv0->if_index; + xor += priv0->if_index ^ priv1->if_index; + xor += priv1->if_index ^ priv2->if_index; + xor += priv2->if_index ^ priv3->if_index; + + if (likely(!xor)) { + /* copy last interface feature and feature mask */ + priv0->current_feature = priv1->current_feature = + priv2->current_feature = priv3->current_feature = + pvar->last_if_feature; + pvar->actual_feat_mask = pvar->speculative_feat_mask; + } else { + /* create a mask for index which does not have feature + * Also override next edge and if feature enabled, get feature + */ + mask = rte_graph_feature_arc_feature_set(arc, flist, priv0->if_index, + &priv0->current_feature, + &pvar->next0); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv1->if_index, + &priv1->current_feature, + &pvar->next1)) << 1); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv2->if_index, + &priv2->current_feature, + &pvar->next2)) << 2); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv3->if_index, + &priv3->current_feature, + &pvar->next3)) << 3); + + /* + * add last tx and last feature regardless even if feature is + * valid or not + */ + pvar->last_tx_interface = priv3->if_index; + pvar->last_if_feature = priv3->current_feature; + /* Set 0xf if invalid feature to last packet, else 0 */ + pvar->speculative_feat_mask = (priv3->current_feature == + RTE_GRAPH_FEATURE_INVALID) ? ALL_PKT_MASK : 0x0; + pvar->actual_feat_mask = mask; + } +} + +static __rte_always_inline uint16_t +__ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs, + const int dyn, const int check_enabled_features, + struct rte_graph_feature_arc *out_feature_arc, + const rte_graph_feature_rt_list_t flist) +{ + struct node_mbuf_priv1 *priv0 = NULL, *priv1 = NULL, *priv2 = NULL, *priv3 = NULL; struct rte_mbuf *mbuf0, *mbuf1, *mbuf2, *mbuf3, **pkts; struct ip4_rewrite_nh_header *nh = ip4_rewrite_nm->nh; - const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); - uint16_t next0, next1, next2, next3, next_index; - struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; uint16_t n_left_from, held = 0, last_spec = 0; + struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; + rewrite_priv_vars_t pvar; + int64_t fd0, fd1, fd2, fd3; + rte_edge_t fix_spec = 0; void *d0, *d1, *d2, *d3; void **to_next, **from; + uint16_t next_index; rte_xmm_t priv01; rte_xmm_t priv23; int i; - /* Speculative next as last next */ + RTE_SET_USED(fd0); + RTE_SET_USED(fd1); + RTE_SET_USED(fd2); + RTE_SET_USED(fd3); + + /* Initialize speculative variables.*/ + + /* Last interface */ + pvar.last_tx_interface = IP4_REWRITE_NODE_LAST_TX(node->ctx); + /*last next from node ctx*/ next_index = IP4_REWRITE_NODE_LAST_NEXT(node->ctx); + pvar.speculative_feat_mask = ALL_PKT_MASK; + pvar.actual_feat_mask = 0; + rte_prefetch0(nh); pkts = (struct rte_mbuf **)objs; @@ -55,20 +172,47 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, n_left_from = nb_objs; for (i = 0; i < 4 && i < n_left_from; i++) - rte_prefetch0(pkts[i]); + prefetch_mbuf_and_dynfield(pkts[i]); /* Get stream for the speculated next node */ to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + + /* prefetch speculative feature and corresponding data */ + if (check_enabled_features) { + /* + * Get first feature enabled, if any, on last_tx_interface + */ + if (unlikely(rte_graph_feature_arc_first_feature_get(out_feature_arc, + flist, + pvar.last_tx_interface, + (rte_graph_feature_t *) + &pvar.last_if_feature))) { + /* prefetch feature cache line */ + rte_graph_feature_arc_feature_prefetch(out_feature_arc, flist, + pvar.last_if_feature); + + /* prefetch feature data cache line */ + rte_graph_feature_arc_data_prefetch(out_feature_arc, flist, + pvar.last_if_feature, + pvar.last_tx_interface); + /* + * Set speculativa_feat mask to indicate, all 4 packets + * going to feature path + */ + pvar.speculative_feat_mask = 0; + } + } + /* Update Ethernet header of pkts */ while (n_left_from >= 4) { if (likely(n_left_from > 7)) { /* Prefetch only next-mbuf struct and priv area. * Data need not be prefetched as we only write. */ - rte_prefetch0(pkts[4]); - rte_prefetch0(pkts[5]); - rte_prefetch0(pkts[6]); - rte_prefetch0(pkts[7]); + prefetch_mbuf_and_dynfield(pkts[4]); + prefetch_mbuf_and_dynfield(pkts[5]); + prefetch_mbuf_and_dynfield(pkts[6]); + prefetch_mbuf_and_dynfield(pkts[7]); } mbuf0 = pkts[0]; @@ -78,66 +222,138 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 4; n_left_from -= 4; + + /* Copy mbuf private data into private variables */ priv01.u64[0] = node_mbuf_priv1(mbuf0, dyn)->u; priv01.u64[1] = node_mbuf_priv1(mbuf1, dyn)->u; priv23.u64[0] = node_mbuf_priv1(mbuf2, dyn)->u; priv23.u64[1] = node_mbuf_priv1(mbuf3, dyn)->u; - /* Increment checksum by one. */ - priv01.u32[1] += rte_cpu_to_be_16(0x0100); - priv01.u32[3] += rte_cpu_to_be_16(0x0100); - priv23.u32[1] += rte_cpu_to_be_16(0x0100); - priv23.u32[3] += rte_cpu_to_be_16(0x0100); - - /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, - nh[priv01.u16[0]].rewrite_len); - - next0 = nh[priv01.u16[0]].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - ip0->time_to_live = priv01.u16[1] - 1; - ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ - d1 = rte_pktmbuf_mtod(mbuf1, void *); - rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, - nh[priv01.u16[4]].rewrite_len); - - next1 = nh[priv01.u16[4]].tx_node; - ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + - sizeof(struct rte_ether_hdr)); - ip1->time_to_live = priv01.u16[5] - 1; - ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ - d2 = rte_pktmbuf_mtod(mbuf2, void *); - rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, - nh[priv23.u16[0]].rewrite_len); - next2 = nh[priv23.u16[0]].tx_node; - ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + - sizeof(struct rte_ether_hdr)); - ip2->time_to_live = priv23.u16[1] - 1; - ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ - d3 = rte_pktmbuf_mtod(mbuf3, void *); - rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, - nh[priv23.u16[4]].rewrite_len); - - next3 = nh[priv23.u16[4]].tx_node; - ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + - sizeof(struct rte_ether_hdr)); - ip3->time_to_live = priv23.u16[5] - 1; - ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + /* Copy next edge from next hop */ + pvar.next0 = nh[priv01.u16[0]].tx_node; + pvar.next1 = nh[priv01.u16[4]].tx_node; + pvar.next2 = nh[priv23.u16[0]].tx_node; + pvar.next3 = nh[priv23.u16[4]].tx_node; + + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + priv1 = node_mbuf_priv1(mbuf1, dyn); + priv2 = node_mbuf_priv1(mbuf2, dyn); + priv3 = node_mbuf_priv1(mbuf3, dyn); + + /* If feature is enabled, override next edge for each mbuf + * and set node_mbuf_priv data appropriately + */ + check_output_feature_x4(out_feature_arc, flist, + &pvar, priv0, priv1, priv2, priv3); + + /* check_output_feature_x4() returns bit mask which indicates + * which packet is not following feature path, hence normal processing + * has to happen on them + */ + if (unlikely(pvar.actual_feat_mask)) { + if (pvar.actual_feat_mask & 0x1) { + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + } + if (pvar.actual_feat_mask & 0x2) { + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + } + if (pvar.actual_feat_mask & 0x4) { + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + } + if (pvar.actual_feat_mask & 0x8) { + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } + } + } else { + /* Case when no feature is enabled */ + + /* Increment checksum by one. */ + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } /* Enqueue four to next node */ - rte_edge_t fix_spec = - ((next_index == next0) && (next0 == next1) && - (next1 == next2) && (next2 == next3)); + fix_spec = next_index ^ pvar.next0; + fix_spec += next_index ^ pvar.next1; + fix_spec += next_index ^ pvar.next2; + fix_spec += next_index ^ pvar.next3; - if (unlikely(fix_spec == 0)) { + if (unlikely(fix_spec != 0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -146,56 +362,56 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, last_spec = 0; /* next0 */ - if (next_index == next0) { + if (next_index == pvar.next0) { to_next[0] = from[0]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next0, + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); } /* next1 */ - if (next_index == next1) { + if (next_index == pvar.next1) { to_next[0] = from[1]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next1, + rte_node_enqueue_x1(graph, node, pvar.next1, from[1]); } /* next2 */ - if (next_index == next2) { + if (next_index == pvar.next2) { to_next[0] = from[2]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next2, + rte_node_enqueue_x1(graph, node, pvar.next2, from[2]); } /* next3 */ - if (next_index == next3) { + if (next_index == pvar.next3) { to_next[0] = from[3]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next3, + rte_node_enqueue_x1(graph, node, pvar.next3, from[3]); } from += 4; /* Change speculation if last two are same */ - if ((next_index != next3) && (next2 == next3)) { + if ((next_index != pvar.next3) && (pvar.next2 == pvar.next3)) { /* Put the current speculated node */ rte_node_next_stream_put(graph, node, next_index, held); held = 0; /* Get next speculated stream */ - next_index = next3; + next_index = pvar.next3; to_next = rte_node_next_stream_get( graph, node, next_index, nb_objs); } @@ -212,20 +428,41 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 1; n_left_from -= 1; - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, - nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); - - next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + - rte_cpu_to_be_16(0x0100); - chksum += chksum >= 0xffff; - ip0->hdr_checksum = chksum; - ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; - - if (unlikely(next_index ^ next0)) { + pvar.next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + if (pvar.next0 != (pvar.last_tx_interface + 1)) { + priv0->if_index = pvar.next0 - 1; + rte_graph_feature_arc_feature_set(out_feature_arc, flist, + priv0->if_index, + &priv0->current_feature, + &pvar.next0); + pvar.last_tx_interface = priv0->if_index; + pvar.last_if_feature = priv0->current_feature; + } else { + /* current mbuf index is same as last_tx_interface */ + priv0->if_index = pvar.last_tx_interface; + priv0->current_feature = pvar.last_if_feature; + } + } + /* Do the needful if either feature arc is disabled OR + * Invalid feature is present + */ + if (!check_enabled_features || + (priv0->current_feature == RTE_GRAPH_FEATURE_INVALID)) { + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, + nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + + rte_cpu_to_be_16(0x0100); + chksum += chksum >= 0xffff; + ip0->hdr_checksum = chksum; + ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; + } + if (unlikely(next_index ^ pvar.next0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -233,13 +470,15 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, held += last_spec; last_spec = 0; - rte_node_enqueue_x1(graph, node, next0, from[0]); + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); from += 1; } else { last_spec += 1; } } + IP4_REWRITE_NODE_LAST_TX(node->ctx) = pvar.last_tx_interface; + /* !!! Home run !!! */ if (likely(last_spec == nb_objs)) { rte_node_next_stream_move(graph, node, next_index); @@ -255,22 +494,78 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, return nb_objs; } +static uint16_t +ip4_rewrite_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = + rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + rte_graph_feature_rt_list_t flist; + + /* If any feature is enabled on this arc */ + if (unlikely(rte_graph_feature_arc_has_any_feature(arc, &flist))) { + if (flist) + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)1); + else + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)0); + } else { + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); + } + return 0; +} + +static uint16_t +ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); +} + static int ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) { + rte_graph_feature_arc_t feature_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; static bool init_once; RTE_SET_USED(graph); RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_node_ctx) > RTE_NODE_CTX_SZ); + RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_nh_header) != RTE_CACHE_LINE_MIN_SIZE); if (!init_once) { node_mbuf_priv1_dynfield_offset = rte_mbuf_dynfield_register( &node_mbuf_priv1_dynfield_desc); if (node_mbuf_priv1_dynfield_offset < 0) return -rte_errno; - init_once = true; + + /* Create ipv4-output feature arc, if not created + */ + if (rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + NULL) < 0) { + if (rte_graph_feature_arc_create(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + RTE_GRAPH_FEATURE_MAX_PER_ARC, + RTE_MAX_ETHPORTS, + ip4_rewrite_node_get(), &feature_arc)) { + return -rte_errno; + } + init_once = true; + } } IP4_REWRITE_NODE_PRIV1_OFF(node->ctx) = node_mbuf_priv1_dynfield_offset; + IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx) = feature_arc; + IP4_REWRITE_NODE_LAST_TX(node->ctx) = UINT16_MAX; node_dbg("ip4_rewrite", "Initialized ip4_rewrite node initialized"); @@ -329,6 +624,7 @@ rte_node_ip4_rewrite_add(uint16_t next_hop, uint8_t *rewrite_data, static struct rte_node_register ip4_rewrite_node = { .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, .name = "ip4_rewrite", /* Default edge i.e '0' is pkt drop */ .nb_edges = 1, diff --git a/lib/node/ip4_rewrite_priv.h b/lib/node/ip4_rewrite_priv.h index 5105ec1d29..52f39601bd 100644 --- a/lib/node/ip4_rewrite_priv.h +++ b/lib/node/ip4_rewrite_priv.h @@ -5,9 +5,11 @@ #define __INCLUDE_IP4_REWRITE_PRIV_H__ #include <rte_common.h> +#include <rte_graph_feature_arc.h> #define RTE_GRAPH_IP4_REWRITE_MAX_NH 64 -#define RTE_GRAPH_IP4_REWRITE_MAX_LEN 56 +#define RTE_GRAPH_IP4_REWRITE_MAX_LEN (sizeof(struct rte_ether_hdr) + \ + (2 * sizeof(struct rte_vlan_hdr))) /** * @internal @@ -15,11 +17,9 @@ * Ipv4 rewrite next hop header data structure. Used to store port specific * rewrite data. */ -struct ip4_rewrite_nh_header { - uint16_t rewrite_len; /**< Header rewrite length. */ +struct __rte_cache_aligned ip4_rewrite_nh_header { uint16_t tx_node; /**< Tx node next index identifier. */ - uint16_t enabled; /**< NH enable flag */ - uint16_t rsvd; + uint16_t rewrite_len; /**< Header rewrite length. */ union { struct { struct rte_ether_addr dst; @@ -30,8 +30,13 @@ struct ip4_rewrite_nh_header { uint8_t rewrite_data[RTE_GRAPH_IP4_REWRITE_MAX_LEN]; /**< Generic rewrite data */ }; + /* used in control path */ + uint8_t enabled; /**< NH enable flag */ }; +_Static_assert(sizeof(struct ip4_rewrite_nh_header) <= (size_t)RTE_CACHE_LINE_SIZE, + "ip4_rewrite_nh_header size must be less or equal to cache line"); + /** * @internal * diff --git a/lib/node/node_private.h b/lib/node/node_private.h index 1de7306792..25db04a9a6 100644 --- a/lib/node/node_private.h +++ b/lib/node/node_private.h @@ -12,6 +12,9 @@ #include <rte_mbuf.h> #include <rte_mbuf_dyn.h> +#include <rte_graph_worker_common.h> +#include <rte_graph_feature_arc_worker.h> + extern int rte_node_logtype; #define RTE_LOGTYPE_NODE rte_node_logtype @@ -29,15 +32,28 @@ extern int rte_node_logtype; */ struct node_mbuf_priv1 { union { - /* IP4/IP6 rewrite */ + /** + * IP4/IP6 rewrite + * only used to pass lookup data from + * ip4-lookup to ip4-rewrite + */ struct { uint16_t nh; uint16_t ttl; uint32_t cksum; }; - uint64_t u; }; + /** + * Feature arc data + */ + struct { + /** interface index */ + uint16_t if_index; + /** feature that current mbuf holds */ + rte_graph_feature_t current_feature; + uint8_t rsvd; + }; }; static const struct rte_mbuf_dynfield node_mbuf_priv1_dynfield_desc = { diff --git a/lib/node/rte_node_ip4_api.h b/lib/node/rte_node_ip4_api.h index 24f8ec843a..0de06f7fc7 100644 --- a/lib/node/rte_node_ip4_api.h +++ b/lib/node/rte_node_ip4_api.h @@ -23,6 +23,7 @@ extern "C" { #include <rte_compat.h> #include <rte_graph.h> +#include <rte_graph_feature_arc_worker.h> /** * IP4 lookup next nodes. @@ -67,6 +68,8 @@ struct rte_node_ip4_reassembly_cfg { /**< Node identifier to configure. */ }; +#define RTE_IP4_OUTPUT_FEATURE_ARC_NAME "ipv4-output" + /** * Add ipv4 route to lookup table. * -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v3 4/5] test/graph_feature_arc: add functional tests 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena ` (2 preceding siblings ...) 2024-10-09 13:30 ` [PATCH v3 3/5] graph: add IPv4 output feature arc Nitin Saxena @ 2024-10-09 13:30 ` Nitin Saxena 2024-10-09 13:30 ` [PATCH v3 5/5] docs: add programming guide for feature arc Nitin Saxena ` (3 subsequent siblings) 7 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-09 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Added functional unit test case for verifying feature arc control plane and fast path APIs How to run: $ echo "graph_feature_arc_autotest" | ./bin/dpdk-test Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++++++++++++ 2 files changed, 1411 insertions(+) create mode 100644 app/test/test_graph_feature_arc.c diff --git a/app/test/meson.build b/app/test/meson.build index e29258e6ec..740fa1bfb4 100644 --- a/app/test/meson.build +++ b/app/test/meson.build @@ -90,6 +90,7 @@ source_file_deps = { 'test_func_reentrancy.c': ['hash', 'lpm'], 'test_graph.c': ['graph'], 'test_graph_perf.c': ['graph'], + 'test_graph_feature_arc.c': ['graph'], 'test_hash.c': ['net', 'hash'], 'test_hash_functions.c': ['hash'], 'test_hash_multiwriter.c': ['hash'], diff --git a/app/test/test_graph_feature_arc.c b/app/test/test_graph_feature_arc.c new file mode 100644 index 0000000000..8ea7a7d3eb --- /dev/null +++ b/app/test/test_graph_feature_arc.c @@ -0,0 +1,1410 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "test.h" + +#include <assert.h> +#include <inttypes.h> +#include <signal.h> +#include <stdalign.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> + +#include <rte_errno.h> + +#ifndef RTE_EXEC_ENV_WINDOWS +#include <rte_graph.h> +#include <rte_graph_worker.h> +#include <rte_mbuf.h> +#include <rte_mbuf_dyn.h> +#include <rte_random.h> +#include <rte_graph_feature_arc.h> +#include <rte_graph_feature_arc_worker.h> + +#define MBUFF_SIZE 512 +#define TEST_ARC1_NAME "arc1" +#define TEST_ARC2_NAME "arc2" +#define MAX_INDEXES 10 +#define MAX_FEATURES 5 + +#define SOURCE1 "test_node_arc_source1" +#define INPUT_STATIC "test_node_arc_input_static" +#define OUTPUT_STATIC "test_node_arc_output_static" +#define PKT_FREE_STATIC "test_node_arc_pkt_free_static" +#define ARC1_FEATURE1 "test_node_arc1_feature1" +#define ARC1_FEATURE2 "test_node_arc1_feature2" +#define ARC2_FEATURE1 "test_node_arc2_feature1" +#define ARC2_FEATURE2 "test_node_arc2_feature2" +#define ARC2_FEATURE3 "test_node_arc2_feature3" +#define DUMMY1_STATIC "test_node_arc_dummy1_static" +#define DUMMY2_STATIC "test_node_arc_dummy2_static" + +/* (Node index, Node Name, feature user data base */ +#define FOREACH_TEST_NODE_ARC { \ + R(0, SOURCE1, 64) \ + R(1, INPUT_STATIC, 128) \ + R(2, OUTPUT_STATIC, 256) \ + R(3, PKT_FREE_STATIC, 512) \ + R(4, ARC1_FEATURE1, 1024) \ + R(5, ARC1_FEATURE2, 2048) \ + R(6, ARC2_FEATURE1, 4096) \ + R(7, ARC2_FEATURE2, 8192) \ + R(8, ARC2_FEATURE3, 16384) \ + R(9, DUMMY1_STATIC, 32768) \ + R(10, DUMMY2_STATIC, 65536) \ + } + +/** + * ARC1: Feature arc on ingress interface + * ARC2: Feature arc on egress interface + * XX_static: Static nodes + * XX_featureX: Feature X on arc + * + * -----> ARC1_FEATURE1 + * | | | + * | | v + * | | ARC1_FEATURE2 + * | | | + * | v v + * SOURCE1 ->-----> INPUT_STATIC --> OUTPUT_STATIC -----> PKT_FREE_STATIC + * | | | ^ ^ ^ + * | | | | | | + * | | --> ARC2_FEATURE1 | | + * | | ^ ^ | | + * | | | | | | + * | ----------c-> ARC2_FEATURE2 | + * | | ^ | + * | | | | + * ----------> ARC2_FEATURE3 ------- + */ +const char *node_names_feature_arc[] = { + SOURCE1, INPUT_STATIC, OUTPUT_STATIC, PKT_FREE_STATIC, + ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE2, ARC2_FEATURE3, + DUMMY1_STATIC, DUMMY2_STATIC +}; + +#define MAX_NODES RTE_DIM(node_names_feature_arc) + +/* Function declarations */ +static uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static int +common_node_init(const struct rte_graph *graph, struct rte_node *node); + +typedef struct test_node_priv { + /* index from 0 - MAX_NODES -1 */ + uint8_t node_index; + + /* feature */ + rte_graph_feature_t feature; + + /* rte_graph node id */ + uint32_t node_id; + + rte_graph_feature_arc_t arc; +} test_node_priv_t; + +typedef struct { + rte_graph_feature_t feature; + uint16_t egress_interface; + uint16_t ingress_interface; +} graph_dynfield_t; + +static int graph_dynfield_offset = -1; +static rte_graph_feature_arc_t arcs[RTE_GRAPH_FEATURE_ARC_MAX + 128]; +static struct rte_mbuf mbuf[MAX_NODES + 1][MBUFF_SIZE]; +static void *mbuf_p[MAX_NODES + 1][MBUFF_SIZE]; +static rte_graph_t graph_id = RTE_GRAPH_ID_INVALID; + +const char *node_patterns_feature_arc[] = { + "test_node_arc*" +}; + +static int32_t +compute_unique_user_data(const char *parent, const char *child, uint32_t interface_index) +{ + uint32_t user_data = interface_index; + + RTE_SET_USED(parent); +#define R(idx, node, node_cookie) { \ + if (!strcmp(child, node)) { \ + user_data += node_cookie; \ + } \ + } + + FOREACH_TEST_NODE_ARC +#undef R + + return user_data; +} + +static int +get_edge(struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t count, i; + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +int +common_node_init(const struct rte_graph *graph, struct rte_node *node) +{ + test_node_priv_t *priv = (test_node_priv_t *)node->ctx; + + RTE_SET_USED(graph); + + priv->node_id = node->id; + priv->feature = RTE_GRAPH_FEATURE_INVALID; + priv->arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + +#define R(idx, _name, user_data) { \ + if (!strcmp(node->name, _name)) { \ + priv->node_index = idx; \ + } \ + } + FOREACH_TEST_NODE_ARC +#undef R + + return 0; +} + +uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register source1 = { + .name = SOURCE1, + .process = source1_fn, + .flags = RTE_NODE_SOURCE_F, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {INPUT_STATIC, DUMMY1_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(source1); + +uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register input = { + .name = INPUT_STATIC, + .process = input_fn, + .feat_arc_proc = input_fa_fn, + .nb_edges = 2, + .init = common_node_init, + .next_nodes = {OUTPUT_STATIC, DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(input); + +uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register output = { + .name = OUTPUT_STATIC, + .process = output_fn, + .feat_arc_proc = output_fa_fn, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC, PKT_FREE_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(output); + +uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register pkt_free = { + .name = PKT_FREE_STATIC, + .process = pkt_free_fn, + .feat_arc_proc = pkt_free_fa_fn, + .nb_edges = 1, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(pkt_free); + +uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy1 = { + .name = DUMMY1_STATIC, + .process = dummy1_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(dummy1); + +uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy2 = { + .name = DUMMY2_STATIC, + .process = dummy2_fn, + .nb_edges = 5, + .init = common_node_init, + .next_nodes = { ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, + ARC2_FEATURE2, ARC2_FEATURE3}, +}; +RTE_NODE_REGISTER(dummy2); + +uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature1 = { + .name = ARC1_FEATURE1, + .process = arc1_feature1_fn, + .feat_arc_proc = arc1_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature1); + +uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature2 = { + .name = ARC1_FEATURE2, + .process = arc1_feature2_fn, + .feat_arc_proc = arc1_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature2); + +uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature1 = { + .name = ARC2_FEATURE1, + .process = arc2_feature1_fn, + .feat_arc_proc = arc2_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature1); + +uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature2 = { + .name = ARC2_FEATURE2, + .process = arc2_feature2_fn, + .feat_arc_proc = arc2_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature2); + +uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature3 = { + .name = ARC2_FEATURE3, + .process = arc2_feature3_fn, + .feat_arc_proc = arc2_feature3_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature3); + +static int +create_graph(void) +{ + struct rte_graph_param gconf = { + .socket_id = SOCKET_ID_ANY, + .nb_node_patterns = 1, + .node_patterns = node_patterns_feature_arc, + }; + + graph_id = rte_graph_create("worker0", &gconf); + if (graph_id == RTE_GRAPH_ID_INVALID) { + printf("Graph creation failed with error = %d\n", rte_errno); + return TEST_FAILED; + } + + return TEST_SUCCESS; +} + +static int +__test_create_feature_arc(rte_graph_feature_arc_t *arcs, int max_arcs) +{ + rte_graph_feature_arc_t arc; + const char *sample_arc_name = "sample_arc"; + char arc_name[256]; + int n_arcs; + + /* Create max number of feature arcs first */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + if (rte_graph_feature_arc_create(arc_name, MAX_FEATURES, + MAX_INDEXES, &dummy1, &arcs[n_arcs])) { + printf("Feature arc creation failed for %u\n", n_arcs); + return TEST_FAILED; + } + } + /* Verify feature arc created more than max_arcs must fail */ + if (!rte_graph_feature_arc_create("negative_test_create_arc", MAX_FEATURES, + MAX_INDEXES, &dummy2, &arc)) { + printf("Feature arc creation success for more than max configured: %u\n", n_arcs); + return TEST_FAILED; + } + /* Make sure lookup passes for all feature arcs */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + if (!rte_graph_feature_arc_lookup_by_name(arc_name, &arc)) { + if (arc != arcs[n_arcs]) { + printf("%s: Feature arc lookup mismatch for arc [%p, exp: %p]\n", + arc_name, (void *)arc, (void *)arcs[n_arcs]); + return TEST_FAILED; + } + } else { + printf("Feature arc lookup %s failed after creation\n", arc_name); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_create(void) +{ + int ret = 0, i; + + /* Create arcs with RTE_GRAPH_FEATURE_ARC_MAX */ + ret = __test_create_feature_arc(arcs, RTE_GRAPH_FEATURE_ARC_MAX); + if (ret) { + printf("Feature arc creation test failed for RTE_GRAPH_FEATURE_ARC_MAX arcs\n"); + return TEST_FAILED; + } + /* destroy all arcs via cleanup API*/ + ret = rte_graph_feature_arc_cleanup(); + if (ret) { + printf("Feature arc cleanup failed\n"); + return TEST_FAILED; + } + +#define NUM_FEAT_ARCS 128 + /* create 128 dummy feature arcs */ + ret = rte_graph_feature_arc_init(NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc init failed for NUM_FEAT_ARCS"); + return TEST_FAILED; + } + ret = __test_create_feature_arc(arcs, NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc creation test failed for NUM_FEAT_ARCS\n"); + return TEST_FAILED; + } + /* destroy all of them*/ + for (i = 0; i < NUM_FEAT_ARCS; i++) { + if (rte_graph_feature_arc_destroy(arcs[i])) { + printf("Feature arc destroy failed for %u\n", i); + return TEST_FAILED; + } + } + rte_graph_feature_arc_cleanup(); + + /* Create two arcs as per test plan */ + /* First arc start/source node is node: SOURCE1 */ + if (rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[0])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + /* Duplicate name should fail */ + if (!rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[1])) { + printf("Duplicate feature arc %s creation is not caught\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Second arc start/source node is node: OUTPUT_STATIC */ + if (rte_graph_feature_arc_create(TEST_ARC2_NAME, MAX_FEATURES, + MAX_INDEXES, &output, &arcs[1])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_features_add(void) +{ + rte_graph_feature_t temp; + + /* First feature to SOURCE1 start node -> ARC1_FEATURE1 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature1, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1); + return TEST_FAILED; + } + /* Second feature to SOURCE1 -> ARC1_FEATURE2 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature2, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + /* adding statically connected INPUT_STATIC as a last feature */ + if (rte_graph_feature_add(arcs[0], &input, ARC1_FEATURE2, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC1_NAME, INPUT_STATIC, ARC1_FEATURE2); + return TEST_FAILED; + } + /* First feature to OUTPUT_STATIC start node -> ARC2_FEATURE3 */ + if (rte_graph_feature_add(arcs[1], &arc2_feature3, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Second feature to OUTPUT_STATIC -> ARC2_FEATURE1 and before feature to + * ARC2_FEATURE3 + */ + if (rte_graph_feature_add(arcs[1], &arc2_feature1, NULL, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Add PKT_FREE node as last feature, next to arc2_feature3 */ + if (rte_graph_feature_add(arcs[1], &pkt_free, ARC2_FEATURE3, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Adding feature ARC2_FEATURE2 between ARC2_FEATURE1 and ARC2_FEATURE3. */ + if (rte_graph_feature_add(arcs[1], &arc2_feature2, ARC2_FEATURE1, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s between [%s - %s]\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Now check feature sequencing is correct for both ARCS */ + + /* arc1_featur1 must be first feature to arcs[0] */ + if (!strstr(ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc1_feature2 must be second feature to arcs[0] */ + if (!strstr(ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* Make sure INPUT_STATIC is the last feature in arcs[0] */ + temp = rte_graph_feature_arc_num_features(arcs[0]); + if (!strstr(INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC1_NAME, INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_featur1 must be first feature to arcs[1] */ + if (!strstr(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc2_feature2 must be second feature to arcs[1] */ + if (!strstr(ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_feature3 must be third feature to arcs[1] */ + if (!strstr(ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(2)))) { + printf("%s: %s is not the third feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(2))); + return TEST_FAILED; + } + + /* Make sure PKT_FREE is the last feature in arcs[1] */ + temp = rte_graph_feature_arc_num_features(arcs[1]); + if (!strstr(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + if (get_edge(&arc2_feature1, &pkt_free, NULL)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, PKT_FREE_STATIC); + return TEST_FAILED; + } + + return create_graph(); +} + +static int +test_graph_feature_arc_first_feature_enable(void) +{ + uint32_t n_indexes, n_features, count = 0; + rte_graph_feature_rt_list_t feature_list, temp = 0; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + char *feature_name = NULL; + int32_t user_data; + rte_edge_t edge = ~0; + + arc = rte_graph_feature_arc_get(arcs[0]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 0, + * On interface 1, enable feature 1 and so on so forth + * + * later verify first feature on every interface index is unique + * and check [rte_edge, user_data] retrieved via fast path APIs + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = n_indexes % 3 /* 3 features added to arc1 */; + feature_name = rte_graph_feature_arc_feature_to_name(arcs[0], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_validate(arcs[0], n_indexes, feature_name, 1, true)) { + printf("%s: Feature validate failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* negative test case. enable feature on invalid index */ + if (!n_indexes && !rte_graph_feature_enable(arcs[0], MAX_INDEXES, feature_name, + (int32_t)user_data)) { + printf("%s: Feature %s should not be enabled on invalid index\n", + TEST_ARC1_NAME, feature_name); + return TEST_FAILED; + } + if (rte_graph_feature_enable(arcs[0], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + if (temp == feature_list) { + printf("%s: Activer feature list not switched from %u -> %u\n", + TEST_ARC1_NAME, temp, feature_list); + return TEST_FAILED; + } + temp = feature_list; + if ((count + 1) != rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Number of enabled mismatches [found: %u, exp: %u]\n", + TEST_ARC1_NAME, + rte_graph_feature_arc_num_enabled_features(arcs[0]), + count + 1); + return TEST_FAILED; + } + count++; + } + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Negative test case */ + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, 1); + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC2_FEATURE1, user_data)) { + printf("%s: Invalid feature %s is enabled on index 1\n", + TEST_ARC1_NAME, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Duplicate enable */ + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC1_FEATURE2, user_data)) { + printf("%s: Duplicate feature %s shouldn't be enabled again on index 1\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC1_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc1_feature1; + else if (1 == (n_indexes % 3)) + child = &arc1_feature2; + else + child = &input; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC1_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +verify_feature_sequencing(struct rte_graph_feature_arc *arc) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_t feature; + uint32_t n_indexes; + rte_edge_t edge = ~0; + int32_t user_data; + + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: feature_list can't be obtained\n", + arc->feature_arc_name); + return TEST_FAILED; + } + /* Verify next features on interface 0 and interface 1*/ + for (n_indexes = 0; n_indexes < 2; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: 0\n", + arc->feature_arc_name); + return TEST_FAILED; + } + parent = arc->start_node; + child = rte_graph_feature_arc_feature_to_node(arcs[1], feature); + /* until fast path API reaches last feature i.e pkt_free */ + while (child != &pkt_free) { + fdata = rte_graph_feature_data_get(arc, + rte_graph_feature_get(arc, feature), + n_indexes); + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + arc->feature_arc_name, parent->name, child->name); + return TEST_FAILED; + } + user_data = compute_unique_user_data(parent->name, child->name, n_indexes); + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != user_data) { + printf("%s: Udata mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->user_data, user_data); + return TEST_FAILED; + } + + feature = fdata->next_enabled_feature; + + parent = child; + child = rte_graph_feature_arc_feature_to_node(arcs[1], + fdata->next_enabled_feature); + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_enable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + uint32_t n_indexes, n_features; + rte_graph_feature_t feature; + char *feature_name = NULL; + rte_edge_t edge = ~0; + int32_t user_data; + + arc = rte_graph_feature_arc_get(arcs[1]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[1])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 2, skip feature 1 for later + * On interface 1, enable feature 3 + * On interface 2, enable pkt_free feature + * On interface 3, continue as interface 0 + * + * later enable next feature sequence for interface 0 from feature2 -> pkt_free + * later enable next feature sequence for interface 1 from feature3 -> pkt_free + * + * also later enable feature-1 and see first feature changes for all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = (n_indexes % 3) + 1; /* feature2 to pkt_free are 3 features */ + feature_name = rte_graph_feature_arc_feature_to_name(arcs[1], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + } + /* Retrieve latest feature_list */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + /* verify first feature */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc2_feature2; + else if (1 == (n_indexes % 3)) + child = &arc2_feature3; + else + child = &pkt_free; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + /* add next_features now + * On interface 0, enable feature-3 and pkt_free + * On interface 1, enable pkt_free + * Skip interface 2 + * On interface 3, same as interface 0 + * On interface 4, same as interface 1 + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (0 == (n_indexes % 3)) { + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE3, + compute_unique_user_data(ARC2_FEATURE2, + ARC2_FEATURE3, + n_indexes))) { + printf("%s: Feature enable failed for %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + /* pkt_free on interface-0, 1, 3, 4 and so on */ + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_enable(arcs[1], n_indexes, PKT_FREE_STATIC, + compute_unique_user_data(ARC2_FEATURE3, + PKT_FREE_STATIC, + n_indexes))) { + printf("%s: Feature enable failed %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + /* Enable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE1, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: None first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature != rte_graph_feature_cast(0)) { + printf("%s: First feature mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(0)); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_first_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* Disable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE1)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE1, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature == rte_graph_feature_cast(0)) { + printf("%s: First feature not disabled on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(1)); + return TEST_FAILED; + } + if (!strncmp(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(ARC2_FEATURE1))) { + printf("%s: First feature mismatch on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + ARC2_FEATURE2); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* + * On interface 0, disable feature 2, keep feature3 and pkt_free enabled + * On interface 1, skip interface 1 where feature3 and pkt_free are enabled + * skip interface 2 as only pkt_free is enabled + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!(n_indexes % 3)) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE2)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, n_indexes); + return TEST_FAILED; + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + } + + /** + * Disable feature 3 on all interface 0 and 1 and check first feature + * is pkt_free on all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE3)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + } + /* Make sure pkt_free is first feature for all indexes */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (strncmp(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(PKT_FREE_STATIC))) { + printf("%s: %s is not first feature found on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + PKT_FREE_STATIC); + return TEST_FAILED; + } + } + + /* Disable PKT_FREE_STATIC from all indexes with no feature enabled on any interface */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, PKT_FREE_STATIC)) { + printf("%s: Feat disable failed for %s on index %u\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + /* Make sure no feature is enabled now on any interface */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: Index: %u should not have first feature enabled\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_destroy(void) +{ + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC1_NAME, &arc)) { + printf("Feature arc lookup failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (arc != arcs[0]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC1_NAME, (void *)arc, (void *)arcs[0]); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC2_NAME, &arc)) { + printf("Feature arc lookup success after destroy for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (arc != arcs[1]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC2_NAME, (void *)arc, (void *)arcs[1]); + return TEST_FAILED; + } + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +graph_feature_arc_setup(void) +{ + unsigned long i, j; + + static const struct rte_mbuf_dynfield graph_dynfield_desc = { + .name = "test_graph_dynfield", + .size = sizeof(graph_dynfield_t), + .align = alignof(graph_dynfield_t), + }; + + graph_dynfield_offset = + rte_mbuf_dynfield_register(&graph_dynfield_desc); + if (graph_dynfield_offset < 0) { + printf("Cannot register mbuf field\n"); + return TEST_FAILED; + } + RTE_SET_USED(graph_dynfield_offset); + + for (i = 0; i <= MAX_NODES; i++) { + for (j = 0; j < MBUFF_SIZE; j++) + mbuf_p[i][j] = &mbuf[i][j]; + } + + return TEST_SUCCESS; + +} + +static void +graph_feature_arc_teardown(void) +{ + if (graph_id != RTE_GRAPH_ID_INVALID) + rte_graph_destroy(graph_id); + + rte_graph_feature_arc_cleanup(); +} + +static struct unit_test_suite graph_feature_arc_testsuite = { + .suite_name = "Graph Feature arc library test suite", + .setup = graph_feature_arc_setup, + .teardown = graph_feature_arc_teardown, + .unit_test_cases = { + TEST_CASE(test_graph_feature_arc_create), + TEST_CASE(test_graph_feature_arc_features_add), + TEST_CASE(test_graph_feature_arc_first_feature_enable), + TEST_CASE(test_graph_feature_arc_next_feature_enable), + TEST_CASE(test_graph_feature_arc_first_feature_disable), + TEST_CASE(test_graph_feature_arc_next_feature_disable), + TEST_CASE(test_graph_feature_arc_destroy), + TEST_CASES_END(), /**< NULL terminate unit test array */ + }, +}; + +static int +graph_feature_arc_autotest_fn(void) +{ + return unit_test_suite_runner(&graph_feature_arc_testsuite); +} + +REGISTER_FAST_TEST(graph_feature_arc_autotest, true, true, graph_feature_arc_autotest_fn); +#endif /* !RTE_EXEC_ENV_WINDOWS */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v3 5/5] docs: add programming guide for feature arc 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena ` (3 preceding siblings ...) 2024-10-09 13:30 ` [PATCH v3 4/5] test/graph_feature_arc: add functional tests Nitin Saxena @ 2024-10-09 13:30 ` Nitin Saxena 2024-10-09 14:21 ` [PATCH v3 0/5] add feature arc in rte_graph Christophe Fontaine ` (2 subsequent siblings) 7 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-09 13:30 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Updated graph library guide with feature arc Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/prog_guide/graph_lib.rst | 288 ++++++++++++++++++++ doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes 4 files changed, 288 insertions(+) create mode 100644 doc/guides/prog_guide/img/feature_arc-1.png create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png diff --git a/doc/guides/prog_guide/graph_lib.rst b/doc/guides/prog_guide/graph_lib.rst index ad09bdfe26..fc231ace99 100644 --- a/doc/guides/prog_guide/graph_lib.rst +++ b/doc/guides/prog_guide/graph_lib.rst @@ -547,3 +547,291 @@ on success packet is enqueued to ``udp4_input`` node. Hash lookup is performed in ``udp4_input`` node with registered destination port and destination port in UDP packet , on success packet is handed to ``udp_user_node``. + +Feature Arc +----------- +`Feature arc` represents an ordered list of `protocols/features` at a given +networking layer. It is a high level abstraction to connect various `feature` +nodes in `rte_graph` instance and allows seamless packets steering based on the +sequence of enabled features at runtime on each interface. + +`Features` (or feature nodes) are nodes which handle partial or complete +protocol processing in a given direction. For instance, `ipv4-rewrite` and +`IPv4 IPsec encryption` are outbound features while `ipv4-lookup` and `IPv4 +IPsec decryption` are inbound features. Further, `ipv4-rewrite` and `IPv4 +IPsec encryption` can collectively represent a `feature arc` towards egress +direction with ordering constraints that `IPv4 IPsec encryption` must be +performed before `ipv4-rewrite`. Similarly, `IPv4 IPsec decryption` and +`ipv4-lookup` can represent a `feature arc` in an ingress direction. Both of +these `feature arc` can co-exist at an IPv4 layer in egress and ingress +direction respectively. + +A `feature` can be represented by a single node or collection of multiple nodes +performing feature processing collectively. + +.. figure:: img/feature_arc-1.png + :alt: feature-arc-1 + :width: 350px + :align: center + + Feature Arc overview + +Each `feature arc` is associated with a `Start` node from which all features in +a feature arc are connected. A `start` node itself is not a `feature` node but +it is where `first enabled feature` is checked in fast path. In above figure, +`Node-A` represents a `start node`. There may be a `Sink` node as well which +is child node for every feature in an arc. 'Sink` node is responsible of +consuming those packets which are not consumed by intermediate enabled features +between `start` and `sink` node. `Sink` node, if present, is the last enabled +feature in a feature arc. A `feature` node statically connected to `start` node +must also be added via feature arc API, `rte_graph_feature_add()``. Here `Node-B` +acts as a `sink` node which is statically linked to `Node A`. `Feature` nodes +are connected via `rte_graph_feature_add()` which takes care of connecting +all `feature` nodes with each other and start node. + +.. code-block:: bash + :linenos: + :emphasize-lines: 8 + :caption: Node-B statically linked to Node-A + + static struct rte_node_register node_A_node = { + .process = node_A_process_func, + ... + ... + .name = "Node-A", + .next_nodes = + { + [0] = "Node-B", + }, + .nb_edges = 1, + }; + +When multiple features are enabled on an interface, it may be required to steer +packets across `features` in a given order. For instance, if `Feature 1` and +`Feature 2` both are enabled on an interface ``X``, it may be required to send +packets to `Feature-1` before `Feature-2`. Such ordering constraints can be +easily expressed with `feature arc`. In this case, `Feature 1` is called as +``First Feature`` and `Feature 2` is called as ``Next Feature`` to `Feature 1`. + +.. figure:: img/feature_arc-2.png + :alt: feature-arc-2 + :width: 600px + :align: center + + First and Next features and their ordering + +In similar manner, even application specific ``custom features`` can be hooked +to standard nodes. It is to be noted that this `custom feature` hooking to +`feature arc` aware node does not require any code changes. + +It may be obvious by now that `features` enabled on one interface does not +affect packets on other interfaces. In above example, if no feature is +enabled on an interface ``X``, packets destined to interface ``X`` would be +directly sent to `Node-B` from `Node-A`. + +.. figure:: img/feature_arc-3.png + :alt: feature-arc-3 + :width: 550px + :align: center + + Feature-2 consumed/non-consumed packet path + +When a `Feature-X` node receives packets via feature arc, it may decide whether +to ``consume packet`` or send to `next enabled feature`. A node can consume +packet by freeing it, sending it on wire or enqueuing it to hardware queue. If a +packet is not consumed by a `Feature-X` node, it may send to `next enabled +feature` on an interface. In above figure, `Feature-2` nodes are represented to +consume packets. Classic example for a node performing consume and non-consume +operation on packets would be IPsec policy node where all packets with +``protect`` actions are consumed while remaining packets with ``bypass`` +actions are sent to next enabled feature. + +In fast path feature node may require to lookup local data structures for each +interface. For example, retrieving policy database per interface for IPsec +processing. ``rte_graph_feature_enable`` API allows to set application +specific cookie per feature per interface. `Feature data` object maintains this +cookie in fast path for each interface. + +`Feature arc design` allows to enable subsequent features in a control plane +without stopping workers which are accessing feature arc's fast path APIs in +``rte_graph_walk()`` context. However for disabling features require RCU like +scheme for synchronization. + +Programming model +~~~~~~~~~~~~~~~~~ +Feature Arc Objects +^^^^^^^^^^^^^^^^^^^ +Control plane and fast path APIs deals with following objects: + +Feature arc +*********** +``rte_graph_feature_arc_t`` is a handle to feature arc which is created via +``rte_graph_feature_arc_create()``. It is a `uint64_t` size object which can be +saved in feature node's context. This object can be translated to fast path +feature arc object ``struct rte_graph_feature_arc`` which is an input +argument to all fast path APIs. Control plane APIs majorly takes +`rte_graph_feature_arc_t` object as an input. + +Feature List +************ +Each feature arc holds two feature lists: `active` and `passive`. While worker +cores uses `active` list, control plane APIs uses `passive` list for +enabling/disabling a feature on any interface with in a arc. After successful +feature enable/disable, ``rte_graph_feature_enable()``/ +``rte_graph_feature_disable()`` atomically switches passive list to active list +and vice-versa. Most of the fast path APIs takes active list as an argument +(``rte_graph_feature_rt_list_t``), which feature node can obtain in start of +it's `process_func()` via ``rte_graph_feature_arc_has_any_feature()`` (in `start` +node) or ``rte_graph_feature_arc_has_feature()`` (in next feature nodes). + +Each feature list holds RTE_GRAPH_MAX_FEATURES number of features and +associated feature data for every interface index + +Feature +******** +Feature is a data structure which holds `feature data` object for every +interface. It is represented via ``rte_graph_feature_t`` which is a `uint8_t` +size object. Fast path internal structure ``struct rte_graph_feature`` can be +obtained from ``rte_graph_feature_t`` via ``rte_graph_feature_get()`` API. + +In `start` node `rte_graph_feature_arc_first_feature_get()` can be used to get +first enabled `rte_graph_feature_t` object for an interface. `rte_edge` from +`start` node to first enabled feature is provided by +``rte_graph_feature_arc_feature_set()`` API. + +In `feature nodes`, next enabled feature is obtained by providing current feature +as an input to ``rte_graph_feature_arc_next_feature_get()`` API. + +Feature data +************ +Feature data object is maintained per feature per interface which holds +following information in fast path + +- ``rte_edge_t`` to send packet to next enabled feature +- ``Next enabled feature`` on current interface +- ``User_data`` per feature per interface set by application via `rte_graph_feature_enable()` + +Enabling Feature Arc processing +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +By default, feature arc processing is disabled in `rte_graph_create()`. To +enable feature arc processing in fast path, `rte_graph_create()` API should be +invoked with `feature_arc_enable` flag set as `true` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Enabling feature are processing in rte_graph_create() + + struct rte_graph_param graph_conf; + + graph_conf.feature_arc_enable = true; + struct rte_graph *graph = rte_graph_create("graph_name", &graph_conf); + +Further as an optimization technique, `rte_graph_walk()` would call newly added +``feat_arc_proc()`` node callback function (if non-NULL) instead of +``preocess_func`` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Feature arc specific node callback function + + static struct rte_node_register ip4_rewrite_node = { + .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, + .name = "ip4_rewrite", + ... + ... + }; + +If `feat_arc_proc` is not provided in node registration, `process_func` would +be called by `rte_graph_walk()` + +Sample Usage +^^^^^^^^^^^^ +.. code-block:: bash + :linenos: + :caption: Feature arc sample usage + + #define MAX_FEATURES 10 + #define MAX_INDEXES 5 + + static uint16_t + feature2_feature_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* features may be enabled */ + } + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc is disabled in rte_graph_create() */ + } + + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc may be enabled or disabled as this process_func() would + * be called for the case when feature arc is enabled in rte_graph_create() + * and also the case when it is disabled + */ + } + + static struct rte_node_register feature2_node = { + .process = feature2_node_process, + .feat_arc_proc = feature2_feature_node_process, + .name = "feature2", + .init = feature2_init_func, + ... + ... + }; + + static struct rte_node_register feature1_node = { + .process = feature1_node_process, + .feat_arc_proc = NULL, + .name = "feature1", + ... + ... + }; + + int worker_cb(void *_em) + { + rte_graph_feature_arc_t arc; + uint32_t user_data; + + rte_graph_feature_arc_lookup_by_name("sample_arc", &arc); + + /* From control thread context (like CLII): + * Enable feature 2 on interface index 4 + */ + if (rte_lcore_id() == rte_get_main_lcore) { + user_data = 0x1234; + rte_graph_feature_enable(arc, 4 /* interface index */, "feature2", user_data); + } else { + while(1) + rte_graph_walk); + } + } + + int main(void) + { + struct rte_graph_param graph_conf; + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_create("sample_arc", MAX_FEATURES, MAX_INDEXES, &arc)) + return -1; + + rte_graph_feature_add(arc, "feature1", NULL, NULL); + rte_graph_feature_add(arc, "feature2", "feature1" /* add feature2 after feature 1*/, NULL); + + /* create graph*/ + ... + ... + graph_conf.feature_arc_enable = true; + + struct rte_graph *graph = rte_graph_create("sample_graph", &graph_conf); + + rte_eal_mp_remote_launch(worker_cb, arg, CALL_MAIN); + } diff --git a/doc/guides/prog_guide/img/feature_arc-1.png b/doc/guides/prog_guide/img/feature_arc-1.png new file mode 100644 index 0000000000000000000000000000000000000000..d518000f42753ea7a1a71668c2735c48fc75c3ed GIT binary patch literal 61532 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 literal 0 HcmV?d00001 diff --git a/doc/guides/prog_guide/img/feature_arc-2.png b/doc/guides/prog_guide/img/feature_arc-2.png new file mode 100644 index 0000000000000000000000000000000000000000..604a67229b29a9c8ace59d28a7324e02c2bc4a89 GIT binary patch literal 155806 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�!fv6yo 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ΐMNRyBlZP4r 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 VhL3#7$M&;o^tFw(Dm5IV{|}tHa(Vy& literal 0 HcmV?d00001 diff --git a/doc/guides/prog_guide/img/feature_arc-3.png b/doc/guides/prog_guide/img/feature_arc-3.png new file mode 100644 index 0000000000000000000000000000000000000000..3991e0aed2933bfa01e9ffad654f4c40a639bc53 GIT binary patch literal 143697 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 JynBzl{|6wHR9pZ6 literal 0 HcmV?d00001 -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [PATCH v3 0/5] add feature arc in rte_graph 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena ` (4 preceding siblings ...) 2024-10-09 13:30 ` [PATCH v3 5/5] docs: add programming guide for feature arc Nitin Saxena @ 2024-10-09 14:21 ` Christophe Fontaine 2024-10-10 4:13 ` [EXTERNAL] " Nitin Saxena 2024-10-09 17:37 ` Stephen Hemminger 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena 7 siblings, 1 reply; 55+ messages in thread From: Christophe Fontaine @ 2024-10-09 14:21 UTC (permalink / raw) To: Nitin Saxena Cc: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, dev, Nitin Saxena > On 9 Oct 2024, at 15:29, Nitin Saxena <nsaxena@marvell.com> wrote: > > Feature arc represents an ordered list of features/protocols at a given > networking layer. It is a high level abstraction to connect various > rte_graph nodes, as feature nodes, and allow packets steering across > these nodes in a generic manner. > > Features (or feature nodes) are nodes which handles partial or complete > handling of a protocol in fast path. Like ipv4-rewrite node, which adds > rewrite data to an outgoing IPv4 packet. > > However in above example, outgoing interface(say "eth0") may have > outbound IPsec policy enabled, hence packets must be steered from > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > policy lookup. On the other hand, packets routed to another interface > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > is disabled on eth1. Feature-arc allows rte_graph applications to manage > such constraints easily > > Feature arc abstraction allows rte_graph based application to > > 1. Seamlessly steer packets across feature nodes based on whether > feature is enabled or disabled on an interface. Features enabled on one > interface may not be enabled on another interface with in a same feature > arc. > > 2. Allow enabling/disabling of features on an interface at runtime, > so that if a feature is disabled, packets associated with that interface > won't be steered to corresponding feature node. > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > node and allow packet steering from feature node to custom node without > changing former's fast path function Hi, As a general comment, I would have expected a modification on rte_graph_walk so the nodes would *not* need to be aware of the enabled feature arcs. Yet, in the series, the nodes need to be aware if a feature arc is enabled or not. Isn’t this a contradiction or did I miss something ? Christophe > > 4. Allow expressing features in a particular sequential order so that > packets are steered in an ordered way across nodes in fast path. For > eg: if IPsec and IPv4 features are enabled on an ingress interface, > packets must be sent to IPsec inbound policy node first and then to ipv4 > lookup node. > > This patch series adds feature arc library in rte_graph and also adds > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > Changes in v3: > - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix > compilation on 32-bit machine > - Updated images in .png format > - Added ABI change section in release notes > - Fixed DPDK CI failures > > Changes in v2: > - Added unit tests for feature arc > - Fixed issues found in testing > - Added new public APIs rte_graph_feature_arc_feature_to_node(), > rte_graph_feature_arc_feature_to_name(), > rte_graph_feature_arc_num_features() > - Added programming guide for feature arc > - Added release notes for feature arc > > Nitin Saxena (5): > graph: add feature arc support > graph: add feature arc option in graph create > graph: add IPv4 output feature arc > test/graph_feature_arc: add functional tests > docs: add programming guide for feature arc > > app/test/meson.build | 1 + > app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ > doc/guides/prog_guide/graph_lib.rst | 288 ++++ > doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes > doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes > doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes > doc/guides/rel_notes/release_24_11.rst | 13 + > lib/graph/graph.c | 1 + > lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++ > lib/graph/graph_populate.c | 7 +- > lib/graph/graph_private.h | 3 + > lib/graph/meson.build | 2 + > lib/graph/node.c | 2 + > lib/graph/rte_graph.h | 3 + > lib/graph/rte_graph_feature_arc.h | 431 ++++++ > lib/graph/rte_graph_feature_arc_worker.h | 674 +++++++++ > lib/graph/version.map | 20 + > lib/node/ip4_rewrite.c | 476 +++++-- > lib/node/ip4_rewrite_priv.h | 15 +- > lib/node/node_private.h | 20 +- > lib/node/rte_node_ip4_api.h | 3 + > 21 files changed, 4494 insertions(+), 98 deletions(-) > create mode 100644 app/test/test_graph_feature_arc.c > create mode 100644 doc/guides/prog_guide/img/feature_arc-1.png > create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png > create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png > create mode 100644 lib/graph/graph_feature_arc.c > create mode 100644 lib/graph/rte_graph_feature_arc.h > create mode 100644 lib/graph/rte_graph_feature_arc_worker.h > > -- > 2.43.0 > ^ permalink raw reply [flat|nested] 55+ messages in thread
* RE: [EXTERNAL] Re: [PATCH v3 0/5] add feature arc in rte_graph 2024-10-09 14:21 ` [PATCH v3 0/5] add feature arc in rte_graph Christophe Fontaine @ 2024-10-10 4:13 ` Nitin Saxena 0 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 4:13 UTC (permalink / raw) To: Christophe Fontaine Cc: Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, Robin Jarry, dev, Nitin Saxena Hi Christophe, Thanks for the review. See my comments inline Thanks, Nitin > -----Original Message----- > From: Christophe Fontaine <cfontain@redhat.com> > Sent: Wednesday, October 9, 2024 7:51 PM > To: Nitin Saxena <nsaxena@marvell.com> > Cc: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda > <kirankumark@marvell.com>; Nithin Kumar Dabilpuram > <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com>; Robin > Jarry <rjarry@redhat.com>; dev@dpdk.org; Nitin Saxena > <nsaxena16@gmail.com> > Subject: [EXTERNAL] Re: [PATCH v3 0/5] add feature arc in rte_graph > > > On 9 Oct 2024, at 15:29, Nitin Saxena <nsaxena@marvell.com> wrote: > > > > Feature arc represents an ordered list of features/protocols at a > > given networking layer. It is a high level abstraction to connect > > various rte_graph nodes, as feature nodes, and allow packets steering > > across these nodes in a generic manner. > > > > Features (or feature nodes) are nodes which handles partial or > > complete handling of a protocol in fast path. Like ipv4-rewrite node, > > which adds rewrite data to an outgoing IPv4 packet. > > > > However in above example, outgoing interface(say "eth0") may have > > outbound IPsec policy enabled, hence packets must be steered from > > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > > policy lookup. On the other hand, packets routed to another interface > > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > > is disabled on eth1. Feature-arc allows rte_graph applications to > > manage such constraints easily > > > > Feature arc abstraction allows rte_graph based application to > > > > 1. Seamlessly steer packets across feature nodes based on whether > > feature is enabled or disabled on an interface. Features enabled on > > one interface may not be enabled on another interface with in a same > > feature arc. > > > > 2. Allow enabling/disabling of features on an interface at runtime, so > > that if a feature is disabled, packets associated with that interface > > won't be steered to corresponding feature node. > > > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > > node and allow packet steering from feature node to custom node > > without changing former's fast path function > > Hi, > > As a general comment, I would have expected a modification on > rte_graph_walk so the nodes would *not* need to be aware of the enabled > feature arcs. If your point is that nodes has to adopt rte_feature_xxx() APIs in its "process_func()" to take advantage of feature arc, then your assessment is true. So it is true that each feature node knows on which *arc* it sits on. However *unaware* part comes from the fact that once a node is integrated with feature_arc APIs, then the node does not need to know what is the next node to which current mbuf needs to be sent. So unaware functionality comes after integrating feature arc APIs. Regarding your rte_graph_walk() comment: IMO, decision of determining next_edge ( or next node) lies with each *node* and not with rte_graph_walk(). Feature arc is extending the functionality of determining next_edge to control plane as well (as control plane enables/disable features). This way without changing feature node code, control plane is able to steer packets to different nodes. This is the major functionality feature arc is trying to simplify > Yet, in the series, the nodes need to be aware if a feature arc is enabled or > not. > Isn’t this a contradiction or did I miss something ? > > Christophe > > > > > 4. Allow expressing features in a particular sequential order so that > > packets are steered in an ordered way across nodes in fast path. For > > eg: if IPsec and IPv4 features are enabled on an ingress interface, > > packets must be sent to IPsec inbound policy node first and then to > > ipv4 lookup node. > > > > This patch series adds feature arc library in rte_graph and also adds > > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > > > Changes in v3: > > - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix > > compilation on 32-bit machine > > - Updated images in .png format > > - Added ABI change section in release notes > > - Fixed DPDK CI failures > > > > Changes in v2: > > - Added unit tests for feature arc > > - Fixed issues found in testing > > - Added new public APIs rte_graph_feature_arc_feature_to_node(), > > rte_graph_feature_arc_feature_to_name(), > > rte_graph_feature_arc_num_features() > > - Added programming guide for feature arc > > - Added release notes for feature arc > > > > Nitin Saxena (5): > > graph: add feature arc support > > graph: add feature arc option in graph create > > graph: add IPv4 output feature arc > > test/graph_feature_arc: add functional tests > > docs: add programming guide for feature arc > > > > app/test/meson.build | 1 + > > app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ > > doc/guides/prog_guide/graph_lib.rst | 288 ++++ > > doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes > > doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes > > doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes > > doc/guides/rel_notes/release_24_11.rst | 13 + > > lib/graph/graph.c | 1 + > > lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++ > > lib/graph/graph_populate.c | 7 +- > > lib/graph/graph_private.h | 3 + > > lib/graph/meson.build | 2 + > > lib/graph/node.c | 2 + > > lib/graph/rte_graph.h | 3 + > > lib/graph/rte_graph_feature_arc.h | 431 ++++++ > > lib/graph/rte_graph_feature_arc_worker.h | 674 +++++++++ > > lib/graph/version.map | 20 + > > lib/node/ip4_rewrite.c | 476 +++++-- > > lib/node/ip4_rewrite_priv.h | 15 +- > > lib/node/node_private.h | 20 +- > > lib/node/rte_node_ip4_api.h | 3 + > > 21 files changed, 4494 insertions(+), 98 deletions(-) create mode > > 100644 app/test/test_graph_feature_arc.c create mode 100644 > > doc/guides/prog_guide/img/feature_arc-1.png > > create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png > > create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png > > create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 > > lib/graph/rte_graph_feature_arc.h create mode 100644 > > lib/graph/rte_graph_feature_arc_worker.h > > > > -- > > 2.43.0 > > ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [PATCH v3 0/5] add feature arc in rte_graph 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena ` (5 preceding siblings ...) 2024-10-09 14:21 ` [PATCH v3 0/5] add feature arc in rte_graph Christophe Fontaine @ 2024-10-09 17:37 ` Stephen Hemminger 2024-10-10 4:24 ` [EXTERNAL] " Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena 7 siblings, 1 reply; 55+ messages in thread From: Stephen Hemminger @ 2024-10-09 17:37 UTC (permalink / raw) To: Nitin Saxena Cc: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine, dev, Nitin Saxena On Wed, 9 Oct 2024 18:59:57 +0530 Nitin Saxena <nsaxena@marvell.com> wrote: > Feature arc represents an ordered list of features/protocols at a given > networking layer. It is a high level abstraction to connect various > rte_graph nodes, as feature nodes, and allow packets steering across > these nodes in a generic manner. > > Features (or feature nodes) are nodes which handles partial or complete > handling of a protocol in fast path. Like ipv4-rewrite node, which adds > rewrite data to an outgoing IPv4 packet. > > However in above example, outgoing interface(say "eth0") may have > outbound IPsec policy enabled, hence packets must be steered from > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > policy lookup. On the other hand, packets routed to another interface > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > is disabled on eth1. Feature-arc allows rte_graph applications to manage > such constraints easily > > Feature arc abstraction allows rte_graph based application to > > 1. Seamlessly steer packets across feature nodes based on whether > feature is enabled or disabled on an interface. Features enabled on one > interface may not be enabled on another interface with in a same feature > arc. > > 2. Allow enabling/disabling of features on an interface at runtime, > so that if a feature is disabled, packets associated with that interface > won't be steered to corresponding feature node. > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > node and allow packet steering from feature node to custom node without > changing former's fast path function > > 4. Allow expressing features in a particular sequential order so that > packets are steered in an ordered way across nodes in fast path. For > eg: if IPsec and IPv4 features are enabled on an ingress interface, > packets must be sent to IPsec inbound policy node first and then to ipv4 > lookup node. > > This patch series adds feature arc library in rte_graph and also adds > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > Changes in v3: > - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix > compilation on 32-bit machine > - Updated images in .png format > - Added ABI change section in release notes > - Fixed DPDK CI failures > > Changes in v2: > - Added unit tests for feature arc > - Fixed issues found in testing > - Added new public APIs rte_graph_feature_arc_feature_to_node(), > rte_graph_feature_arc_feature_to_name(), > rte_graph_feature_arc_num_features() > - Added programming guide for feature arc > - Added release notes for feature arc > > Nitin Saxena (5): > graph: add feature arc support > graph: add feature arc option in graph create > graph: add IPv4 output feature arc > test/graph_feature_arc: add functional tests > docs: add programming guide for feature arc > > app/test/meson.build | 1 + > app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ > doc/guides/prog_guide/graph_lib.rst | 288 ++++ > doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes > doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes > doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes > doc/guides/rel_notes/release_24_11.rst | 13 + > lib/graph/graph.c | 1 + > lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++ > lib/graph/graph_populate.c | 7 +- > lib/graph/graph_private.h | 3 + > lib/graph/meson.build | 2 + > lib/graph/node.c | 2 + > lib/graph/rte_graph.h | 3 + > lib/graph/rte_graph_feature_arc.h | 431 ++++++ > lib/graph/rte_graph_feature_arc_worker.h | 674 +++++++++ > lib/graph/version.map | 20 + > lib/node/ip4_rewrite.c | 476 +++++-- > lib/node/ip4_rewrite_priv.h | 15 +- > lib/node/node_private.h | 20 +- > lib/node/rte_node_ip4_api.h | 3 + > 21 files changed, 4494 insertions(+), 98 deletions(-) > create mode 100644 app/test/test_graph_feature_arc.c > create mode 100644 doc/guides/prog_guide/img/feature_arc-1.png > create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png > create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png > create mode 100644 lib/graph/graph_feature_arc.c > create mode 100644 lib/graph/rte_graph_feature_arc.h > create mode 100644 lib/graph/rte_graph_feature_arc_worker.h > Looks good, but likely missing an RTE_ATOMIC() around the feature enable bitmask. Build fails: #################################################################################### #### [Begin job log] "ubuntu-22.04-clang-stdatomic" at step Build and test #################################################################################### rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], bitmask, ^ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ../lib/eal/include/rte_stdatomic.h:76:2: note: expanded from macro 'rte_atomic_store_explicit' atomic_store_explicit(ptr, val, memorder) ^ ~~~ /usr/lib/llvm-14/lib/clang/14.0.0/include/stdatomic.h:127:31: note: expanded from macro 'atomic_store_explicit' #define atomic_store_explicit __c11_atomic_store ^ ../lib/graph/graph_feature_arc.c:1084:2: error: address argument to atomic operation must be a pointer to _Atomic type ('rte_graph_feature_rt_list_t *' (aka 'unsigned short *') invalid) rte_atomic_store_explicit(&arc->active_feature_list, passive_list, ^ ~~~~~~~~~~~~~~~~~~~~~~~~~ ../lib/eal/include/rte_stdatomic.h:76:2: note: expanded from macro 'rte_atomic_store_explicit' atomic_store_explicit(ptr, val, memorder) ^ ~~~ /usr/lib/llvm-14/lib/clang/14.0.0/include/stdatomic.h:127:31: note: expanded from macro 'atomic_store_explicit' #define atomic_store_explicit __c11_atomic_store ^ 10 errors generated. ^ permalink raw reply [flat|nested] 55+ messages in thread
* RE: [EXTERNAL] Re: [PATCH v3 0/5] add feature arc in rte_graph 2024-10-09 17:37 ` Stephen Hemminger @ 2024-10-10 4:24 ` Nitin Saxena 0 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 4:24 UTC (permalink / raw) To: Stephen Hemminger Cc: Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine, dev, Nitin Saxena Thanks Stephen. Will fix compilation in next version. Thanks, Nitin > -----Original Message----- > From: Stephen Hemminger <stephen@networkplumber.org> > Sent: Wednesday, October 9, 2024 11:08 PM > To: Nitin Saxena <nsaxena@marvell.com> > Cc: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda > <kirankumark@marvell.com>; Nithin Kumar Dabilpuram > <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com>; Robin > Jarry <rjarry@redhat.com>; Christophe Fontaine <cfontain@redhat.com>; > dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com> > Subject: [EXTERNAL] Re: [PATCH v3 0/5] add feature arc in rte_graph > > On Wed, 9 Oct 2024 18: 59: 57 +0530 Nitin Saxena <nsaxena@ marvell. com> > wrote: > Feature arc represents an ordered list of features/protocols at a given > > networking layer. It is a high level abstraction to connect various > rte_graph > > On Wed, 9 Oct 2024 18:59:57 +0530 > Nitin Saxena <nsaxena@marvell.com> wrote: > > > Feature arc represents an ordered list of features/protocols at a > > given networking layer. It is a high level abstraction to connect > > various rte_graph nodes, as feature nodes, and allow packets steering > > across these nodes in a generic manner. > > > > Features (or feature nodes) are nodes which handles partial or > > complete handling of a protocol in fast path. Like ipv4-rewrite node, > > which adds rewrite data to an outgoing IPv4 packet. > > > > However in above example, outgoing interface(say "eth0") may have > > outbound IPsec policy enabled, hence packets must be steered from > > ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec > > policy lookup. On the other hand, packets routed to another interface > > (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature > > is disabled on eth1. Feature-arc allows rte_graph applications to > > manage such constraints easily > > > > Feature arc abstraction allows rte_graph based application to > > > > 1. Seamlessly steer packets across feature nodes based on whether > > feature is enabled or disabled on an interface. Features enabled on > > one interface may not be enabled on another interface with in a same > > feature arc. > > > > 2. Allow enabling/disabling of features on an interface at runtime, so > > that if a feature is disabled, packets associated with that interface > > won't be steered to corresponding feature node. > > > > 3. Provides mechanism to hook custom/user-defined nodes to a feature > > node and allow packet steering from feature node to custom node > > without changing former's fast path function > > > > 4. Allow expressing features in a particular sequential order so that > > packets are steered in an ordered way across nodes in fast path. For > > eg: if IPsec and IPv4 features are enabled on an ingress interface, > > packets must be sent to IPsec inbound policy node first and then to > > ipv4 lookup node. > > > > This patch series adds feature arc library in rte_graph and also adds > > "ipv4-output" feature arc handling in "ipv4-rewrite" node. > > > > Changes in v3: > > - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix > > compilation on 32-bit machine > > - Updated images in .png format > > - Added ABI change section in release notes > > - Fixed DPDK CI failures > > > > Changes in v2: > > - Added unit tests for feature arc > > - Fixed issues found in testing > > - Added new public APIs rte_graph_feature_arc_feature_to_node(), > > rte_graph_feature_arc_feature_to_name(), > > rte_graph_feature_arc_num_features() > > - Added programming guide for feature arc > > - Added release notes for feature arc > > > > Nitin Saxena (5): > > graph: add feature arc support > > graph: add feature arc option in graph create > > graph: add IPv4 output feature arc > > test/graph_feature_arc: add functional tests > > docs: add programming guide for feature arc > > > > app/test/meson.build | 1 + > > app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ > > doc/guides/prog_guide/graph_lib.rst | 288 ++++ > > doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes > > doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes > > doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes > > doc/guides/rel_notes/release_24_11.rst | 13 + > > lib/graph/graph.c | 1 + > > lib/graph/graph_feature_arc.c | 1223 ++++++++++++++++ > > lib/graph/graph_populate.c | 7 +- > > lib/graph/graph_private.h | 3 + > > lib/graph/meson.build | 2 + > > lib/graph/node.c | 2 + > > lib/graph/rte_graph.h | 3 + > > lib/graph/rte_graph_feature_arc.h | 431 ++++++ > > lib/graph/rte_graph_feature_arc_worker.h | 674 +++++++++ > > lib/graph/version.map | 20 + > > lib/node/ip4_rewrite.c | 476 +++++-- > > lib/node/ip4_rewrite_priv.h | 15 +- > > lib/node/node_private.h | 20 +- > > lib/node/rte_node_ip4_api.h | 3 + > > 21 files changed, 4494 insertions(+), 98 deletions(-) create mode > > 100644 app/test/test_graph_feature_arc.c create mode 100644 > > doc/guides/prog_guide/img/feature_arc-1.png > > create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png > > create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png > > create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 > > lib/graph/rte_graph_feature_arc.h create mode 100644 > > lib/graph/rte_graph_feature_arc_worker.h > > > > Looks good, but likely missing an RTE_ATOMIC() around the feature enable > bitmask. > Build fails: > > ################################################################ > #################### > #### [Begin job log] "ubuntu-22.04-clang-stdatomic" at step Build and test > ################################################################ > #################### > rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], > bitmask, > ^ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ > ../lib/eal/include/rte_stdatomic.h:76:2: note: expanded from macro > 'rte_atomic_store_explicit' > atomic_store_explicit(ptr, val, memorder) > ^ ~~~ > /usr/lib/llvm-14/lib/clang/14.0.0/include/stdatomic.h:127:31: note: expanded > from macro 'atomic_store_explicit' > #define atomic_store_explicit __c11_atomic_store > ^ > ../lib/graph/graph_feature_arc.c:1084:2: error: address argument to atomic > operation must be a pointer to _Atomic type ('rte_graph_feature_rt_list_t *' > (aka 'unsigned short *') invalid) > rte_atomic_store_explicit(&arc->active_feature_list, passive_list, > ^ ~~~~~~~~~~~~~~~~~~~~~~~~~ > ../lib/eal/include/rte_stdatomic.h:76:2: note: expanded from macro > 'rte_atomic_store_explicit' > atomic_store_explicit(ptr, val, memorder) > ^ ~~~ > /usr/lib/llvm-14/lib/clang/14.0.0/include/stdatomic.h:127:31: note: expanded > from macro 'atomic_store_explicit' > #define atomic_store_explicit __c11_atomic_store > ^ > 10 errors generated. ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v4 0/5] add feature arc in rte_graph 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena ` (6 preceding siblings ...) 2024-10-09 17:37 ` Stephen Hemminger @ 2024-10-10 13:31 ` Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 1/5] graph: add feature arc support Nitin Saxena ` (5 more replies) 7 siblings, 6 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 13:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on whether feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Changes in v4: - Fixed clang build compilations - Captured `feat_arc_proc` function in ABI change section of release notes Changes in v3: - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix compilation on 32-bit machine - Updated images in .png format - Added ABI change section in release notes - Fixed DPDK CI failures Changes in v2: - Added unit tests for feature arc - Fixed issues found in testing - Added new public APIs rte_graph_feature_arc_feature_to_node(), rte_graph_feature_arc_feature_to_name(), rte_graph_feature_arc_num_features() - Added programming guide for feature arc - Added release notes for feature arc Nitin Saxena (5): graph: add feature arc support graph: add feature arc option in graph create graph: add IPv4 output feature arc test/graph_feature_arc: add functional tests docs: add programming guide for feature arc app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ doc/guides/prog_guide/graph_lib.rst | 288 ++++ doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes doc/guides/rel_notes/release_24_11.rst | 17 + lib/graph/graph.c | 1 + lib/graph/graph_feature_arc.c | 1236 ++++++++++++++++ lib/graph/graph_populate.c | 7 +- lib/graph/graph_private.h | 3 + lib/graph/meson.build | 2 + lib/graph/node.c | 2 + lib/graph/rte_graph.h | 3 + lib/graph/rte_graph_feature_arc.h | 431 ++++++ lib/graph/rte_graph_feature_arc_worker.h | 679 +++++++++ lib/graph/version.map | 20 + lib/node/ip4_rewrite.c | 476 +++++-- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 21 files changed, 4516 insertions(+), 98 deletions(-) create mode 100644 app/test/test_graph_feature_arc.c create mode 100644 doc/guides/prog_guide/img/feature_arc-1.png create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v4 1/5] graph: add feature arc support 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena @ 2024-10-10 13:31 ` Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 2/5] graph: add feature arc option in graph create Nitin Saxena ` (4 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 13:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena add feature arc to allow dynamic steering of packets across graph nodes based on protocol features enabled on incoming or outgoing interface Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 10 + lib/graph/graph_feature_arc.c | 1236 ++++++++++++++++++++++ lib/graph/meson.build | 2 + lib/graph/rte_graph_feature_arc.h | 431 ++++++++ lib/graph/rte_graph_feature_arc_worker.h | 679 ++++++++++++ lib/graph/version.map | 20 + 6 files changed, 2378 insertions(+) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index 2f78f2d125..bd5589b01c 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -82,6 +82,16 @@ New Features The new statistics are useful for debugging and profiling. +* **Added feature arc abstraction in graph library.** + + Feature arc abstraction helps ``rte_graph`` based applications to steer + packets across different node path(s) based on the features (or protocols) + enabled on interfaces. Different feature node paths can be enabled/disabled + at runtime on some or on all interfaces. This abstraction also help + applications to hook their ``custom nodes`` in standard DPDK node paths + without any code changes in the later. + + * Added ``ip4-output`` feature arc processing in ``ip4_rewrite`` node. Removed Items ------------- diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c new file mode 100644 index 0000000000..0f8633c317 --- /dev/null +++ b/lib/graph/graph_feature_arc.c @@ -0,0 +1,1236 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "graph_private.h" +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> + +#define ARC_PASSIVE_LIST(list) (list ^ 0x1) + +#define rte_graph_uint_cast(x) ((unsigned int)x) +#define feat_dbg graph_dbg + +static rte_graph_feature_arc_main_t *__rte_graph_feature_arc_main; + +/* Make sure fast path cache line is compact */ +_Static_assert((offsetof(struct rte_graph_feature_arc, slow_path_variables) + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) + <= RTE_CACHE_LINE_SIZE, + "Fast path feature arc variables exceed cache line size"); + +#define connect_graph_nodes(node1, node2, edge, arc_name) \ + __connect_graph_nodes(node1, node2, edge, arc_name, __LINE__) + +#define FEAT_COND_ERR(cond, fmt, ...) \ + do { \ + if (cond) \ + graph_err(fmt, ##__VA_ARGS__); \ + } while (0) + +/* + * lookup feature name and get control path node_list as well as feature index + * at which it is inserted + */ +static int +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = UINT32_MAX; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + fi++; + } + return -1; +} + +/* Lookup used only during rte_graph_feature_add() */ +static int +feature_add_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + /* Update slot where new feature can be added */ + if (slot) + *slot = fi; + fi++; + } + + return -1; +} + +/* Get control path node info from provided input feature_index */ +static int +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t feature_index, + struct rte_graph_feature_node_list **ppfinfo, + const int do_sanity_check) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + if (!ppfinfo) + return -1; + + *ppfinfo = NULL; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + /* Check sanity */ + if (do_sanity_check) + if (finfo->node_index != index) + RTE_VERIFY(0); + if (index == feature_index) { + *ppfinfo = finfo; + return 0; + } + index++; + } + return -1; +} + +/* prepare feature arc after addition of all features */ +static void +prepare_feature_arc_before_first_enable(struct rte_graph_feature_arc *arc) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + rte_atomic_store_explicit(&arc->active_feature_list, 0, + rte_memory_order_relaxed); + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + finfo->node_index = index; + feat_dbg("\t%s prepare: %s added to list at index: %u", arc->feature_arc_name, + finfo->feature_node->name, index); + index++; + } +} + +/* feature arc lookup in array */ +static int +feature_arc_lookup(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (arc == (rte_graph_feature_arc_get(dm->feature_arcs[iter]))) + return 0; + } + return -1; +} + +/* Check valid values for known fields in arc to make sure arc is sane */ +static int check_feature_arc_sanity(rte_graph_feature_arc_t _arc, int iter) +{ +#ifdef FEATURE_ARC_DEBUG + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + RTE_VERIFY(arc->feature_arc_main == __rte_graph_feature_arc_main); + RTE_VERIFY(arc->feature_arc_index == iter); + + RTE_VERIFY(arc->feature_list[0]->indexed_by_features = arc->features[0]); + RTE_VERIFY(arc->feature_list[1]->indexed_by_features = arc->features[1]); + + RTE_VERIFY(rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed) < 2); +#else + RTE_SET_USED(_arc); + RTE_SET_USED(iter); +#endif + return 0; +} + +/* Perform sanity on all arc if any corruption occurred */ +static int do_sanity_all_arcs(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!dm) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (check_feature_arc_sanity(dm->feature_arcs[iter], iter)) + return -1; + } + return 0; +} + +/* get existing edge from parent_node -> child_node */ +static int +get_existing_edge(const char *arc_name, struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t i, count = 0; + + RTE_SET_USED(arc_name); + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +/* create or retrieve already existing edge from parent_node -> child_node */ +static int +__connect_graph_nodes(struct rte_node_register *parent_node, struct rte_node_register *child_node, + rte_edge_t *_edge, char *arc_name, int lineno) +{ + const char *next_node = NULL; + rte_edge_t edge; + + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { + feat_dbg("\t%s/%d: %s[%u]: \"%s\", edge reused", arc_name, lineno, + parent_node->name, edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; + } + + /* Node to be added */ + next_node = child_node->name; + + edge = rte_node_edge_update(parent_node->id, RTE_EDGE_ID_INVALID, &next_node, 1); + + if (edge == RTE_EDGE_ID_INVALID) { + graph_err("edge invalid"); + return -1; + } + edge = rte_node_edge_count(parent_node->id) - 1; + + feat_dbg("\t%s/%d: %s[%u]: \"%s\", new edge added", arc_name, lineno, parent_node->name, + edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; +} + +/* feature arc initialization */ +static int +feature_arc_main_init(rte_graph_feature_arc_main_t **pfl, uint32_t max_feature_arcs) +{ + rte_graph_feature_arc_main_t *pm = NULL; + uint32_t i; + size_t sz; + + if (!pfl) + return -1; + + sz = sizeof(rte_graph_feature_arc_main_t) + + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); + + pm = rte_malloc("rte_graph_feature_arc_main", sz, 0); + if (!pm) + return -1; + + memset(pm, 0, sz); + + for (i = 0; i < max_feature_arcs; i++) + pm->feature_arcs[i] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + pm->max_feature_arcs = max_feature_arcs; + + *pfl = pm; + + return 0; +} + +/* feature arc initialization, public API */ +int +rte_graph_feature_arc_init(int max_feature_arcs) +{ + if (!max_feature_arcs) + return -1; + + if (__rte_graph_feature_arc_main) + return -1; + + return feature_arc_main_init(&__rte_graph_feature_arc_main, max_feature_arcs); +} + +/* reset feature list before switching to passive list */ +static void +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t list_index) +{ + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + uint32_t i, j; + + list = arc->feature_list[list_index]; + feat = arc->features[list_index]; + + /*Initialize variables*/ + memset(feat, 0, arc->feature_size * arc->max_features); + memset(list, 0, arc->feature_list_size); + + /* Initialize feature and feature_data */ + for (i = 0; i < arc->max_features; i++) { + feat = __rte_graph_feature_get(arc, i, list_index); + feat->this_feature_index = i; + + for (j = 0; j < arc->max_indexes; j++) { + fdata = rte_graph_feature_data_get(arc, feat, j); + fdata->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + fdata->next_edge = UINT16_MAX; + fdata->user_data = UINT32_MAX; + } + } + + for (i = 0; i < arc->max_indexes; i++) + list->first_enabled_feature_by_index[i] = RTE_GRAPH_FEATURE_INVALID; +} + +static int +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char *flist_name, + rte_graph_feature_list_t **pplist, + struct rte_graph_feature **ppfeature, uint32_t list_index) +{ + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + size_t list_size, feat_size, fdata_size; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + + list_size = sizeof(struct rte_graph_feature_list) + + (sizeof(list->first_enabled_feature_by_index[0]) * arc->max_indexes); + + list_size = RTE_ALIGN_CEIL(list_size, RTE_CACHE_LINE_SIZE); + + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); + if (!list) + return -ENOMEM; + + memset(list, 0, list_size); + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); + + /* Let one feature and its associated data per index capture complete + * cache lines + */ + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + fdata_size, + RTE_CACHE_LINE_SIZE); + + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, "feat"); + + feat = rte_malloc(fname, feat_size * arc->max_features, RTE_CACHE_LINE_SIZE); + if (!feat) { + rte_free(list); + return -ENOMEM; + } + arc->feature_size = feat_size; + arc->feature_data_size = fdata_size; + arc->feature_list_size = list_size; + + /* Initialize list */ + list->indexed_by_features = feat; + *pplist = list; + *ppfeature = feat; + + feature_arc_list_reset(arc, list_index); + + return 0; +} + +/* free resources allocated in feature_arc_list_init() */ +static void +feature_arc_list_destroy(struct rte_graph_feature_arc *arc, int list_index) +{ + rte_graph_feature_list_t *list = NULL; + + list = arc->feature_list[list_index]; + + rte_free(list->indexed_by_features); + + arc->features[list_index] = NULL; + + rte_free(list); + + arc->feature_list[list_index] = NULL; +} + +int +rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, rte_graph_feature_arc_t *_arc) +{ + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_arc_main_t *dfm = NULL; + struct rte_graph_feature_arc *arc = NULL; + struct rte_graph_feature *df = NULL; + uint32_t iter, j, arc_index; + size_t sz; + + if (!_arc) + SET_ERR_JMP(EINVAL, err, "%s: Invalid _arc", feature_arc_name); + + if (max_features < 2) + SET_ERR_JMP(EINVAL, err, "%s: max_features must be greater than 1", + feature_arc_name); + + if (!start_node) + SET_ERR_JMP(EINVAL, err, "%s: start_node cannot be NULL", + feature_arc_name); + + if (!feature_arc_name) + SET_ERR_JMP(EINVAL, err, "%s: feature_arc name cannot be NULL", + feature_arc_name); + + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) + SET_ERR_JMP(EAGAIN, err, "%s: number of features cannot be greater than 64", + feature_arc_name); + + /* + * Application hasn't called rte_graph_feature_arc_init(). Initialize with + * default values + */ + if (!__rte_graph_feature_arc_main) { + if (rte_graph_feature_arc_init((int)RTE_GRAPH_FEATURE_ARC_MAX) < 0) { + graph_err("rte_graph_feature_arc_init() failed"); + return -1; + } + } + + /* If name is not unique */ + if (!rte_graph_feature_arc_lookup_by_name(feature_arc_name, NULL)) + SET_ERR_JMP(EINVAL, err, "%s: feature arc name already exists", + feature_arc_name); + + dfm = __rte_graph_feature_arc_main; + + /* threshold check */ + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) + SET_ERR_JMP(EAGAIN, err, "%s: max number (%u) of feature arcs reached", + feature_arc_name, dfm->max_feature_arcs); + + /* Find the free slot for feature arc */ + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { + if (dfm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + break; + } + arc_index = iter; + + if (arc_index >= dfm->max_feature_arcs) { + graph_err("No free slot found for num_feature_arc"); + return -1; + } + + /* This should not happen */ + RTE_VERIFY(dfm->feature_arcs[arc_index] == RTE_GRAPH_FEATURE_ARC_INITIALIZER); + + /* size of feature arc + feature_bit_mask_by_index */ + sz = RTE_ALIGN_CEIL(sizeof(*arc) + (sizeof(uint64_t) * max_indexes), RTE_CACHE_LINE_SIZE); + + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); + + if (!arc) { + graph_err("malloc failed for feature_arc_create()"); + return -1; + } + + memset(arc, 0, sz); + + /* Initialize rte_graph port group fixed variables */ + STAILQ_INIT(&arc->all_features); + strncpy(arc->feature_arc_name, feature_arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + arc->feature_arc_main = (void *)dfm; + arc->start_node = start_node; + arc->max_features = max_features; + arc->max_indexes = max_indexes; + arc->feature_arc_index = arc_index; + + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc->features[0], 0) < 0) { + rte_free(arc); + graph_err("feature_arc_list_init(0) failed"); + return -1; + } + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc->features[1], 1) < 0) { + feature_arc_list_destroy(arc, 0); + rte_free(arc); + graph_err("feature_arc_list_init(1) failed"); + return -1; + } + + for (iter = 0; iter < arc->max_features; iter++) { + df = rte_graph_feature_get(arc, iter); + for (j = 0; j < arc->max_indexes; j++) { + gfd = rte_graph_feature_data_get(arc, df, j); + gfd->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + } + } + dfm->feature_arcs[arc->feature_arc_index] = (rte_graph_feature_arc_t)arc; + dfm->num_feature_arcs++; + + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + + do_sanity_all_arcs(); + + feat_dbg("Feature arc %s[%p] created with max_features: %u and indexes: %u", + feature_arc_name, (void *)arc, max_features, max_indexes); + return 0; + +err: + return -rte_errno; +} + +int +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *_runs_after, const char *runs_before) +{ + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo = NULL; + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; + char feature_name[3*RTE_GRAPH_FEATURE_ARC_NAMELEN]; + const char *runs_after = NULL; + uint32_t num_feature = 0; + uint32_t slot, add_flag; + rte_edge_t edge = -1; + + /* sanity */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + graph_err("feature arc not created: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (feature_arc_lookup(_arc)) { + graph_err("invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (arc->runtime_enabled_features) { + graph_err("adding features after enabling any one of them is not supported"); + return -1; + } + + if ((_runs_after != NULL) && (runs_before != NULL) && + (_runs_after == runs_before)) { + graph_err("runs_after and runs_before are same '%s:%s]", _runs_after, + runs_before); + return -1; + } + + if (!feature_node) { + graph_err("feature_node: %p invalid", feature_node); + return -1; + } + + arc = rte_graph_feature_arc_get(_arc); + + if (feature_node->id == RTE_NODE_ID_INVALID) { + graph_err("Invalid node: %s", feature_node->name); + return -1; + } + + if (!feature_add_lookup(arc, feature_node->name, &finfo, &slot)) { + graph_err("%s feature already added", feature_node->name); + return -1; + } + + if (slot >= arc->max_features) { + graph_err("%s: Max features %u added to feature arc", + arc->feature_arc_name, slot); + return -1; + } + + if (strstr(feature_node->name, arc->start_node->name)) { + graph_err("Feature %s cannot point to itself: %s", feature_node->name, + arc->start_node->name); + return -1; + } + + feat_dbg("%s: adding feature node: %s at feature index: %u", arc->feature_arc_name, + feature_node->name, slot); + + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc->feature_arc_name)) { + graph_err("unable to connect %s -> %s", arc->start_node->name, feature_node->name); + return -1; + } + + snprintf(feature_name, sizeof(feature_name), "%s-%s-finfo", + arc->feature_arc_name, feature_node->name); + + finfo = rte_malloc(feature_name, sizeof(*finfo), 0); + if (!finfo) { + graph_err("%s/%s: rte_malloc failed", arc->feature_arc_name, feature_node->name); + return -1; + } + + memset(finfo, 0, sizeof(*finfo)); + + finfo->feature_arc = (void *)arc; + finfo->feature_node = feature_node; + finfo->edge_to_this_feature = edge; + arc->runtime_enabled_features = 0; + + /* + * if no constraints given and provided feature is not the first feature, + * explicitly set "runs_after" as last_feature. Handles the case: + * + * add(f1, NULL, NULL); + * add(f2, NULL, NULL); + */ + num_feature = rte_graph_feature_arc_num_features(_arc); + if (!_runs_after && !runs_before && num_feature) + runs_after = rte_graph_feature_arc_feature_to_name(_arc, num_feature - 1); + else + runs_after = _runs_after; + + /* Check for before and after constraints */ + if (runs_before) { + /* runs_before sanity */ + if (feature_lookup(arc, runs_before, &before_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid before feature name: %s", runs_before); + + if (!before_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_before %s does not exist", runs_before); + + /* + * Starting from 0 to runs_before, continue connecting edges + */ + add_flag = 1; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (!add_flag) + /* Nodes after seeing "runs_before", finfo connects to temp*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, + NULL, arc->feature_arc_name); + /* + * As soon as we see runs_before. stop adding edges + */ + if (!strncmp(temp->feature_node->name, runs_before, + RTE_GRAPH_NAMESIZE)) { + if (!connect_graph_nodes(finfo->feature_node, temp->feature_node, + &edge, arc->feature_arc_name)) + add_flag = 0; + } + + if (add_flag) + /* Nodes before seeing "run_before" are connected to finfo */ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + } + } + + if (runs_after) { + if (feature_lookup(arc, runs_after, &after_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid after feature_name %s", runs_after); + + if (!after_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_after %s does not exist", runs_after); + + /* Starting from runs_after to end continue connecting edges */ + add_flag = 0; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (add_flag) + /* We have already seen runs_after now */ + /* Add all features as next node to current feature*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, NULL, + arc->feature_arc_name); + else + /* Connect initial nodes to newly added node*/ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + + /* as soon as we see runs_after. start adding edges + * from next iteration + */ + if (!strncmp(temp->feature_node->name, runs_after, RTE_GRAPH_NAMESIZE)) + add_flag = 1; + } + + /* add feature next to runs_after */ + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, next_feature); + } else { + if (before_finfo) { + /* add finfo before "before_finfo" element in the list */ + after_finfo = NULL; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (before_finfo == temp) { + if (after_finfo) + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, + finfo, next_feature); + else + STAILQ_INSERT_HEAD(&arc->all_features, finfo, + next_feature); + + return 0; + } + after_finfo = temp; + } + } else { + /* Very first feature just needs to be added to list */ + STAILQ_INSERT_TAIL(&arc->all_features, finfo, next_feature); + } + } + + return 0; + +finfo_free: + rte_free(finfo); + + return -1; +} + +int +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot; + + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { + *feat = (rte_graph_feature_t) slot; + return 0; + } + + return -1; +} + +int +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int is_enable_disable, bool emit_logs) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + rte_graph_feature_rt_list_t active_list; + struct rte_graph_feature *gf = NULL; + uint32_t slot; + + /* validate _arc */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + FEAT_COND_ERR(emit_logs, "invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -EINVAL; + } + + /* validate index */ + if (index >= arc->max_indexes) { + FEAT_COND_ERR(emit_logs, "%s: Invalid provided index: %u >= %u configured", + arc->feature_arc_name, index, arc->max_indexes); + return -1; + } + + /* validate feature_name is already added or not */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) { + FEAT_COND_ERR(emit_logs, "%s: No feature %s added", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + if (!finfo) { + FEAT_COND_ERR(emit_logs, "%s: No feature: %s found", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + /* slot should be in valid range */ + if (slot >= arc->max_features) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid free slot %u(max=%u) for feature", + arc->feature_arc_name, feature_name, slot, arc->max_features); + return -EINVAL; + } + + /* slot should be in range of 0 - 63 */ + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid slot: %u", arc->feature_arc_name, + feature_name, slot); + return -EINVAL; + } + + if (finfo->node_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s/%s: lookup slot mismatch for finfo idx: %u and lookup slot: %u", + arc->feature_arc_name, feature_name, finfo->node_index, slot); + return -1; + } + + active_list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + + /* Get feature from active list */ + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(active_list)); + if (gf->this_feature_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s: %s rcvd feature_idx: %u does not match with saved: %u", + arc->feature_arc_name, feature_name, slot, gf->this_feature_index); + return -1; + } + + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & + RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s already enabled on index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + if (!is_enable_disable && !arc->runtime_enabled_features) { + FEAT_COND_ERR(emit_logs, "%s: No feature enabled to disable", + arc->feature_arc_name); + return -1; + } + + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s not enabled in bitmask for index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + return 0; +} + +/* + * Before switch to passive list, user_data needs to be copied from active list to passive list + */ +static void +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t dest_list_index, + uint16_t src_list_index) +{ + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; + struct rte_graph_feature *sgf = NULL, *dgf = NULL; + uint32_t i, j; + + for (i = 0; i < arc->max_features; i++) { + sgf = __rte_graph_feature_get(arc, i, src_list_index); + dgf = __rte_graph_feature_get(arc, i, dest_list_index); + for (j = 0; j < arc->max_indexes; j++) { + sgfd = rte_graph_feature_data_get(arc, sgf, j); + dgfd = rte_graph_feature_data_get(arc, dgf, j); + dgfd->user_data = sgfd->user_data; + } + } +} +/* + * Fill fast path information like + * - next_edge + * - next_enabled_feature + */ +static void +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t list_index) +{ + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; + uint32_t fi = UINT32_MAX, di = UINT32_MAX, prev_fi = UINT32_MAX; + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; + rte_graph_feature_list_t *flist = NULL; + rte_edge_t edge = UINT16_MAX; + uint64_t bitmask = 0; + + flist = arc->feature_list[list_index]; + + for (di = 0; di < arc->max_indexes; di++) { + bitmask = arc->feature_bit_mask_by_index[di]; + prev_fi = RTE_GRAPH_FEATURE_INVALID; + /* for each feature set for index, set fast path data */ + while (rte_bsf64_safe(bitmask, &fi)) { + gf = __rte_graph_feature_get(arc, fi, list_index); + gfd = rte_graph_feature_data_get(arc, gf, di); + RTE_VERIFY(!feature_arc_node_info_lookup(arc, fi, &finfo, 1)); + + /* If previous feature_index was valid in last loop */ + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { + prev_gf = __rte_graph_feature_get(arc, prev_fi, list_index); + prev_gfd = rte_graph_feature_data_get(arc, prev_gf, di); + /* + * Get edge of previous feature node connecting + * to this feature node + */ + RTE_VERIFY(!feature_arc_node_info_lookup(arc, prev_fi, + &prev_finfo, 1)); + if (!get_existing_edge(arc->feature_arc_name, + prev_finfo->feature_node, + finfo->feature_node, &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (%u->%u)%s[%u] = %s", + arc->feature_arc_name, list_index, di, + prev_gfd->user_data, prev_fi, fi, + prev_finfo->feature_node->name, + edge, finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* + * Fill current feature as next enabled + * feature to previous one + */ + prev_gfd->next_enabled_feature = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* On first feature edge of the node to be added */ + if (fi == rte_bsf64(arc->feature_bit_mask_by_index[di])) { + if (!get_existing_edge(arc->feature_arc_name, arc->start_node, + finfo->feature_node, + &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (->%u)%s[%u]=%s", + arc->feature_arc_name, list_index, di, + gfd->user_data, fi, + arc->start_node->name, edge, + finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* Set first feature set array for index*/ + flist->first_enabled_feature_by_index[di] = + (rte_graph_feature_t)fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* Clear current feature index */ + bitmask &= ~RTE_BIT64(fi); + } + } +} + +int +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const + char *feature_name, int32_t user_data) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_rt_list_t passive_list, active_list; + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Enabling feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (!arc->runtime_enabled_features) + prepare_feature_arc_before_first_enable(arc); + + if (rte_graph_feature_validate(_arc, index, feature_name, 1, true)) + return -1; + + /** This should not fail as validate() has passed */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) + RTE_VERIFY(0); + + active_list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + + passive_list = ARC_PASSIVE_LIST(active_list); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, active_list); + + /* Set current user-data */ + gfd->user_data = user_data; + + /* Set bitmask in control path bitmask */ + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + refill_feature_fastpath_data(arc, passive_list); + + /* If first time feature getting enabled */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[active_list], + rte_memory_order_relaxed); + + /* On very first feature enable instance */ + if (!finfo->ref_count) + bitmask |= RTE_BIT64(slot); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], + bitmask, rte_memory_order_relaxed); + + /* Slow path updates */ + arc->runtime_enabled_features++; + + /* Increase feature node info reference count */ + finfo->ref_count++; + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After enable, switched active feature list to %u", + arc->feature_arc_name, feature_name, passive_list); + + return 0; +} + +int +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_rt_list_t passive_list, active_list; + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Disable feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (rte_graph_feature_validate(_arc, index, feature_name, 0, true)) + return -1; + + if (feature_lookup(arc, feature_name, &finfo, &slot)) + return -1; + + active_list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + + passive_list = ARC_PASSIVE_LIST(active_list); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, active_list); + + /* Reset current user-data */ + gfd->user_data = ~0; + + refill_feature_fastpath_data(arc, passive_list); + + finfo->ref_count--; + arc->runtime_enabled_features--; + + /* If no feature enabled, reset feature in u64 fast path bitmask */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[active_list], + rte_memory_order_relaxed); + + /* When last feature is disabled */ + if (!finfo->ref_count) + bitmask &= ~(RTE_BIT64(slot)); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], bitmask, + rte_memory_order_relaxed); + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After disable, switched active feature list to %u", + arc->feature_arc_name, feature_name, passive_list); + + return 0; +} + +int +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_node_list *node_info = NULL; + + while (!STAILQ_EMPTY(&arc->all_features)) { + node_info = STAILQ_FIRST(&arc->all_features); + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); + rte_free(node_info); + } + feature_arc_list_destroy(arc, 0); + feature_arc_list_destroy(arc, 1); + + dm->feature_arcs[arc->feature_arc_index] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + rte_free(arc); + + do_sanity_all_arcs(); + + return 0; +} + +int +rte_graph_feature_arc_cleanup(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm->feature_arcs[iter]); + } + rte_free(dm); + + __rte_graph_feature_arc_main = NULL; + + return 0; +} + +int +rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_arc *arc = NULL; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + if (_arc) + *_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); + + if ((strstr(arc->feature_arc_name, arc_name)) && + (strlen(arc->feature_arc_name) == strlen(arc_name))) { + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + return 0; + } + } + + return -1; +} + +uint32_t +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + return arc->runtime_enabled_features; +} + +uint32_t +rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t count = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) + count++; + + return count; +} + +char * +rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node->name; + + return NULL; +} + +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node; + + return NULL; + +} diff --git a/lib/graph/meson.build b/lib/graph/meson.build index 0cb15442ab..d916176fb7 100644 --- a/lib/graph/meson.build +++ b/lib/graph/meson.build @@ -14,11 +14,13 @@ sources = files( 'graph_debug.c', 'graph_stats.c', 'graph_populate.c', + 'graph_feature_arc.c', 'graph_pcap.c', 'rte_graph_worker.c', 'rte_graph_model_mcore_dispatch.c', ) headers = files('rte_graph.h', 'rte_graph_worker.h') +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') indirect_headers += files( 'rte_graph_model_mcore_dispatch.h', 'rte_graph_model_rtc.h', diff --git a/lib/graph/rte_graph_feature_arc.h b/lib/graph/rte_graph_feature_arc.h new file mode 100644 index 0000000000..1615f8e1c8 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc.h @@ -0,0 +1,431 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ +#define _RTE_GRAPH_FEATURE_ARC_H_ + +#include <assert.h> +#include <errno.h> +#include <signal.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <rte_common.h> +#include <rte_compat.h> +#include <rte_debug.h> +#include <rte_graph.h> +#include <rte_graph_worker.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @file + * + * rte_graph_feature_arc.h + * + * Define APIs and structures/variables with respect to feature arc + * + * - Feature arc(s) + * - Feature(s) + * + * A feature arc represents an ordered list of features/protocol-nodes at a + * given networking layer. Feature arc provides a high level abstraction to + * connect various *rte_graph* nodes, designated as *feature nodes*, and + * allowing steering of packets across these feature nodes fast path processing + * in a generic manner. In a typical network stack, often a protocol or feature + * must be first enabled on a given interface, before any packet is steered + * towards it for feature processing. For eg: incoming IPv4 packets are sent to + * routing sub-system only after a valid IPv4 address is assigned to the + * received interface. In other words, often packets needs to be steered across + * features not based on the packet content but based on whether a feature is + * enable or disable on a given incoming/outgoing interface. Feature arc + * provides mechanism to enable/disable feature(s) on each interface at runtime + * and allow seamless packet steering across runtime enabled feature nodes in + * fast path. + * + * Feature arc also provides a way to steer packets from standard nodes to + * custom/user-defined *feature nodes* without any change in standard node's + * fast path functions + * + * On a given interface multiple feature(s) might be enabled in a particular + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" + * features may be enabled on "eth0" interface in "L3-output" feature arc. + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" + * interface in same "L3-output" feature arc. + * + * When multiple features are present in a given feature arc, its imperative + * to allow each feature processing in a particular sequential order. For + * instance, in "L3-input" feature arc it may be required to run "IPsec + * input" feature first, for packet decryption, before "ip-lookup". So a + * sequential order must be maintained among features present in a feature arc. + * + * Features are enabled/disabled multiple times at runtime to some or all + * available interfaces present in the system. Enable/disabling features on one + * interface is independent of other interface. + * + * A given feature might consume packet (if it's configured to consume) or may + * forward it to next enabled feature. For instance, "IPsec input" feature may + * consume/drop all packets with "Protect" policy action while all packets with + * policy action as "Bypass" may be forwarded to next enabled feature (with in + * same feature arc) + * + * This library facilitates rte graph based applications to steer packets in + * fast path to different feature nodes with-in a feature arc and support all + * functionalities described above + * + * In order to use feature-arc APIs, applications needs to do following in + * control path: + * - Initialize feature arc library via rte_graph_feature_arc_init() + * - Create feature arc via rte_graph_feature_arc_create() + * - *Before calling rte_graph_create()*, features must be added to feature-arc + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding + * features in a sequential order with "runs_after" and "runs_before" + * constraints. + * - Post rte_graph_create(), features can be enabled/disabled at runtime on + * any interface via rte_graph_feature_enable()/rte_graph_feature_disable() + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() + * + * In fast path, APIs are provided to steer packets towards feature path from + * - start_node (provided as an argument to rte_graph_feature_arc_create()) + * - feature nodes (which are added via rte_graph_feature_add()) + * + * For typical steering of packets across feature nodes, application required + * to know "rte_edges" which are saved in feature data object. Feature data + * object is unique for every interface per feature with in a feature arc. + * + * When steering packets from start_node to feature node: + * - rte_graph_feature_arc_first_feature_get() provides first enabled feature. + * - Next rte_edge from start_node to first enabled feature can be obtained via + * rte_graph_feature_arc_feature_set() + * + * rte_mbuf can carry [current feature, interface index] from start_node of an + * arc to other feature nodes + * + * At the time of feature enable(rte_graph_feature_enable), application can set + * 32-bit unique user_data specific to feature per interface. In fast path + * user_data can be retrieved via rte_graph_feature_user_data_get(). User data + * can hold application specific cookie like IPsec policy database index, FIB + * table index etc. + * + * If feature node is not consuming packet, next enabled feature and next + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() + * + * It is application responsibility to ensure that at-least *last feature*(or + * sink feature) must be enabled from where packet can exit feature-arc path, + * if *NO* intermediate feature is consuming the packet and it has reached till + * the end of feature arc path + * + * It is recommended that all features *MUST* be added to feature arc before + * calling `rte_graph_create()`. Addition of features after + * `rte_graph_create()` may not work functionally. + * Although,rte_graph_feature_enable()/rte_graph_feature_disable() should be + * called after `rte_graph_create()` in control plane. + * + * Synchronization among cores + * --------------------------- + * Subsequent calls to rte_graph_feature_enable() is allowed while worker cores + * are processing in rte_graph_walk() loop. However, for + * rte_graph_feature_disable() application must use RCU based synchronization + */ + +/** Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER ((rte_graph_feature_arc_t)UINT64_MAX) + +/** Max number of feature arcs which can be created */ +#define RTE_GRAPH_FEATURE_ARC_MAX 64 + +/** Max number of features supported in a given feature arc */ +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 + +/** Length of feature arc name */ +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE + +/** @internal */ +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) + +/**< Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_INVALID rte_graph_feature_cast(UINT8_MAX) + +/** rte_graph feature arc object */ +typedef uintptr_t rte_graph_feature_arc_t; + +/** rte_graph feature object */ +typedef uint8_t rte_graph_feature_t; + +/** runtime active feature list index with in feature arc*/ +typedef uint16_t rte_graph_feature_rt_list_t; + +/** per feature arc monotonically increasing counter to synchronize fast path APIs */ +typedef uint16_t rte_graph_feature_counter_t; + +/** + * Initialize feature arc subsystem + * + * @param max_feature_arcs + * Maximum number of feature arcs required to be supported + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_init(int max_feature_arcs); + +/** + * Create a feature arc + * + * @param feature_arc_name + * Feature arc name with max length of @ref RTE_GRAPH_FEATURE_ARC_NAMELEN + * @param max_features + * Maximum number of features to be supported in this feature arc + * @param max_indexes + * Maximum number of interfaces/ports/indexes to be supported + * @param start_node + * Base node where this feature arc's features are checked in fast path + * @param[out] _arc + * Feature arc object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, + rte_graph_feature_arc_t *_arc); + +/** + * Get feature arc object with name + * + * @param arc_name + * Feature arc name provided to successful @ref rte_graph_feature_arc_create + * @param[out] _arc + * Feature arc object returned. Valid only when API returns SUCCESS + * + * @return + * 0: Success + * <0: Failure. + */ +__rte_experimental +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc); + +/** + * Add a feature to already created feature arc. For instance + * + * 1. Add first feature node: "ipv4-input" to input arc + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); + * + * 2. Add "ipsec-input" feature node after "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", NULL); + * + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", NULL, "ipv4-input"); + * + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4-input", "ipsec-input"); + * + * @param _arc + * Feature arc handle returned from @ref rte_graph_feature_arc_create() + * @param feature_node + * Graph node representing feature. On success, feature_node is next_node of + * feature_arc->start_node + * @param runs_after + * Add this feature_node after already added "runs_after". Creates + * start_node -> runs_after -> this_feature sequence + * @param runs_before + * Add this feature_node before already added "runs_before". Creates + * start_node -> this_feature -> runs_before sequence + * + * <I> Must be called before rte_graph_create() </I> + * <I> rte_graph_feature_add() is not allowed after call to + * rte_graph_feature_enable() so all features must be added before they can be + * enabled </I> + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *runs_after, const char *runs_before); + +/** + * Enable feature within a feature arc + * + * Must be called after @b rte_graph_create(). + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param user_data + * Application specific data which is retrieved in fast path + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int32_t user_data); + +/** + * Validate whether subsequent enable/disable feature would succeed or not. + * API is thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param is_enable_disable + * If 1, validate whether subsequent @ref rte_graph_feature_enable would pass or not + * If 0, validate whether subsequent @ref rte_graph_feature_disable would pass or not + * @param emit_logs + * If passed true, emit error logs when failure is returned + * If passed false, do not emit error logs when failure is returned + * + * @return + * 0: Subsequent enable/disable API would pass + * <0: Subsequent enable/disable API would not pass + */ +__rte_experimental +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, int is_enable_disable, bool emit_logs); + +/** + * Disable already enabled feature within a feature arc + * + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name); + +/** + * Get rte_graph_feature_t object from feature name + * + * @param arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param feature_name + * Feature name provided to @ref rte_graph_feature_add + * @param[out] feature + * Feature object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_lookup(rte_graph_feature_arc_t arc, const char *feature_name, + rte_graph_feature_t *feature); + +/** + * Delete feature_arc object + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); + +/** + * Cleanup all feature arcs + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_cleanup(void); + +/** + * Slow path API to know how many features are added (NOT enabled) within a + * feature arc + * + * @param _arc + * Feature arc object + * + * @return: Number of added features to arc + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to know how many features are currently enabled within a + * feature arc across all indexes. If a single feature is enabled on all interfaces, + * this API would return "number_of_interfaces" as count (but not "1") + * + * @param _arc + * Feature arc object + * + * @return: Number of enabled features across all indexes + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to get feature node name from rte_graph_feature_t object + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: Name of the feature node + */ +__rte_experimental +char *rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + +/** + * Slow path API to get corresponding struct rte_node_register * from + * rte_graph_feature_t + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: struct rte_node_register * of feature node on SUCCESS else NULL + */ +__rte_experimental +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/lib/graph/rte_graph_feature_arc_worker.h b/lib/graph/rte_graph_feature_arc_worker.h new file mode 100644 index 0000000000..9b720e366c --- /dev/null +++ b/lib/graph/rte_graph_feature_arc_worker.h @@ -0,0 +1,679 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ + +#include <stddef.h> +#include <rte_graph_feature_arc.h> +#include <rte_bitops.h> + +/** + * @file + * + * rte_graph_feature_arc_worker.h + * + * Defines fast path structure + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/** @internal + * + * Slow path feature node info list + */ +struct rte_graph_feature_node_list { + /** Next feature */ + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; + + /** node representing feature */ + struct rte_node_register *feature_node; + + /** How many indexes/interfaces using this feature */ + int32_t ref_count; + + /* node_index in list (after feature_enable())*/ + uint32_t node_index; + + /** Back pointer to feature arc */ + void *feature_arc; + + /** rte_edge_t to this feature node from feature_arc->start_node */ + rte_edge_t edge_to_this_feature; +}; + +/** + * Feature data object: + * + * Feature data stores information to steer packets for: + * - a feature with in feature arc + * - Index i.e. Port/Interface index + * + * Each feature data object holds + * - User data of current feature retrieved via rte_graph_feature_user_data_get() + * - next_edge is used in two conditions when packet to be steered from + * -- start_node to first enabled feature on an interface index + * -- current feature node to next enabled feature on an interface index + * - next_enabled_feature on interface index, if current feature is not + * consuming packet + * + * While user_data corresponds to current enabled feature node however + * next_edge and next_enabled_feature corresponds to next enabled feature + * node on an interface index + * + * First enabled feature on interface index can be retrieved via: + * - rte_graph_feature_first_feature_get() if arc's start_node is trying to + * steer packet from start_node to first enabled feature on interface index + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_next_feature_get() if current node is not arc's + * start_node. Input to rte_graph_feature_next_feature_get() is current + * enabled feature and interface index + */ +typedef struct __rte_packed rte_graph_feature_data { + /** edge from current node to next enabled feature */ + rte_edge_t next_edge; + + union { + uint16_t reserved; + struct { + /** next enabled feature on index from current feature */ + rte_graph_feature_t next_enabled_feature; + }; + }; + + /** user_data set by application in rte_graph_feature_enable() for + * - current feature + * - interface index + */ + int32_t user_data; +} rte_graph_feature_data_t; + +/** + * Feature object + * + * Feature object holds feature data object for every index/interface within + * feature + * + * Within a given arc and interface index, first feature object can be + * retrieved in arc's start_node via: + * - rte_graph_feature_arc_first_feature_get() + * + * Feature data information can be retrieved for first feature in start node via + * - rte_graph_feature_arc_feature_set() + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_arc_next_feature_get() + * + * Typically application stores rte_graph_feature_t object in rte_mbuf. + * rte_graph_feature_t can be translated to (struct rte_graph_feature *) via + * rte_graph_feature_get() in fast path. Further if needed, feature data for an + * index within a feature can be retrieved via rte_graph_feature_data_get() + */ +struct __rte_cache_aligned rte_graph_feature { + /** feature index or rte_graph_feature_t */ + uint16_t this_feature_index; + + /* + * Array of size arc->feature_data_size + * + * <----------------- Feature --------------------------> + * [data-index-0][data-index-1]...[data-index-max_index-1] + * + * sizeof(feature_data_by_index[0] == sizeof(rte_graph_feature_data_t) + * + */ + uint8_t feature_data_by_index[]; +}; + +/** + * Feature list object + * + * Feature list is required to decouple fast path APIs with control path APIs. + * + * There are two feature lists: active, passive + * Passive list is duplicate of active list in terms of memory. + * + * While fast path APIs always work on active list but control plane work on + * passive list. When control plane needs to enable/disable any feature, it + * populates passive list afresh and atomically switch passive list to active + * list to make it available for fast path APIs + * + * Each feature node in start of its fast path function, must grab active list from + * arc via + * - rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * + * Retrieved list must be provided to other feature arc fast path APIs so that + * any control plane changes of active list should not impact current node + * execution iteration. Active list change would be reflected to current node + * in next iteration + * + * With active/passive lists and RCU mechanism in graph worker + * loop, application can update features at runtime without stopping fast path + * cores. A RCU synchronization is required when a feature needs to be + * disabled via rte_graph_feature_disable(). On enabling a feature, RCU + * synchronization may not be required + * + */ +typedef struct __rte_cache_aligned rte_graph_feature_list { + /** + * fast path array holding per_feature data. + * Duplicate entry as feature-arc also hold this pointer + * arc->features[] + * + *<-------------feature-0 ---------><---------feature-1 -------------->... + *[index-0][index-1]...[max_index-1]<-ALIGN->[index-0][index-1] ...[max_index-1]... + */ + struct rte_graph_feature *indexed_by_features; + /* + * fast path array holding first enabled feature per index + * (Required in start_node. In non start_node, mbuf can hold next enabled + * feature) + */ + rte_graph_feature_t first_enabled_feature_by_index[]; +} rte_graph_feature_list_t; + +/** + * rte_graph Feature arc object + * + * Feature arc object holds control plane and fast path information for all + * features and all interface index information for steering packets across + * feature nodes + * + * Within a feature arc, only RTE_GRAPH_FEATURE_MAX_PER_ARC features can be + * added. If more features needs to be added, another feature arc can be + * created + * + * Application gets rte_graph_feature_arc_t object via + * - rte_graph_feature_arc_create() OR + * - rte_graph_feature_arc_lookup_by_name() + * + * In fast path, rte_graph_feature_arc_t can be translated to (struct + * rte_graph_feature_arc *) via rte_graph_feature_arc_get(). Later is needed to + * add as an input argument to all fast path feature arc APIs + */ +struct __rte_cache_aligned rte_graph_feature_arc { + /* First 64B is fast path variables */ + RTE_MARKER fast_path_variables; + + /** runtime active feature list */ + RTE_ATOMIC(rte_graph_feature_rt_list_t) active_feature_list; + + /** Actual Size of feature_list object */ + uint16_t feature_list_size; + + /** + * Size each feature in fastpath. + * Required to navigate from feature to another feature in fast path + */ + uint16_t feature_size; + + /** + * Size of all feature data for an index + * Required to navigate through various feature data within a feature + * in fast path + */ + uint16_t feature_data_size; + + /** + * Quick fast path bitmask indicating if any feature enabled or not on + * any of the indexes. Helps in optimally process packets for the case + * when features are added but not enabled + * + * Separate for active and passive list + */ + RTE_ATOMIC(uint64_t) feature_enable_bitmask[2]; + + /** + * Pointer to both active and passive feature list object + */ + rte_graph_feature_list_t *feature_list[2]; + + /** + * Feature objects for each list + */ + struct rte_graph_feature *features[2]; + + /** index in feature_arc_main */ + uint16_t feature_arc_index; + + uint16_t reserved[3]; + + /** Slow path variables follows*/ + RTE_MARKER slow_path_variables; + + /** feature arc name */ + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** All feature lists */ + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; + + /** control plane counter to track enabled features */ + uint32_t runtime_enabled_features; + + /** Back pointer to feature_arc_main */ + void *feature_arc_main; + + /** Arc's start/base node */ + struct rte_node_register *start_node; + + /** maximum number of features supported by this arc */ + uint32_t max_features; + + /** maximum number of index supported by this arc */ + uint32_t max_indexes; + + /** Slow path bit mask per feature per index */ + uint64_t feature_bit_mask_by_index[]; +}; + +/** + * Feature arc main object + * + * Holds all feature arcs created by application + * + * RTE_GRAPH_FEATURE_ARC_MAX number of feature arcs can be created by + * application via rte_graph_feature_arc_create() + */ +typedef struct feature_arc_main { + /** number of feature arcs created by application */ + uint32_t num_feature_arcs; + + /** max features arcs allowed */ + uint32_t max_feature_arcs; + + /** feature arcs */ + rte_graph_feature_arc_t feature_arcs[]; +} rte_graph_feature_arc_main_t; + +/** @internal Get feature arc pointer from object */ +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc *)arc) + +extern rte_graph_feature_arc_main_t *__feature_arc_main; + +/** + * API to know if feature is valid or not + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_is_valid(rte_graph_feature_t feature) +{ + return (feature != RTE_GRAPH_FEATURE_INVALID); +} + +/** + * Get rte_graph_feature object with no checks + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * @param feature_list + * active feature list retrieved from rte_graph_feature_arc_has_any_feature() + * or rte_graph_feature_arc_has_feature() + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature, + const rte_graph_feature_rt_list_t feature_list) +{ + return ((struct rte_graph_feature *)(((uint8_t *)arc->features[feature_list]) + + (feature * arc->feature_size))); +} + +/** + * Get rte_graph_feature object for a given interface/index from feature arc + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature) +{ + rte_graph_feature_rt_list_t list; + + if (unlikely(feature >= arc->max_features)) + RTE_VERIFY(0); + + if (likely(rte_graph_feature_is_valid(feature))) { + list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + return __rte_graph_feature_get(arc, feature, list); + } + + return NULL; +} + +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + RTE_SET_USED(arc); + return ((rte_graph_feature_data_t *)(((uint8_t *)feature->feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Get rte_graph feature data object for a index in feature + * + * @param arc + * feature arc + * @param feature + * Pointer to feature object + * @param index + * Index of feature maintained in slow path linked list + * + * @return + * Valid feature data + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + if (likely(index < arc->max_indexes)) + return __rte_graph_feature_data_get(arc, feature, index); + + RTE_VERIFY(0); +} + +/** + * Fast path API to check if any feature enabled on a feature arc + * Typically from arc->start_node process function + * + * @param arc + * Feature arc object + * @param[out] plist + * Pointer to runtime active feature list which needs to be provided to other + * fast path APIs + * + * @return + * 0: If no feature enabled + * Non-Zero: Bitmask of features enabled. plist is valid + * + */ +__rte_experimental +static __rte_always_inline uint64_t +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_rt_list_t *plist) +{ + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Fast path API to check if provided feature is enabled on any interface/index + * or not + * + * @param arc + * Feature arc object + * @param feature + * Input rte_graph_feature_t that needs to be checked + * @param[out] plist + * Returns active list to caller which needs to be provided to other fast path + * APIs + * + * @return + * 1: If input [feature] is enabled in arc + * 0: If input [feature] is not enabled in arc + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_t feature, + rte_graph_feature_rt_list_t *plist) +{ + uint64_t bitmask = RTE_BIT64(feature); + + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (bitmask & rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Prefetch feature arc fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) +{ + rte_prefetch0((void *)&arc->fast_path_variables); +} + +/** + * Prefetch feature related fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature) +{ + /* feature cache line */ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, list)); +} + +/** + * Prefetch feature data upfront. Perform sanity + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object returned from @ref + * rte_graph_feature_arc_first_feature_get() + * @param index + * Interface/index + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, uint32_t index) +{ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)((uint8_t *)arc->features[list] + + offsetof(struct rte_graph_feature, feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Fast path API to get first enabled feature on interface index + * Typically required in arc->start_node so that from returned feature, + * feature-data can be retrieved to steer packets + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. + * + * @return + * 1. Success. If first feature field is enabled and returned [feature] is valid + * 0. Failure. If first feature field is disabled in arc + * + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + + *feature = feature_list->first_enabled_feature_by_index[index]; + + return rte_graph_feature_is_valid(*feature); +} + +/** + * Fast path API to get next enabled feature on interface index with provided + * input feature + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. API sets next enabled feature on [index] + * from provided input feature. Valid only if API returns Success + * @param[out] next_edge + * Edge from current feature to next feature. Valid only if next feature is valid + * + * @return + * 1. Success. first feature field is enabled/valid + * 0. Failure. first feature field is disabled/invalid + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature, + rte_edge_t *next_edge) +{ + rte_graph_feature_data_t *feature_data = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(*feature))) { + f = __rte_graph_feature_get(arc, *feature, list); + feature_data = rte_graph_feature_data_get(arc, f, index); + *feature = feature_data->next_enabled_feature; + *next_edge = feature_data->next_edge; + return rte_graph_feature_is_valid(*feature); + } + + return 0; +} + +/** + * Set fields with respect to first enabled feature in an arc and return edge + * Typically returned feature and interface index must be saved in rte_mbuf + * structure to pass this information to next feature node + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param index + * Index (of interface) + * @param[out] gf + * Pointer to rte_graph_feature_t. Valid if API returns Success + * @param[out] edge + * Edge to steer packet from arc->start_node to first enabled feature. Valid + * only if API returns Success + * + * @return + * 0: If valid feature is enabled and set by API in *gf + * 1: If valid feature is NOT enabled + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_t +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *gf, + rte_edge_t *edge) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + struct rte_graph_feature_data *feature_data = NULL; + struct rte_graph_feature *feature = NULL; + rte_graph_feature_t f; + + f = feature_list->first_enabled_feature_by_index[index]; + + if (unlikely(rte_graph_feature_is_valid(f))) { + feature = __rte_graph_feature_get(arc, f, list); + feature_data = rte_graph_feature_data_get(arc, feature, index); + *gf = f; + *edge = feature_data->next_edge; + return 0; + } + + return 1; +} + +__rte_experimental +static __rte_always_inline int32_t +__rte_graph_feature_user_data_get(rte_graph_feature_data_t *fdata) +{ + return fdata->user_data; +} + +/** + * Get user data corresponding to current feature set by application in + * rte_graph_feature_enable() + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Feature index + * @param index + * Interface index + * + * @return + * -1: Failure + * Valid user data: Success + */ +__rte_experimental +static __rte_always_inline int32_t +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, + uint32_t index) +{ + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(feature))) { + f = __rte_graph_feature_get(arc, feature, list); + fdata = rte_graph_feature_data_get(arc, f, index); + return __rte_graph_feature_user_data_get(fdata); + } + + return -1; +} +#ifdef __cplusplus +} +#endif +#endif diff --git a/lib/graph/version.map b/lib/graph/version.map index 2c83425ddc..3b7f475afd 100644 --- a/lib/graph/version.map +++ b/lib/graph/version.map @@ -52,3 +52,23 @@ DPDK_25 { local: *; }; + +EXPERIMENTAL { + global: + + # added in 24.11 + rte_graph_feature_arc_init; + rte_graph_feature_arc_create; + rte_graph_feature_arc_lookup_by_name; + rte_graph_feature_add; + rte_graph_feature_enable; + rte_graph_feature_validate; + rte_graph_feature_disable; + rte_graph_feature_lookup; + rte_graph_feature_arc_destroy; + rte_graph_feature_arc_cleanup; + rte_graph_feature_arc_num_enabled_features; + rte_graph_feature_arc_num_features; + rte_graph_feature_arc_feature_to_name; + rte_graph_feature_arc_feature_to_node; +}; -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v4 2/5] graph: add feature arc option in graph create 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 1/5] graph: add feature arc support Nitin Saxena @ 2024-10-10 13:31 ` Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 3/5] graph: add IPv4 output feature arc Nitin Saxena ` (3 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 13:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena, Pavan Nikhilesh Added option in graph create to call feature-specific process node functions. This removes extra overhead for checking feature arc status in nodes where application is not using feature arc processing Signed-off-by: Pavan Nikhilesh <pbhagavatula@marvell.com> Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 7 +++++++ lib/graph/graph.c | 1 + lib/graph/graph_populate.c | 7 ++++++- lib/graph/graph_private.h | 3 +++ lib/graph/node.c | 2 ++ lib/graph/rte_graph.h | 3 +++ 6 files changed, 22 insertions(+), 1 deletion(-) diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index bd5589b01c..237c057647 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -137,6 +137,13 @@ ABI Changes Also, make sure to start the actual text at the margin. ======================================================= +* graph: Added feature arc specific `feat_arc_proc` node callback function in + `struct rte_node_register`. If this function is not NULL and + `feature_arc_enable` is set to `true` in `struct rte_graph_param`, + rte_graph_walk() calls `feat_arc_proc` callback function instead of `process` + +* graph: Added `feature_arc_enable` parameter in `struct rte_graph_param` for + calling non-NULL `feat_arc_proc` callback function by `rte_graph_walk()` Known Issues ------------ diff --git a/lib/graph/graph.c b/lib/graph/graph.c index d5b8c9f918..b0ad3a83ae 100644 --- a/lib/graph/graph.c +++ b/lib/graph/graph.c @@ -455,6 +455,7 @@ rte_graph_create(const char *name, struct rte_graph_param *prm) graph->parent_id = RTE_GRAPH_ID_INVALID; graph->lcore_id = RTE_MAX_LCORE; graph->num_pkt_to_capture = prm->num_pkt_to_capture; + graph->feature_arc_enabled = prm->feature_arc_enable; if (prm->pcap_filename) rte_strscpy(graph->pcap_filename, prm->pcap_filename, RTE_GRAPH_PCAP_FILE_SZ); diff --git a/lib/graph/graph_populate.c b/lib/graph/graph_populate.c index ed596a7711..5d8aa7b903 100644 --- a/lib/graph/graph_populate.c +++ b/lib/graph/graph_populate.c @@ -79,8 +79,13 @@ graph_nodes_populate(struct graph *_graph) if (graph_pcap_is_enable()) { node->process = graph_pcap_dispatch; node->original_process = graph_node->node->process; - } else + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->original_process = graph_node->node->feat_arc_proc; + } else { node->process = graph_node->node->process; + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->process = graph_node->node->feat_arc_proc; + } memcpy(node->name, graph_node->node->name, RTE_GRAPH_NAMESIZE); pid = graph_node->node->parent_id; if (pid != RTE_NODE_ID_INVALID) { /* Cloned node */ diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index d557d55f2d..58ba0abeff 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -56,6 +56,7 @@ struct node { unsigned int lcore_id; /**< Node runs on the Lcore ID used for mcore dispatch model. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Allocated identifier for the node. */ @@ -126,6 +127,8 @@ struct graph { /**< Number of packets to be captured per core. */ char pcap_filename[RTE_GRAPH_PCAP_FILE_SZ]; /**< pcap file name/path. */ + uint8_t feature_arc_enabled; + /**< Graph feature arc. */ STAILQ_HEAD(gnode_list, graph_node) node_list; /**< Nodes in a graph. */ }; diff --git a/lib/graph/node.c b/lib/graph/node.c index 99a9622779..d8fd273543 100644 --- a/lib/graph/node.c +++ b/lib/graph/node.c @@ -90,6 +90,7 @@ __rte_node_register(const struct rte_node_register *reg) goto free; node->flags = reg->flags; node->process = reg->process; + node->feat_arc_proc = reg->feat_arc_proc; node->init = reg->init; node->fini = reg->fini; node->nb_edges = reg->nb_edges; @@ -137,6 +138,7 @@ node_clone(struct node *node, const char *name) /* Clone the source node */ reg->flags = node->flags; reg->process = node->process; + reg->feat_arc_proc = node->feat_arc_proc; reg->init = node->init; reg->fini = node->fini; reg->nb_edges = node->nb_edges; diff --git a/lib/graph/rte_graph.h b/lib/graph/rte_graph.h index ecfec2068a..f07272b308 100644 --- a/lib/graph/rte_graph.h +++ b/lib/graph/rte_graph.h @@ -172,6 +172,8 @@ struct rte_graph_param { uint32_t mp_capacity; /**< Capacity of memory pool for dispatch model. */ } dispatch; }; + + bool feature_arc_enable; /**< Enable Graph feature arc. */ }; /** @@ -470,6 +472,7 @@ struct rte_node_register { uint64_t flags; /**< Node configuration flag. */ #define RTE_NODE_SOURCE_F (1ULL << 0) /**< Node type is source. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arc specific process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Node Identifier. */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v4 3/5] graph: add IPv4 output feature arc 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 1/5] graph: add feature arc support Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 2/5] graph: add feature arc option in graph create Nitin Saxena @ 2024-10-10 13:31 ` Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 4/5] test/graph_feature_arc: add functional tests Nitin Saxena ` (2 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 13:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena add ipv4-output feature arc in ipv4-rewrite node to allow custom/standard nodes(like outbound IPsec policy node) in outgoing forwarding path Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/node/ip4_rewrite.c | 476 +++++++++++++++++++++++++++++------- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 4 files changed, 417 insertions(+), 97 deletions(-) diff --git a/lib/node/ip4_rewrite.c b/lib/node/ip4_rewrite.c index 34a920df5e..824ef9a4cd 100644 --- a/lib/node/ip4_rewrite.c +++ b/lib/node/ip4_rewrite.c @@ -15,39 +15,156 @@ #include "ip4_rewrite_priv.h" #include "node_private.h" +#define ALL_PKT_MASK 0xf + struct ip4_rewrite_node_ctx { + rte_graph_feature_arc_t output_feature_arc; /* Dynamic offset to mbuf priv1 */ int mbuf_priv1_off; /* Cached next index */ uint16_t next_index; + uint16_t last_tx; }; +typedef struct rewrite_priv_vars { + union { + struct { + rte_xmm_t xmm1; + }; + struct __rte_packed { + uint16_t next0; + uint16_t next1; + uint16_t next2; + uint16_t next3; + uint16_t last_tx_interface; + uint16_t last_if_feature; + uint16_t actual_feat_mask; + uint16_t speculative_feat_mask; + }; + }; +} rewrite_priv_vars_t; + static struct ip4_rewrite_node_main *ip4_rewrite_nm; #define IP4_REWRITE_NODE_LAST_NEXT(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->next_index) +#define IP4_REWRITE_NODE_LAST_TX(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->last_tx) + #define IP4_REWRITE_NODE_PRIV1_OFF(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->mbuf_priv1_off) -static uint16_t -ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, - void **objs, uint16_t nb_objs) +#define IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->output_feature_arc) + +static __rte_always_inline void +prefetch_mbuf_and_dynfield(struct rte_mbuf *mbuf) { + /* prefetch first cache line required for accessing buf_addr */ + rte_prefetch0((void *)mbuf); +} + +static __rte_always_inline void +check_output_feature_x4(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t flist, + rewrite_priv_vars_t *pvar, struct node_mbuf_priv1 *priv0, + struct node_mbuf_priv1 *priv1, struct node_mbuf_priv1 *priv2, + struct node_mbuf_priv1 *priv3) +{ + uint32_t mask = 0; + uint16_t xor = 0; + + /* + * interface edge's start from 1 and not from 0 as "pkt_drop" + * is next node at 0th index + */ + priv0->if_index = pvar->next0 - 1; + priv1->if_index = pvar->next1 - 1; + priv2->if_index = pvar->next2 - 1; + priv3->if_index = pvar->next3 - 1; + + /* Find out if all packets are sent to last_tx_interface */ + xor = pvar->last_tx_interface ^ priv0->if_index; + xor += priv0->if_index ^ priv1->if_index; + xor += priv1->if_index ^ priv2->if_index; + xor += priv2->if_index ^ priv3->if_index; + + if (likely(!xor)) { + /* copy last interface feature and feature mask */ + priv0->current_feature = priv1->current_feature = + priv2->current_feature = priv3->current_feature = + pvar->last_if_feature; + pvar->actual_feat_mask = pvar->speculative_feat_mask; + } else { + /* create a mask for index which does not have feature + * Also override next edge and if feature enabled, get feature + */ + mask = rte_graph_feature_arc_feature_set(arc, flist, priv0->if_index, + &priv0->current_feature, + &pvar->next0); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv1->if_index, + &priv1->current_feature, + &pvar->next1)) << 1); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv2->if_index, + &priv2->current_feature, + &pvar->next2)) << 2); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv3->if_index, + &priv3->current_feature, + &pvar->next3)) << 3); + + /* + * add last tx and last feature regardless even if feature is + * valid or not + */ + pvar->last_tx_interface = priv3->if_index; + pvar->last_if_feature = priv3->current_feature; + /* Set 0xf if invalid feature to last packet, else 0 */ + pvar->speculative_feat_mask = (priv3->current_feature == + RTE_GRAPH_FEATURE_INVALID) ? ALL_PKT_MASK : 0x0; + pvar->actual_feat_mask = mask; + } +} + +static __rte_always_inline uint16_t +__ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs, + const int dyn, const int check_enabled_features, + struct rte_graph_feature_arc *out_feature_arc, + const rte_graph_feature_rt_list_t flist) +{ + struct node_mbuf_priv1 *priv0 = NULL, *priv1 = NULL, *priv2 = NULL, *priv3 = NULL; struct rte_mbuf *mbuf0, *mbuf1, *mbuf2, *mbuf3, **pkts; struct ip4_rewrite_nh_header *nh = ip4_rewrite_nm->nh; - const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); - uint16_t next0, next1, next2, next3, next_index; - struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; uint16_t n_left_from, held = 0, last_spec = 0; + struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; + rewrite_priv_vars_t pvar; + int64_t fd0, fd1, fd2, fd3; + rte_edge_t fix_spec = 0; void *d0, *d1, *d2, *d3; void **to_next, **from; + uint16_t next_index; rte_xmm_t priv01; rte_xmm_t priv23; int i; - /* Speculative next as last next */ + RTE_SET_USED(fd0); + RTE_SET_USED(fd1); + RTE_SET_USED(fd2); + RTE_SET_USED(fd3); + + /* Initialize speculative variables.*/ + + /* Last interface */ + pvar.last_tx_interface = IP4_REWRITE_NODE_LAST_TX(node->ctx); + /*last next from node ctx*/ next_index = IP4_REWRITE_NODE_LAST_NEXT(node->ctx); + pvar.speculative_feat_mask = ALL_PKT_MASK; + pvar.actual_feat_mask = 0; + rte_prefetch0(nh); pkts = (struct rte_mbuf **)objs; @@ -55,20 +172,47 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, n_left_from = nb_objs; for (i = 0; i < 4 && i < n_left_from; i++) - rte_prefetch0(pkts[i]); + prefetch_mbuf_and_dynfield(pkts[i]); /* Get stream for the speculated next node */ to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + + /* prefetch speculative feature and corresponding data */ + if (check_enabled_features) { + /* + * Get first feature enabled, if any, on last_tx_interface + */ + if (unlikely(rte_graph_feature_arc_first_feature_get(out_feature_arc, + flist, + pvar.last_tx_interface, + (rte_graph_feature_t *) + &pvar.last_if_feature))) { + /* prefetch feature cache line */ + rte_graph_feature_arc_feature_prefetch(out_feature_arc, flist, + pvar.last_if_feature); + + /* prefetch feature data cache line */ + rte_graph_feature_arc_data_prefetch(out_feature_arc, flist, + pvar.last_if_feature, + pvar.last_tx_interface); + /* + * Set speculativa_feat mask to indicate, all 4 packets + * going to feature path + */ + pvar.speculative_feat_mask = 0; + } + } + /* Update Ethernet header of pkts */ while (n_left_from >= 4) { if (likely(n_left_from > 7)) { /* Prefetch only next-mbuf struct and priv area. * Data need not be prefetched as we only write. */ - rte_prefetch0(pkts[4]); - rte_prefetch0(pkts[5]); - rte_prefetch0(pkts[6]); - rte_prefetch0(pkts[7]); + prefetch_mbuf_and_dynfield(pkts[4]); + prefetch_mbuf_and_dynfield(pkts[5]); + prefetch_mbuf_and_dynfield(pkts[6]); + prefetch_mbuf_and_dynfield(pkts[7]); } mbuf0 = pkts[0]; @@ -78,66 +222,138 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 4; n_left_from -= 4; + + /* Copy mbuf private data into private variables */ priv01.u64[0] = node_mbuf_priv1(mbuf0, dyn)->u; priv01.u64[1] = node_mbuf_priv1(mbuf1, dyn)->u; priv23.u64[0] = node_mbuf_priv1(mbuf2, dyn)->u; priv23.u64[1] = node_mbuf_priv1(mbuf3, dyn)->u; - /* Increment checksum by one. */ - priv01.u32[1] += rte_cpu_to_be_16(0x0100); - priv01.u32[3] += rte_cpu_to_be_16(0x0100); - priv23.u32[1] += rte_cpu_to_be_16(0x0100); - priv23.u32[3] += rte_cpu_to_be_16(0x0100); - - /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, - nh[priv01.u16[0]].rewrite_len); - - next0 = nh[priv01.u16[0]].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - ip0->time_to_live = priv01.u16[1] - 1; - ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ - d1 = rte_pktmbuf_mtod(mbuf1, void *); - rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, - nh[priv01.u16[4]].rewrite_len); - - next1 = nh[priv01.u16[4]].tx_node; - ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + - sizeof(struct rte_ether_hdr)); - ip1->time_to_live = priv01.u16[5] - 1; - ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ - d2 = rte_pktmbuf_mtod(mbuf2, void *); - rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, - nh[priv23.u16[0]].rewrite_len); - next2 = nh[priv23.u16[0]].tx_node; - ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + - sizeof(struct rte_ether_hdr)); - ip2->time_to_live = priv23.u16[1] - 1; - ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ - d3 = rte_pktmbuf_mtod(mbuf3, void *); - rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, - nh[priv23.u16[4]].rewrite_len); - - next3 = nh[priv23.u16[4]].tx_node; - ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + - sizeof(struct rte_ether_hdr)); - ip3->time_to_live = priv23.u16[5] - 1; - ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + /* Copy next edge from next hop */ + pvar.next0 = nh[priv01.u16[0]].tx_node; + pvar.next1 = nh[priv01.u16[4]].tx_node; + pvar.next2 = nh[priv23.u16[0]].tx_node; + pvar.next3 = nh[priv23.u16[4]].tx_node; + + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + priv1 = node_mbuf_priv1(mbuf1, dyn); + priv2 = node_mbuf_priv1(mbuf2, dyn); + priv3 = node_mbuf_priv1(mbuf3, dyn); + + /* If feature is enabled, override next edge for each mbuf + * and set node_mbuf_priv data appropriately + */ + check_output_feature_x4(out_feature_arc, flist, + &pvar, priv0, priv1, priv2, priv3); + + /* check_output_feature_x4() returns bit mask which indicates + * which packet is not following feature path, hence normal processing + * has to happen on them + */ + if (unlikely(pvar.actual_feat_mask)) { + if (pvar.actual_feat_mask & 0x1) { + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + } + if (pvar.actual_feat_mask & 0x2) { + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + } + if (pvar.actual_feat_mask & 0x4) { + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + } + if (pvar.actual_feat_mask & 0x8) { + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } + } + } else { + /* Case when no feature is enabled */ + + /* Increment checksum by one. */ + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } /* Enqueue four to next node */ - rte_edge_t fix_spec = - ((next_index == next0) && (next0 == next1) && - (next1 == next2) && (next2 == next3)); + fix_spec = next_index ^ pvar.next0; + fix_spec += next_index ^ pvar.next1; + fix_spec += next_index ^ pvar.next2; + fix_spec += next_index ^ pvar.next3; - if (unlikely(fix_spec == 0)) { + if (unlikely(fix_spec != 0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -146,56 +362,56 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, last_spec = 0; /* next0 */ - if (next_index == next0) { + if (next_index == pvar.next0) { to_next[0] = from[0]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next0, + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); } /* next1 */ - if (next_index == next1) { + if (next_index == pvar.next1) { to_next[0] = from[1]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next1, + rte_node_enqueue_x1(graph, node, pvar.next1, from[1]); } /* next2 */ - if (next_index == next2) { + if (next_index == pvar.next2) { to_next[0] = from[2]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next2, + rte_node_enqueue_x1(graph, node, pvar.next2, from[2]); } /* next3 */ - if (next_index == next3) { + if (next_index == pvar.next3) { to_next[0] = from[3]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next3, + rte_node_enqueue_x1(graph, node, pvar.next3, from[3]); } from += 4; /* Change speculation if last two are same */ - if ((next_index != next3) && (next2 == next3)) { + if ((next_index != pvar.next3) && (pvar.next2 == pvar.next3)) { /* Put the current speculated node */ rte_node_next_stream_put(graph, node, next_index, held); held = 0; /* Get next speculated stream */ - next_index = next3; + next_index = pvar.next3; to_next = rte_node_next_stream_get( graph, node, next_index, nb_objs); } @@ -212,20 +428,41 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 1; n_left_from -= 1; - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, - nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); - - next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + - rte_cpu_to_be_16(0x0100); - chksum += chksum >= 0xffff; - ip0->hdr_checksum = chksum; - ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; - - if (unlikely(next_index ^ next0)) { + pvar.next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + if (pvar.next0 != (pvar.last_tx_interface + 1)) { + priv0->if_index = pvar.next0 - 1; + rte_graph_feature_arc_feature_set(out_feature_arc, flist, + priv0->if_index, + &priv0->current_feature, + &pvar.next0); + pvar.last_tx_interface = priv0->if_index; + pvar.last_if_feature = priv0->current_feature; + } else { + /* current mbuf index is same as last_tx_interface */ + priv0->if_index = pvar.last_tx_interface; + priv0->current_feature = pvar.last_if_feature; + } + } + /* Do the needful if either feature arc is disabled OR + * Invalid feature is present + */ + if (!check_enabled_features || + (priv0->current_feature == RTE_GRAPH_FEATURE_INVALID)) { + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, + nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + + rte_cpu_to_be_16(0x0100); + chksum += chksum >= 0xffff; + ip0->hdr_checksum = chksum; + ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; + } + if (unlikely(next_index ^ pvar.next0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -233,13 +470,15 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, held += last_spec; last_spec = 0; - rte_node_enqueue_x1(graph, node, next0, from[0]); + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); from += 1; } else { last_spec += 1; } } + IP4_REWRITE_NODE_LAST_TX(node->ctx) = pvar.last_tx_interface; + /* !!! Home run !!! */ if (likely(last_spec == nb_objs)) { rte_node_next_stream_move(graph, node, next_index); @@ -255,22 +494,78 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, return nb_objs; } +static uint16_t +ip4_rewrite_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = + rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + rte_graph_feature_rt_list_t flist; + + /* If any feature is enabled on this arc */ + if (unlikely(rte_graph_feature_arc_has_any_feature(arc, &flist))) { + if (flist) + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)1); + else + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)0); + } else { + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); + } + return 0; +} + +static uint16_t +ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); +} + static int ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) { + rte_graph_feature_arc_t feature_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; static bool init_once; RTE_SET_USED(graph); RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_node_ctx) > RTE_NODE_CTX_SZ); + RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_nh_header) != RTE_CACHE_LINE_SIZE); if (!init_once) { node_mbuf_priv1_dynfield_offset = rte_mbuf_dynfield_register( &node_mbuf_priv1_dynfield_desc); if (node_mbuf_priv1_dynfield_offset < 0) return -rte_errno; - init_once = true; + + /* Create ipv4-output feature arc, if not created + */ + if (rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + NULL) < 0) { + if (rte_graph_feature_arc_create(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + RTE_GRAPH_FEATURE_MAX_PER_ARC, + RTE_MAX_ETHPORTS, + ip4_rewrite_node_get(), &feature_arc)) { + return -rte_errno; + } + init_once = true; + } } IP4_REWRITE_NODE_PRIV1_OFF(node->ctx) = node_mbuf_priv1_dynfield_offset; + IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx) = feature_arc; + IP4_REWRITE_NODE_LAST_TX(node->ctx) = UINT16_MAX; node_dbg("ip4_rewrite", "Initialized ip4_rewrite node initialized"); @@ -329,6 +624,7 @@ rte_node_ip4_rewrite_add(uint16_t next_hop, uint8_t *rewrite_data, static struct rte_node_register ip4_rewrite_node = { .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, .name = "ip4_rewrite", /* Default edge i.e '0' is pkt drop */ .nb_edges = 1, diff --git a/lib/node/ip4_rewrite_priv.h b/lib/node/ip4_rewrite_priv.h index 5105ec1d29..52f39601bd 100644 --- a/lib/node/ip4_rewrite_priv.h +++ b/lib/node/ip4_rewrite_priv.h @@ -5,9 +5,11 @@ #define __INCLUDE_IP4_REWRITE_PRIV_H__ #include <rte_common.h> +#include <rte_graph_feature_arc.h> #define RTE_GRAPH_IP4_REWRITE_MAX_NH 64 -#define RTE_GRAPH_IP4_REWRITE_MAX_LEN 56 +#define RTE_GRAPH_IP4_REWRITE_MAX_LEN (sizeof(struct rte_ether_hdr) + \ + (2 * sizeof(struct rte_vlan_hdr))) /** * @internal @@ -15,11 +17,9 @@ * Ipv4 rewrite next hop header data structure. Used to store port specific * rewrite data. */ -struct ip4_rewrite_nh_header { - uint16_t rewrite_len; /**< Header rewrite length. */ +struct __rte_cache_aligned ip4_rewrite_nh_header { uint16_t tx_node; /**< Tx node next index identifier. */ - uint16_t enabled; /**< NH enable flag */ - uint16_t rsvd; + uint16_t rewrite_len; /**< Header rewrite length. */ union { struct { struct rte_ether_addr dst; @@ -30,8 +30,13 @@ struct ip4_rewrite_nh_header { uint8_t rewrite_data[RTE_GRAPH_IP4_REWRITE_MAX_LEN]; /**< Generic rewrite data */ }; + /* used in control path */ + uint8_t enabled; /**< NH enable flag */ }; +_Static_assert(sizeof(struct ip4_rewrite_nh_header) <= (size_t)RTE_CACHE_LINE_SIZE, + "ip4_rewrite_nh_header size must be less or equal to cache line"); + /** * @internal * diff --git a/lib/node/node_private.h b/lib/node/node_private.h index 1de7306792..25db04a9a6 100644 --- a/lib/node/node_private.h +++ b/lib/node/node_private.h @@ -12,6 +12,9 @@ #include <rte_mbuf.h> #include <rte_mbuf_dyn.h> +#include <rte_graph_worker_common.h> +#include <rte_graph_feature_arc_worker.h> + extern int rte_node_logtype; #define RTE_LOGTYPE_NODE rte_node_logtype @@ -29,15 +32,28 @@ extern int rte_node_logtype; */ struct node_mbuf_priv1 { union { - /* IP4/IP6 rewrite */ + /** + * IP4/IP6 rewrite + * only used to pass lookup data from + * ip4-lookup to ip4-rewrite + */ struct { uint16_t nh; uint16_t ttl; uint32_t cksum; }; - uint64_t u; }; + /** + * Feature arc data + */ + struct { + /** interface index */ + uint16_t if_index; + /** feature that current mbuf holds */ + rte_graph_feature_t current_feature; + uint8_t rsvd; + }; }; static const struct rte_mbuf_dynfield node_mbuf_priv1_dynfield_desc = { diff --git a/lib/node/rte_node_ip4_api.h b/lib/node/rte_node_ip4_api.h index 950751a525..da4995662a 100644 --- a/lib/node/rte_node_ip4_api.h +++ b/lib/node/rte_node_ip4_api.h @@ -19,6 +19,7 @@ #include <rte_compat.h> #include <rte_graph.h> +#include <rte_graph_feature_arc_worker.h> #ifdef __cplusplus extern "C" { @@ -67,6 +68,8 @@ struct rte_node_ip4_reassembly_cfg { /**< Node identifier to configure. */ }; +#define RTE_IP4_OUTPUT_FEATURE_ARC_NAME "ipv4-output" + /** * Add ipv4 route to lookup table. * -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v4 4/5] test/graph_feature_arc: add functional tests 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena ` (2 preceding siblings ...) 2024-10-10 13:31 ` [PATCH v4 3/5] graph: add IPv4 output feature arc Nitin Saxena @ 2024-10-10 13:31 ` Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 5/5] docs: add programming guide for feature arc Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 13:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Added functional unit test case for verifying feature arc control plane and fast path APIs How to run: $ echo "graph_feature_arc_autotest" | ./bin/dpdk-test Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++++++++++++ 2 files changed, 1411 insertions(+) create mode 100644 app/test/test_graph_feature_arc.c diff --git a/app/test/meson.build b/app/test/meson.build index e29258e6ec..740fa1bfb4 100644 --- a/app/test/meson.build +++ b/app/test/meson.build @@ -90,6 +90,7 @@ source_file_deps = { 'test_func_reentrancy.c': ['hash', 'lpm'], 'test_graph.c': ['graph'], 'test_graph_perf.c': ['graph'], + 'test_graph_feature_arc.c': ['graph'], 'test_hash.c': ['net', 'hash'], 'test_hash_functions.c': ['hash'], 'test_hash_multiwriter.c': ['hash'], diff --git a/app/test/test_graph_feature_arc.c b/app/test/test_graph_feature_arc.c new file mode 100644 index 0000000000..58b676e215 --- /dev/null +++ b/app/test/test_graph_feature_arc.c @@ -0,0 +1,1410 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "test.h" + +#include <assert.h> +#include <inttypes.h> +#include <signal.h> +#include <stdalign.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> + +#include <rte_errno.h> + +#ifndef RTE_EXEC_ENV_WINDOWS +#include <rte_graph.h> +#include <rte_graph_worker.h> +#include <rte_mbuf.h> +#include <rte_mbuf_dyn.h> +#include <rte_random.h> +#include <rte_graph_feature_arc.h> +#include <rte_graph_feature_arc_worker.h> + +#define MBUFF_SIZE 512 +#define TEST_ARC1_NAME "arc1" +#define TEST_ARC2_NAME "arc2" +#define MAX_INDEXES 10 +#define MAX_FEATURES 5 + +#define SOURCE1 "test_node_arc_source1" +#define INPUT_STATIC "test_node_arc_input_static" +#define OUTPUT_STATIC "test_node_arc_output_static" +#define PKT_FREE_STATIC "test_node_arc_pkt_free_static" +#define ARC1_FEATURE1 "test_node_arc1_feature1" +#define ARC1_FEATURE2 "test_node_arc1_feature2" +#define ARC2_FEATURE1 "test_node_arc2_feature1" +#define ARC2_FEATURE2 "test_node_arc2_feature2" +#define ARC2_FEATURE3 "test_node_arc2_feature3" +#define DUMMY1_STATIC "test_node_arc_dummy1_static" +#define DUMMY2_STATIC "test_node_arc_dummy2_static" + +/* (Node index, Node Name, feature user data base */ +#define FOREACH_TEST_NODE_ARC { \ + R(0, SOURCE1, 64) \ + R(1, INPUT_STATIC, 128) \ + R(2, OUTPUT_STATIC, 256) \ + R(3, PKT_FREE_STATIC, 512) \ + R(4, ARC1_FEATURE1, 1024) \ + R(5, ARC1_FEATURE2, 2048) \ + R(6, ARC2_FEATURE1, 4096) \ + R(7, ARC2_FEATURE2, 8192) \ + R(8, ARC2_FEATURE3, 16384) \ + R(9, DUMMY1_STATIC, 32768) \ + R(10, DUMMY2_STATIC, 65536) \ + } + +/** + * ARC1: Feature arc on ingress interface + * ARC2: Feature arc on egress interface + * XX_static: Static nodes + * XX_featureX: Feature X on arc + * + * -----> ARC1_FEATURE1 + * | | | + * | | v + * | | ARC1_FEATURE2 + * | | | + * | v v + * SOURCE1 ->-----> INPUT_STATIC --> OUTPUT_STATIC -----> PKT_FREE_STATIC + * | | | ^ ^ ^ + * | | | | | | + * | | --> ARC2_FEATURE1 | | + * | | ^ ^ | | + * | | | | | | + * | ----------c-> ARC2_FEATURE2 | + * | | ^ | + * | | | | + * ----------> ARC2_FEATURE3 ------- + */ +const char *node_names_feature_arc[] = { + SOURCE1, INPUT_STATIC, OUTPUT_STATIC, PKT_FREE_STATIC, + ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE2, ARC2_FEATURE3, + DUMMY1_STATIC, DUMMY2_STATIC +}; + +#define MAX_NODES RTE_DIM(node_names_feature_arc) + +/* Function declarations */ +static uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static int +common_node_init(const struct rte_graph *graph, struct rte_node *node); + +typedef struct test_node_priv { + /* index from 0 - MAX_NODES -1 */ + uint8_t node_index; + + /* feature */ + rte_graph_feature_t feature; + + /* rte_graph node id */ + uint32_t node_id; + + rte_graph_feature_arc_t arc; +} test_node_priv_t; + +typedef struct { + rte_graph_feature_t feature; + uint16_t egress_interface; + uint16_t ingress_interface; +} graph_dynfield_t; + +static int graph_dynfield_offset = -1; +static rte_graph_feature_arc_t arcs[RTE_GRAPH_FEATURE_ARC_MAX + 128]; +static struct rte_mbuf mbuf[MAX_NODES + 1][MBUFF_SIZE]; +static void *mbuf_p[MAX_NODES + 1][MBUFF_SIZE]; +static rte_graph_t graph_id = RTE_GRAPH_ID_INVALID; + +const char *node_patterns_feature_arc[] = { + "test_node_arc*" +}; + +static int32_t +compute_unique_user_data(const char *parent, const char *child, uint32_t interface_index) +{ + uint32_t user_data = interface_index; + + RTE_SET_USED(parent); +#define R(idx, node, node_cookie) { \ + if (!strcmp(child, node)) { \ + user_data += node_cookie; \ + } \ + } + + FOREACH_TEST_NODE_ARC +#undef R + + return user_data; +} + +static int +get_edge(struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t count, i; + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +int +common_node_init(const struct rte_graph *graph, struct rte_node *node) +{ + test_node_priv_t *priv = (test_node_priv_t *)node->ctx; + + RTE_SET_USED(graph); + + priv->node_id = node->id; + priv->feature = RTE_GRAPH_FEATURE_INVALID; + priv->arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + +#define R(idx, _name, user_data) { \ + if (!strcmp(node->name, _name)) { \ + priv->node_index = idx; \ + } \ + } + FOREACH_TEST_NODE_ARC +#undef R + + return 0; +} + +uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register source1 = { + .name = SOURCE1, + .process = source1_fn, + .flags = RTE_NODE_SOURCE_F, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {INPUT_STATIC, DUMMY1_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(source1); + +uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register input = { + .name = INPUT_STATIC, + .process = input_fn, + .feat_arc_proc = input_fa_fn, + .nb_edges = 2, + .init = common_node_init, + .next_nodes = {OUTPUT_STATIC, DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(input); + +uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register output = { + .name = OUTPUT_STATIC, + .process = output_fn, + .feat_arc_proc = output_fa_fn, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC, PKT_FREE_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(output); + +uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register pkt_free = { + .name = PKT_FREE_STATIC, + .process = pkt_free_fn, + .feat_arc_proc = pkt_free_fa_fn, + .nb_edges = 1, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(pkt_free); + +uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy1 = { + .name = DUMMY1_STATIC, + .process = dummy1_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(dummy1); + +uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy2 = { + .name = DUMMY2_STATIC, + .process = dummy2_fn, + .nb_edges = 5, + .init = common_node_init, + .next_nodes = { ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, + ARC2_FEATURE2, ARC2_FEATURE3}, +}; +RTE_NODE_REGISTER(dummy2); + +uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature1 = { + .name = ARC1_FEATURE1, + .process = arc1_feature1_fn, + .feat_arc_proc = arc1_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature1); + +uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature2 = { + .name = ARC1_FEATURE2, + .process = arc1_feature2_fn, + .feat_arc_proc = arc1_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature2); + +uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature1 = { + .name = ARC2_FEATURE1, + .process = arc2_feature1_fn, + .feat_arc_proc = arc2_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature1); + +uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature2 = { + .name = ARC2_FEATURE2, + .process = arc2_feature2_fn, + .feat_arc_proc = arc2_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature2); + +uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature3 = { + .name = ARC2_FEATURE3, + .process = arc2_feature3_fn, + .feat_arc_proc = arc2_feature3_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature3); + +static int +create_graph(void) +{ + struct rte_graph_param gconf = { + .socket_id = SOCKET_ID_ANY, + .nb_node_patterns = 1, + .node_patterns = node_patterns_feature_arc, + }; + + graph_id = rte_graph_create("worker0", &gconf); + if (graph_id == RTE_GRAPH_ID_INVALID) { + printf("Graph creation failed with error = %d\n", rte_errno); + return TEST_FAILED; + } + + return TEST_SUCCESS; +} + +static int +__test_create_feature_arc(rte_graph_feature_arc_t *arcs, int max_arcs) +{ + rte_graph_feature_arc_t arc; + const char *sample_arc_name = "sample_arc"; + char arc_name[256]; + int n_arcs; + + /* Create max number of feature arcs first */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + if (rte_graph_feature_arc_create(arc_name, MAX_FEATURES, + MAX_INDEXES, &dummy1, &arcs[n_arcs])) { + printf("Feature arc creation failed for %u\n", n_arcs); + return TEST_FAILED; + } + } + /* Verify feature arc created more than max_arcs must fail */ + if (!rte_graph_feature_arc_create("negative_test_create_arc", MAX_FEATURES, + MAX_INDEXES, &dummy2, &arc)) { + printf("Feature arc creation success for more than max configured: %u\n", n_arcs); + return TEST_FAILED; + } + /* Make sure lookup passes for all feature arcs */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + if (!rte_graph_feature_arc_lookup_by_name(arc_name, &arc)) { + if (arc != arcs[n_arcs]) { + printf("%s: Feature arc lookup mismatch for arc [%p, exp: %p]\n", + arc_name, (void *)arc, (void *)arcs[n_arcs]); + return TEST_FAILED; + } + } else { + printf("Feature arc lookup %s failed after creation\n", arc_name); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_create(void) +{ + int ret = 0, i; + + /* Create arcs with RTE_GRAPH_FEATURE_ARC_MAX */ + ret = __test_create_feature_arc(arcs, RTE_GRAPH_FEATURE_ARC_MAX); + if (ret) { + printf("Feature arc creation test failed for RTE_GRAPH_FEATURE_ARC_MAX arcs\n"); + return TEST_FAILED; + } + /* destroy all arcs via cleanup API*/ + ret = rte_graph_feature_arc_cleanup(); + if (ret) { + printf("Feature arc cleanup failed\n"); + return TEST_FAILED; + } + +#define NUM_FEAT_ARCS 128 + /* create 128 dummy feature arcs */ + ret = rte_graph_feature_arc_init(NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc init failed for NUM_FEAT_ARCS"); + return TEST_FAILED; + } + ret = __test_create_feature_arc(arcs, NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc creation test failed for NUM_FEAT_ARCS\n"); + return TEST_FAILED; + } + /* destroy all of them*/ + for (i = 0; i < NUM_FEAT_ARCS; i++) { + if (rte_graph_feature_arc_destroy(arcs[i])) { + printf("Feature arc destroy failed for %u\n", i); + return TEST_FAILED; + } + } + rte_graph_feature_arc_cleanup(); + + /* Create two arcs as per test plan */ + /* First arc start/source node is node: SOURCE1 */ + if (rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[0])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + /* Duplicate name should fail */ + if (!rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[1])) { + printf("Duplicate feature arc %s creation is not caught\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Second arc start/source node is node: OUTPUT_STATIC */ + if (rte_graph_feature_arc_create(TEST_ARC2_NAME, MAX_FEATURES, + MAX_INDEXES, &output, &arcs[1])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_features_add(void) +{ + rte_graph_feature_t temp; + + /* First feature to SOURCE1 start node -> ARC1_FEATURE1 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature1, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1); + return TEST_FAILED; + } + /* Second feature to SOURCE1 -> ARC1_FEATURE2 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature2, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + /* adding statically connected INPUT_STATIC as a last feature */ + if (rte_graph_feature_add(arcs[0], &input, ARC1_FEATURE2, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC1_NAME, INPUT_STATIC, ARC1_FEATURE2); + return TEST_FAILED; + } + /* First feature to OUTPUT_STATIC start node -> ARC2_FEATURE3 */ + if (rte_graph_feature_add(arcs[1], &arc2_feature3, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Second feature to OUTPUT_STATIC -> ARC2_FEATURE1 and before feature to + * ARC2_FEATURE3 + */ + if (rte_graph_feature_add(arcs[1], &arc2_feature1, NULL, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Add PKT_FREE node as last feature, next to arc2_feature3 */ + if (rte_graph_feature_add(arcs[1], &pkt_free, ARC2_FEATURE3, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Adding feature ARC2_FEATURE2 between ARC2_FEATURE1 and ARC2_FEATURE3. */ + if (rte_graph_feature_add(arcs[1], &arc2_feature2, ARC2_FEATURE1, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s between [%s - %s]\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Now check feature sequencing is correct for both ARCS */ + + /* arc1_feature1 must be first feature to arcs[0] */ + if (!strstr(ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc1_feature2 must be second feature to arcs[0] */ + if (!strstr(ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* Make sure INPUT_STATIC is the last feature in arcs[0] */ + temp = rte_graph_feature_arc_num_features(arcs[0]); + if (!strstr(INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC1_NAME, INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_feature1 must be first feature to arcs[1] */ + if (!strstr(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc2_feature2 must be second feature to arcs[1] */ + if (!strstr(ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_feature3 must be third feature to arcs[1] */ + if (!strstr(ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(2)))) { + printf("%s: %s is not the third feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(2))); + return TEST_FAILED; + } + + /* Make sure PKT_FREE is the last feature in arcs[1] */ + temp = rte_graph_feature_arc_num_features(arcs[1]); + if (!strstr(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + if (get_edge(&arc2_feature1, &pkt_free, NULL)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, PKT_FREE_STATIC); + return TEST_FAILED; + } + + return create_graph(); +} + +static int +test_graph_feature_arc_first_feature_enable(void) +{ + uint32_t n_indexes, n_features, count = 0; + rte_graph_feature_rt_list_t feature_list, temp = 0; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + char *feature_name = NULL; + int32_t user_data; + rte_edge_t edge = ~0; + + arc = rte_graph_feature_arc_get(arcs[0]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 0, + * On interface 1, enable feature 1 and so on so forth + * + * later verify first feature on every interface index is unique + * and check [rte_edge, user_data] retrieved via fast path APIs + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = n_indexes % 3 /* 3 features added to arc1 */; + feature_name = rte_graph_feature_arc_feature_to_name(arcs[0], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_validate(arcs[0], n_indexes, feature_name, 1, true)) { + printf("%s: Feature validate failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* negative test case. enable feature on invalid index */ + if (!n_indexes && !rte_graph_feature_enable(arcs[0], MAX_INDEXES, feature_name, + (int32_t)user_data)) { + printf("%s: Feature %s should not be enabled on invalid index\n", + TEST_ARC1_NAME, feature_name); + return TEST_FAILED; + } + if (rte_graph_feature_enable(arcs[0], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + if (temp == feature_list) { + printf("%s: Activer feature list not switched from %u -> %u\n", + TEST_ARC1_NAME, temp, feature_list); + return TEST_FAILED; + } + temp = feature_list; + if ((count + 1) != rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Number of enabled mismatches [found: %u, exp: %u]\n", + TEST_ARC1_NAME, + rte_graph_feature_arc_num_enabled_features(arcs[0]), + count + 1); + return TEST_FAILED; + } + count++; + } + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Negative test case */ + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, 1); + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC2_FEATURE1, user_data)) { + printf("%s: Invalid feature %s is enabled on index 1\n", + TEST_ARC1_NAME, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Duplicate enable */ + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC1_FEATURE2, user_data)) { + printf("%s: Duplicate feature %s shouldn't be enabled again on index 1\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC1_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc1_feature1; + else if (1 == (n_indexes % 3)) + child = &arc1_feature2; + else + child = &input; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC1_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +verify_feature_sequencing(struct rte_graph_feature_arc *arc) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_t feature; + uint32_t n_indexes; + rte_edge_t edge = ~0; + int32_t user_data; + + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: feature_list can't be obtained\n", + arc->feature_arc_name); + return TEST_FAILED; + } + /* Verify next features on interface 0 and interface 1*/ + for (n_indexes = 0; n_indexes < 2; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: 0\n", + arc->feature_arc_name); + return TEST_FAILED; + } + parent = arc->start_node; + child = rte_graph_feature_arc_feature_to_node(arcs[1], feature); + /* until fast path API reaches last feature i.e pkt_free */ + while (child != &pkt_free) { + fdata = rte_graph_feature_data_get(arc, + rte_graph_feature_get(arc, feature), + n_indexes); + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + arc->feature_arc_name, parent->name, child->name); + return TEST_FAILED; + } + user_data = compute_unique_user_data(parent->name, child->name, n_indexes); + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != user_data) { + printf("%s: Udata mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->user_data, user_data); + return TEST_FAILED; + } + + feature = fdata->next_enabled_feature; + + parent = child; + child = rte_graph_feature_arc_feature_to_node(arcs[1], + fdata->next_enabled_feature); + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_enable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + uint32_t n_indexes, n_features; + rte_graph_feature_t feature; + char *feature_name = NULL; + rte_edge_t edge = ~0; + int32_t user_data; + + arc = rte_graph_feature_arc_get(arcs[1]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[1])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 2, skip feature 1 for later + * On interface 1, enable feature 3 + * On interface 2, enable pkt_free feature + * On interface 3, continue as interface 0 + * + * later enable next feature sequence for interface 0 from feature2 -> pkt_free + * later enable next feature sequence for interface 1 from feature3 -> pkt_free + * + * also later enable feature-1 and see first feature changes for all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = (n_indexes % 3) + 1; /* feature2 to pkt_free are 3 features */ + feature_name = rte_graph_feature_arc_feature_to_name(arcs[1], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + } + /* Retrieve latest feature_list */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + /* verify first feature */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc2_feature2; + else if (1 == (n_indexes % 3)) + child = &arc2_feature3; + else + child = &pkt_free; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + /* add next_features now + * On interface 0, enable feature-3 and pkt_free + * On interface 1, enable pkt_free + * Skip interface 2 + * On interface 3, same as interface 0 + * On interface 4, same as interface 1 + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (0 == (n_indexes % 3)) { + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE3, + compute_unique_user_data(ARC2_FEATURE2, + ARC2_FEATURE3, + n_indexes))) { + printf("%s: Feature enable failed for %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + /* pkt_free on interface-0, 1, 3, 4 and so on */ + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_enable(arcs[1], n_indexes, PKT_FREE_STATIC, + compute_unique_user_data(ARC2_FEATURE3, + PKT_FREE_STATIC, + n_indexes))) { + printf("%s: Feature enable failed %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + /* Enable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE1, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: None first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature != rte_graph_feature_cast(0)) { + printf("%s: First feature mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(0)); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_first_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* Disable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE1)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE1, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature == rte_graph_feature_cast(0)) { + printf("%s: First feature not disabled on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(1)); + return TEST_FAILED; + } + if (!strncmp(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(ARC2_FEATURE1))) { + printf("%s: First feature mismatch on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + ARC2_FEATURE2); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* + * On interface 0, disable feature 2, keep feature3 and pkt_free enabled + * On interface 1, skip interface 1 where feature3 and pkt_free are enabled + * skip interface 2 as only pkt_free is enabled + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!(n_indexes % 3)) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE2)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, n_indexes); + return TEST_FAILED; + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + } + + /** + * Disable feature 3 on all interface 0 and 1 and check first feature + * is pkt_free on all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE3)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + } + /* Make sure pkt_free is first feature for all indexes */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (strncmp(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(PKT_FREE_STATIC))) { + printf("%s: %s is not first feature found on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + PKT_FREE_STATIC); + return TEST_FAILED; + } + } + + /* Disable PKT_FREE_STATIC from all indexes with no feature enabled on any interface */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, PKT_FREE_STATIC)) { + printf("%s: Feat disable failed for %s on index %u\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + /* Make sure no feature is enabled now on any interface */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: Index: %u should not have first feature enabled\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_destroy(void) +{ + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC1_NAME, &arc)) { + printf("Feature arc lookup failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (arc != arcs[0]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC1_NAME, (void *)arc, (void *)arcs[0]); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC2_NAME, &arc)) { + printf("Feature arc lookup success after destroy for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (arc != arcs[1]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC2_NAME, (void *)arc, (void *)arcs[1]); + return TEST_FAILED; + } + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +graph_feature_arc_setup(void) +{ + unsigned long i, j; + + static const struct rte_mbuf_dynfield graph_dynfield_desc = { + .name = "test_graph_dynfield", + .size = sizeof(graph_dynfield_t), + .align = alignof(graph_dynfield_t), + }; + + graph_dynfield_offset = + rte_mbuf_dynfield_register(&graph_dynfield_desc); + if (graph_dynfield_offset < 0) { + printf("Cannot register mbuf field\n"); + return TEST_FAILED; + } + RTE_SET_USED(graph_dynfield_offset); + + for (i = 0; i <= MAX_NODES; i++) { + for (j = 0; j < MBUFF_SIZE; j++) + mbuf_p[i][j] = &mbuf[i][j]; + } + + return TEST_SUCCESS; + +} + +static void +graph_feature_arc_teardown(void) +{ + if (graph_id != RTE_GRAPH_ID_INVALID) + rte_graph_destroy(graph_id); + + rte_graph_feature_arc_cleanup(); +} + +static struct unit_test_suite graph_feature_arc_testsuite = { + .suite_name = "Graph Feature arc library test suite", + .setup = graph_feature_arc_setup, + .teardown = graph_feature_arc_teardown, + .unit_test_cases = { + TEST_CASE(test_graph_feature_arc_create), + TEST_CASE(test_graph_feature_arc_features_add), + TEST_CASE(test_graph_feature_arc_first_feature_enable), + TEST_CASE(test_graph_feature_arc_next_feature_enable), + TEST_CASE(test_graph_feature_arc_first_feature_disable), + TEST_CASE(test_graph_feature_arc_next_feature_disable), + TEST_CASE(test_graph_feature_arc_destroy), + TEST_CASES_END(), /**< NULL terminate unit test array */ + }, +}; + +static int +graph_feature_arc_autotest_fn(void) +{ + return unit_test_suite_runner(&graph_feature_arc_testsuite); +} + +REGISTER_FAST_TEST(graph_feature_arc_autotest, true, true, graph_feature_arc_autotest_fn); +#endif /* !RTE_EXEC_ENV_WINDOWS */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v4 5/5] docs: add programming guide for feature arc 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena ` (3 preceding siblings ...) 2024-10-10 13:31 ` [PATCH v4 4/5] test/graph_feature_arc: add functional tests Nitin Saxena @ 2024-10-10 13:31 ` Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-10 13:31 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Updated graph library guide with feature arc Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/prog_guide/graph_lib.rst | 288 ++++++++++++++++++++ doc/guides/prog_guide/img/feature_arc-1.png | Bin 0 -> 61532 bytes doc/guides/prog_guide/img/feature_arc-2.png | Bin 0 -> 155806 bytes doc/guides/prog_guide/img/feature_arc-3.png | Bin 0 -> 143697 bytes 4 files changed, 288 insertions(+) create mode 100644 doc/guides/prog_guide/img/feature_arc-1.png create mode 100644 doc/guides/prog_guide/img/feature_arc-2.png create mode 100644 doc/guides/prog_guide/img/feature_arc-3.png diff --git a/doc/guides/prog_guide/graph_lib.rst b/doc/guides/prog_guide/graph_lib.rst index ad09bdfe26..0db7b6d99c 100644 --- a/doc/guides/prog_guide/graph_lib.rst +++ b/doc/guides/prog_guide/graph_lib.rst @@ -547,3 +547,291 @@ on success packet is enqueued to ``udp4_input`` node. Hash lookup is performed in ``udp4_input`` node with registered destination port and destination port in UDP packet , on success packet is handed to ``udp_user_node``. + +Feature Arc +----------- +`Feature arc` represents an ordered list of `protocols/features` at a given +networking layer. It is a high level abstraction to connect various `feature` +nodes in `rte_graph` instance and allows seamless packets steering based on the +sequence of enabled features at runtime on each interface. + +`Features` (or feature nodes) are nodes which handle partial or complete +protocol processing in a given direction. For instance, `ipv4-rewrite` and +`IPv4 IPsec encryption` are outbound features while `ipv4-lookup` and `IPv4 +IPsec decryption` are inbound features. Further, `ipv4-rewrite` and `IPv4 +IPsec encryption` can collectively represent a `feature arc` towards egress +direction with ordering constraints that `IPv4 IPsec encryption` must be +performed before `ipv4-rewrite`. Similarly, `IPv4 IPsec decryption` and +`ipv4-lookup` can represent a `feature arc` in an ingress direction. Both of +these `feature arc` can co-exist at an IPv4 layer in egress and ingress +direction respectively. + +A `feature` can be represented by a single node or collection of multiple nodes +performing feature processing collectively. + +.. figure:: img/feature_arc-1.png + :alt: feature-arc-1 + :width: 350px + :align: center + + Feature Arc overview + +Each `feature arc` is associated with a `Start` node from which all features in +a feature arc are connected. A `start` node itself is not a `feature` node but +it is where `first enabled feature` is checked in fast path. In above figure, +`Node-A` represents a `start node`. There may be a `Sink` node as well which +is child node for every feature in an arc. 'Sink` node is responsible of +consuming those packets which are not consumed by intermediate enabled features +between `start` and `sink` node. `Sink` node, if present, is the last enabled +feature in a feature arc. A `feature` node statically connected to `start` node +must also be added via feature arc API, `rte_graph_feature_add()``. Here `Node-B` +acts as a `sink` node which is statically linked to `Node A`. `Feature` nodes +are connected via `rte_graph_feature_add()` which takes care of connecting +all `feature` nodes with each other and start node. + +.. code-block:: bash + :linenos: + :emphasize-lines: 8 + :caption: Node-B statically linked to Node-A + + static struct rte_node_register node_A_node = { + .process = node_A_process_func, + ... + ... + .name = "Node-A", + .next_nodes = + { + [0] = "Node-B", + }, + .nb_edges = 1, + }; + +When multiple features are enabled on an interface, it may be required to steer +packets across `features` in a given order. For instance, if `Feature 1` and +`Feature 2` both are enabled on an interface ``X``, it may be required to send +packets to `Feature-1` before `Feature-2`. Such ordering constraints can be +easily expressed with `feature arc`. In this case, `Feature 1` is called as +``First Feature`` and `Feature 2` is called as ``Next Feature`` to `Feature 1`. + +.. figure:: img/feature_arc-2.png + :alt: feature-arc-2 + :width: 600px + :align: center + + First and Next features and their ordering + +In similar manner, even application specific ``custom features`` can be hooked +to standard nodes. It is to be noted that this `custom feature` hooking to +`feature arc` aware node does not require any code changes. + +It may be obvious by now that `features` enabled on one interface does not +affect packets on other interfaces. In above example, if no feature is +enabled on an interface ``X``, packets destined to interface ``X`` would be +directly sent to `Node-B` from `Node-A`. + +.. figure:: img/feature_arc-3.png + :alt: feature-arc-3 + :width: 550px + :align: center + + Feature-2 consumed/non-consumed packet path + +When a `Feature-X` node receives packets via feature arc, it may decide whether +to ``consume packet`` or send to `next enabled feature`. A node can consume +packet by freeing it, sending it on wire or enqueuing it to hardware queue. If a +packet is not consumed by a `Feature-X` node, it may send to `next enabled +feature` on an interface. In above figure, `Feature-2` nodes are represented to +consume packets. Classic example for a node performing consume and non-consume +operation on packets would be IPsec policy node where all packets with +``protect`` actions are consumed while remaining packets with ``bypass`` +actions are sent to next enabled feature. + +In fast path feature node may require to lookup local data structures for each +interface. For example, retrieving policy database per interface for IPsec +processing. ``rte_graph_feature_enable`` API allows to set application +specific cookie per feature per interface. `Feature data` object maintains this +cookie in fast path for each interface. + +`Feature arc design` allows to enable subsequent features in a control plane +without stopping workers which are accessing feature arc's fast path APIs in +``rte_graph_walk()`` context. However for disabling features require RCU like +scheme for synchronization. + +Programming model +~~~~~~~~~~~~~~~~~ +Feature Arc Objects +^^^^^^^^^^^^^^^^^^^ +Control plane and fast path APIs deals with following objects: + +Feature arc +*********** +``rte_graph_feature_arc_t`` is a handle to feature arc which is created via +``rte_graph_feature_arc_create()``. It is a `uint64_t` size object which can be +saved in feature node's context. This object can be translated to fast path +feature arc object ``struct rte_graph_feature_arc`` which is an input +argument to all fast path APIs. Control plane APIs majorly takes +`rte_graph_feature_arc_t` object as an input. + +Feature List +************ +Each feature arc holds two feature lists: `active` and `passive`. While worker +cores uses `active` list, control plane APIs uses `passive` list for +enabling/disabling a feature on any interface with in a arc. After successful +feature enable/disable, ``rte_graph_feature_enable()``/ +``rte_graph_feature_disable()`` atomically switches passive list to active list +and vice-versa. Most of the fast path APIs takes active list as an argument +(``rte_graph_feature_rt_list_t``), which feature node can obtain in start of +it's `process_func()` via ``rte_graph_feature_arc_has_any_feature()`` (in `start` +node) or ``rte_graph_feature_arc_has_feature()`` (in next feature nodes). + +Each feature list holds RTE_GRAPH_MAX_FEATURES number of features and +associated feature data for every interface index + +Feature +******** +Feature is a data structure which holds `feature data` object for every +interface. It is represented via ``rte_graph_feature_t`` which is a `uint8_t` +size object. Fast path internal structure ``struct rte_graph_feature`` can be +obtained from ``rte_graph_feature_t`` via ``rte_graph_feature_get()`` API. + +In `start` node `rte_graph_feature_arc_first_feature_get()` can be used to get +first enabled `rte_graph_feature_t` object for an interface. `rte_edge` from +`start` node to first enabled feature is provided by +``rte_graph_feature_arc_feature_set()`` API. + +In `feature nodes`, next enabled feature is obtained by providing current feature +as an input to ``rte_graph_feature_arc_next_feature_get()`` API. + +Feature data +************ +Feature data object is maintained per feature per interface which holds +following information in fast path + +- ``rte_edge_t`` to send packet to next enabled feature +- ``Next enabled feature`` on current interface +- ``User_data`` per feature per interface set by application via `rte_graph_feature_enable()` + +Enabling Feature Arc processing +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +By default, feature arc processing is disabled in `rte_graph_create()`. To +enable feature arc processing in fast path, `rte_graph_create()` API should be +invoked with `feature_arc_enable` flag set as `true` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Enabling feature are processing in rte_graph_create() + + struct rte_graph_param graph_conf; + + graph_conf.feature_arc_enable = true; + struct rte_graph *graph = rte_graph_create("graph_name", &graph_conf); + +Further as an optimization technique, `rte_graph_walk()` would call newly added +``feat_arc_proc()`` node callback function (if non-NULL) instead of +``process`` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Feature arc specific node callback function + + static struct rte_node_register ip4_rewrite_node = { + .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, + .name = "ip4_rewrite", + ... + ... + }; + +If `feat_arc_proc` is not provided in node registration, `process_func` would +be called by `rte_graph_walk()` + +Sample Usage +^^^^^^^^^^^^ +.. code-block:: bash + :linenos: + :caption: Feature arc sample usage + + #define MAX_FEATURES 10 + #define MAX_INDEXES 5 + + static uint16_t + feature2_feature_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* features may be enabled */ + } + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc is disabled in rte_graph_create() */ + } + + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc may be enabled or disabled as this process_func() would + * be called for the case when feature arc is enabled in rte_graph_create() + * and also the case when it is disabled + */ + } + + static struct rte_node_register feature2_node = { + .process = feature2_node_process, + .feat_arc_proc = feature2_feature_node_process, + .name = "feature2", + .init = feature2_init_func, + ... + ... + }; + + static struct rte_node_register feature1_node = { + .process = feature1_node_process, + .feat_arc_proc = NULL, + .name = "feature1", + ... + ... + }; + + int worker_cb(void *_em) + { + rte_graph_feature_arc_t arc; + uint32_t user_data; + + rte_graph_feature_arc_lookup_by_name("sample_arc", &arc); + + /* From control thread context (like CLII): + * Enable feature 2 on interface index 4 + */ + if (rte_lcore_id() == rte_get_main_lcore) { + user_data = 0x1234; + rte_graph_feature_enable(arc, 4 /* interface index */, "feature2", user_data); + } else { + while(1) + rte_graph_walk); + } + } + + int main(void) + { + struct rte_graph_param graph_conf; + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_create("sample_arc", MAX_FEATURES, MAX_INDEXES, &arc)) + return -1; + + rte_graph_feature_add(arc, "feature1", NULL, NULL); + rte_graph_feature_add(arc, "feature2", "feature1" /* add feature2 after feature 1*/, NULL); + + /* create graph*/ + ... + ... + graph_conf.feature_arc_enable = true; + + struct rte_graph *graph = rte_graph_create("sample_graph", &graph_conf); + + rte_eal_mp_remote_launch(worker_cb, arg, CALL_MAIN); + } diff --git a/doc/guides/prog_guide/img/feature_arc-1.png b/doc/guides/prog_guide/img/feature_arc-1.png new file mode 100644 index 0000000000000000000000000000000000000000..d518000f42753ea7a1a71668c2735c48fc75c3ed GIT binary patch literal 61532 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 literal 0 HcmV?d00001 diff --git a/doc/guides/prog_guide/img/feature_arc-2.png b/doc/guides/prog_guide/img/feature_arc-2.png new file mode 100644 index 0000000000000000000000000000000000000000..604a67229b29a9c8ace59d28a7324e02c2bc4a89 GIT binary patch literal 155806 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�!fv6yo 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ΐMNRyBlZP4r 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 VhL3#7$M&;o^tFw(Dm5IV{|}tHa(Vy& literal 0 HcmV?d00001 diff --git a/doc/guides/prog_guide/img/feature_arc-3.png b/doc/guides/prog_guide/img/feature_arc-3.png new file mode 100644 index 0000000000000000000000000000000000000000..3991e0aed2933bfa01e9ffad654f4c40a639bc53 GIT binary patch literal 143697 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 JynBzl{|6wHR9pZ6 literal 0 HcmV?d00001 -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v5 0/5] add feature arc in rte_graph 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena ` (4 preceding siblings ...) 2024-10-10 13:31 ` [PATCH v4 5/5] docs: add programming guide for feature arc Nitin Saxena @ 2024-10-14 14:33 ` Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 1/5] graph: add feature arc support Nitin Saxena ` (5 more replies) 5 siblings, 6 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 14:33 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on whether feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Changes in v2: - Added unit tests for feature arc - Fixed issues found in testing - Added new public APIs rte_graph_feature_arc_feature_to_node(), rte_graph_feature_arc_feature_to_name(), rte_graph_feature_arc_num_features() - Added programming guide for feature arc - Added release notes for feature arc Changes in v3: - rte_graph_feature_arc_t typedef from uint64_t to uintptr_t to fix compilation on 32-bit machine - Updated images in .png format - Added ABI change section in release notes - Fixed DPDK CI failures Changes in v4: - Fixed clang build compilations - Captured `feat_arc_proc` function in ABI change section of release notes Changes in v5: - Updated images in .svg format Nitin Saxena (5): graph: add feature arc support graph: add feature arc option in graph create graph: add IPv4 output feature arc test/graph_feature_arc: add functional tests docs: add programming guide for feature arc app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++ doc/guides/prog_guide/graph_lib.rst | 288 ++++ doc/guides/prog_guide/img/feature_arc-1.svg | 277 ++++ doc/guides/prog_guide/img/feature_arc-2.svg | 511 +++++++ doc/guides/prog_guide/img/feature_arc-3.svg | 318 +++++ doc/guides/rel_notes/release_24_11.rst | 17 + lib/graph/graph.c | 1 + lib/graph/graph_feature_arc.c | 1236 ++++++++++++++++ lib/graph/graph_populate.c | 7 +- lib/graph/graph_private.h | 3 + lib/graph/meson.build | 2 + lib/graph/node.c | 2 + lib/graph/rte_graph.h | 3 + lib/graph/rte_graph_feature_arc.h | 431 ++++++ lib/graph/rte_graph_feature_arc_worker.h | 679 +++++++++ lib/graph/version.map | 20 + lib/node/ip4_rewrite.c | 476 +++++-- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 21 files changed, 5622 insertions(+), 98 deletions(-) create mode 100644 app/test/test_graph_feature_arc.c create mode 100644 doc/guides/prog_guide/img/feature_arc-1.svg create mode 100644 doc/guides/prog_guide/img/feature_arc-2.svg create mode 100644 doc/guides/prog_guide/img/feature_arc-3.svg create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v5 1/5] graph: add feature arc support 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena @ 2024-10-14 14:33 ` Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 2/5] graph: add feature arc option in graph create Nitin Saxena ` (4 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 14:33 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena add feature arc to allow dynamic steering of packets across graph nodes based on protocol features enabled on incoming or outgoing interface Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 10 + lib/graph/graph_feature_arc.c | 1236 ++++++++++++++++++++++ lib/graph/meson.build | 2 + lib/graph/rte_graph_feature_arc.h | 431 ++++++++ lib/graph/rte_graph_feature_arc_worker.h | 679 ++++++++++++ lib/graph/version.map | 20 + 6 files changed, 2378 insertions(+) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index a563812d02..1299de886a 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -162,6 +162,16 @@ New Features * Added independent enqueue feature. +* **Added feature arc abstraction in graph library.** + + Feature arc abstraction helps ``rte_graph`` based applications to steer + packets across different node path(s) based on the features (or protocols) + enabled on interfaces. Different feature node paths can be enabled/disabled + at runtime on some or on all interfaces. This abstraction also help + applications to hook their ``custom nodes`` in standard DPDK node paths + without any code changes in the later. + + * Added ``ip4-output`` feature arc processing in ``ip4_rewrite`` node. Removed Items ------------- diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c new file mode 100644 index 0000000000..0f8633c317 --- /dev/null +++ b/lib/graph/graph_feature_arc.c @@ -0,0 +1,1236 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "graph_private.h" +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> + +#define ARC_PASSIVE_LIST(list) (list ^ 0x1) + +#define rte_graph_uint_cast(x) ((unsigned int)x) +#define feat_dbg graph_dbg + +static rte_graph_feature_arc_main_t *__rte_graph_feature_arc_main; + +/* Make sure fast path cache line is compact */ +_Static_assert((offsetof(struct rte_graph_feature_arc, slow_path_variables) + - offsetof(struct rte_graph_feature_arc, fast_path_variables)) + <= RTE_CACHE_LINE_SIZE, + "Fast path feature arc variables exceed cache line size"); + +#define connect_graph_nodes(node1, node2, edge, arc_name) \ + __connect_graph_nodes(node1, node2, edge, arc_name, __LINE__) + +#define FEAT_COND_ERR(cond, fmt, ...) \ + do { \ + if (cond) \ + graph_err(fmt, ##__VA_ARGS__); \ + } while (0) + +/* + * lookup feature name and get control path node_list as well as feature index + * at which it is inserted + */ +static int +feature_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = UINT32_MAX; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + fi++; + } + return -1; +} + +/* Lookup used only during rte_graph_feature_add() */ +static int +feature_add_lookup(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + const char *name; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + name = rte_node_id_to_name(finfo->feature_node->id); + if (!strncmp(name, feat_name, strlen(name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + /* Update slot where new feature can be added */ + if (slot) + *slot = fi; + fi++; + } + + return -1; +} + +/* Get control path node info from provided input feature_index */ +static int +feature_arc_node_info_lookup(struct rte_graph_feature_arc *arc, uint32_t feature_index, + struct rte_graph_feature_node_list **ppfinfo, + const int do_sanity_check) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + if (!ppfinfo) + return -1; + + *ppfinfo = NULL; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + /* Check sanity */ + if (do_sanity_check) + if (finfo->node_index != index) + RTE_VERIFY(0); + if (index == feature_index) { + *ppfinfo = finfo; + return 0; + } + index++; + } + return -1; +} + +/* prepare feature arc after addition of all features */ +static void +prepare_feature_arc_before_first_enable(struct rte_graph_feature_arc *arc) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + rte_atomic_store_explicit(&arc->active_feature_list, 0, + rte_memory_order_relaxed); + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + finfo->node_index = index; + feat_dbg("\t%s prepare: %s added to list at index: %u", arc->feature_arc_name, + finfo->feature_node->name, index); + index++; + } +} + +/* feature arc lookup in array */ +static int +feature_arc_lookup(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (arc == (rte_graph_feature_arc_get(dm->feature_arcs[iter]))) + return 0; + } + return -1; +} + +/* Check valid values for known fields in arc to make sure arc is sane */ +static int check_feature_arc_sanity(rte_graph_feature_arc_t _arc, int iter) +{ +#ifdef FEATURE_ARC_DEBUG + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + RTE_VERIFY(arc->feature_arc_main == __rte_graph_feature_arc_main); + RTE_VERIFY(arc->feature_arc_index == iter); + + RTE_VERIFY(arc->feature_list[0]->indexed_by_features = arc->features[0]); + RTE_VERIFY(arc->feature_list[1]->indexed_by_features = arc->features[1]); + + RTE_VERIFY(rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed) < 2); +#else + RTE_SET_USED(_arc); + RTE_SET_USED(iter); +#endif + return 0; +} + +/* Perform sanity on all arc if any corruption occurred */ +static int do_sanity_all_arcs(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!dm) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (check_feature_arc_sanity(dm->feature_arcs[iter], iter)) + return -1; + } + return 0; +} + +/* get existing edge from parent_node -> child_node */ +static int +get_existing_edge(const char *arc_name, struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t i, count = 0; + + RTE_SET_USED(arc_name); + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +/* create or retrieve already existing edge from parent_node -> child_node */ +static int +__connect_graph_nodes(struct rte_node_register *parent_node, struct rte_node_register *child_node, + rte_edge_t *_edge, char *arc_name, int lineno) +{ + const char *next_node = NULL; + rte_edge_t edge; + + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { + feat_dbg("\t%s/%d: %s[%u]: \"%s\", edge reused", arc_name, lineno, + parent_node->name, edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; + } + + /* Node to be added */ + next_node = child_node->name; + + edge = rte_node_edge_update(parent_node->id, RTE_EDGE_ID_INVALID, &next_node, 1); + + if (edge == RTE_EDGE_ID_INVALID) { + graph_err("edge invalid"); + return -1; + } + edge = rte_node_edge_count(parent_node->id) - 1; + + feat_dbg("\t%s/%d: %s[%u]: \"%s\", new edge added", arc_name, lineno, parent_node->name, + edge, child_node->name); + + if (_edge) + *_edge = edge; + + return 0; +} + +/* feature arc initialization */ +static int +feature_arc_main_init(rte_graph_feature_arc_main_t **pfl, uint32_t max_feature_arcs) +{ + rte_graph_feature_arc_main_t *pm = NULL; + uint32_t i; + size_t sz; + + if (!pfl) + return -1; + + sz = sizeof(rte_graph_feature_arc_main_t) + + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); + + pm = rte_malloc("rte_graph_feature_arc_main", sz, 0); + if (!pm) + return -1; + + memset(pm, 0, sz); + + for (i = 0; i < max_feature_arcs; i++) + pm->feature_arcs[i] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + pm->max_feature_arcs = max_feature_arcs; + + *pfl = pm; + + return 0; +} + +/* feature arc initialization, public API */ +int +rte_graph_feature_arc_init(int max_feature_arcs) +{ + if (!max_feature_arcs) + return -1; + + if (__rte_graph_feature_arc_main) + return -1; + + return feature_arc_main_init(&__rte_graph_feature_arc_main, max_feature_arcs); +} + +/* reset feature list before switching to passive list */ +static void +feature_arc_list_reset(struct rte_graph_feature_arc *arc, uint32_t list_index) +{ + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + uint32_t i, j; + + list = arc->feature_list[list_index]; + feat = arc->features[list_index]; + + /*Initialize variables*/ + memset(feat, 0, arc->feature_size * arc->max_features); + memset(list, 0, arc->feature_list_size); + + /* Initialize feature and feature_data */ + for (i = 0; i < arc->max_features; i++) { + feat = __rte_graph_feature_get(arc, i, list_index); + feat->this_feature_index = i; + + for (j = 0; j < arc->max_indexes; j++) { + fdata = rte_graph_feature_data_get(arc, feat, j); + fdata->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + fdata->next_edge = UINT16_MAX; + fdata->user_data = UINT32_MAX; + } + } + + for (i = 0; i < arc->max_indexes; i++) + list->first_enabled_feature_by_index[i] = RTE_GRAPH_FEATURE_INVALID; +} + +static int +feature_arc_list_init(struct rte_graph_feature_arc *arc, const char *flist_name, + rte_graph_feature_list_t **pplist, + struct rte_graph_feature **ppfeature, uint32_t list_index) +{ + char fname[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + size_t list_size, feat_size, fdata_size; + rte_graph_feature_list_t *list = NULL; + struct rte_graph_feature *feat = NULL; + + list_size = sizeof(struct rte_graph_feature_list) + + (sizeof(list->first_enabled_feature_by_index[0]) * arc->max_indexes); + + list_size = RTE_ALIGN_CEIL(list_size, RTE_CACHE_LINE_SIZE); + + list = rte_malloc(flist_name, list_size, RTE_CACHE_LINE_SIZE); + if (!list) + return -ENOMEM; + + memset(list, 0, list_size); + fdata_size = arc->max_indexes * sizeof(rte_graph_feature_data_t); + + /* Let one feature and its associated data per index capture complete + * cache lines + */ + feat_size = RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature) + fdata_size, + RTE_CACHE_LINE_SIZE); + + snprintf(fname, sizeof(fname), "%s-%s", arc->feature_arc_name, "feat"); + + feat = rte_malloc(fname, feat_size * arc->max_features, RTE_CACHE_LINE_SIZE); + if (!feat) { + rte_free(list); + return -ENOMEM; + } + arc->feature_size = feat_size; + arc->feature_data_size = fdata_size; + arc->feature_list_size = list_size; + + /* Initialize list */ + list->indexed_by_features = feat; + *pplist = list; + *ppfeature = feat; + + feature_arc_list_reset(arc, list_index); + + return 0; +} + +/* free resources allocated in feature_arc_list_init() */ +static void +feature_arc_list_destroy(struct rte_graph_feature_arc *arc, int list_index) +{ + rte_graph_feature_list_t *list = NULL; + + list = arc->feature_list[list_index]; + + rte_free(list->indexed_by_features); + + arc->features[list_index] = NULL; + + rte_free(list); + + arc->feature_list[list_index] = NULL; +} + +int +rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, rte_graph_feature_arc_t *_arc) +{ + char name[2 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + struct rte_graph_feature_data *gfd = NULL; + rte_graph_feature_arc_main_t *dfm = NULL; + struct rte_graph_feature_arc *arc = NULL; + struct rte_graph_feature *df = NULL; + uint32_t iter, j, arc_index; + size_t sz; + + if (!_arc) + SET_ERR_JMP(EINVAL, err, "%s: Invalid _arc", feature_arc_name); + + if (max_features < 2) + SET_ERR_JMP(EINVAL, err, "%s: max_features must be greater than 1", + feature_arc_name); + + if (!start_node) + SET_ERR_JMP(EINVAL, err, "%s: start_node cannot be NULL", + feature_arc_name); + + if (!feature_arc_name) + SET_ERR_JMP(EINVAL, err, "%s: feature_arc name cannot be NULL", + feature_arc_name); + + if (max_features > RTE_GRAPH_FEATURE_MAX_PER_ARC) + SET_ERR_JMP(EAGAIN, err, "%s: number of features cannot be greater than 64", + feature_arc_name); + + /* + * Application hasn't called rte_graph_feature_arc_init(). Initialize with + * default values + */ + if (!__rte_graph_feature_arc_main) { + if (rte_graph_feature_arc_init((int)RTE_GRAPH_FEATURE_ARC_MAX) < 0) { + graph_err("rte_graph_feature_arc_init() failed"); + return -1; + } + } + + /* If name is not unique */ + if (!rte_graph_feature_arc_lookup_by_name(feature_arc_name, NULL)) + SET_ERR_JMP(EINVAL, err, "%s: feature arc name already exists", + feature_arc_name); + + dfm = __rte_graph_feature_arc_main; + + /* threshold check */ + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) + SET_ERR_JMP(EAGAIN, err, "%s: max number (%u) of feature arcs reached", + feature_arc_name, dfm->max_feature_arcs); + + /* Find the free slot for feature arc */ + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { + if (dfm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + break; + } + arc_index = iter; + + if (arc_index >= dfm->max_feature_arcs) { + graph_err("No free slot found for num_feature_arc"); + return -1; + } + + /* This should not happen */ + RTE_VERIFY(dfm->feature_arcs[arc_index] == RTE_GRAPH_FEATURE_ARC_INITIALIZER); + + /* size of feature arc + feature_bit_mask_by_index */ + sz = RTE_ALIGN_CEIL(sizeof(*arc) + (sizeof(uint64_t) * max_indexes), RTE_CACHE_LINE_SIZE); + + arc = rte_malloc(feature_arc_name, sz, RTE_CACHE_LINE_SIZE); + + if (!arc) { + graph_err("malloc failed for feature_arc_create()"); + return -1; + } + + memset(arc, 0, sz); + + /* Initialize rte_graph port group fixed variables */ + STAILQ_INIT(&arc->all_features); + strncpy(arc->feature_arc_name, feature_arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + arc->feature_arc_main = (void *)dfm; + arc->start_node = start_node; + arc->max_features = max_features; + arc->max_indexes = max_indexes; + arc->feature_arc_index = arc_index; + + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist0"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[0], &arc->features[0], 0) < 0) { + rte_free(arc); + graph_err("feature_arc_list_init(0) failed"); + return -1; + } + snprintf(name, sizeof(name), "%s-%s", feature_arc_name, "flist1"); + + if (feature_arc_list_init(arc, name, &arc->feature_list[1], &arc->features[1], 1) < 0) { + feature_arc_list_destroy(arc, 0); + rte_free(arc); + graph_err("feature_arc_list_init(1) failed"); + return -1; + } + + for (iter = 0; iter < arc->max_features; iter++) { + df = rte_graph_feature_get(arc, iter); + for (j = 0; j < arc->max_indexes; j++) { + gfd = rte_graph_feature_data_get(arc, df, j); + gfd->next_enabled_feature = RTE_GRAPH_FEATURE_INVALID; + } + } + dfm->feature_arcs[arc->feature_arc_index] = (rte_graph_feature_arc_t)arc; + dfm->num_feature_arcs++; + + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + + do_sanity_all_arcs(); + + feat_dbg("Feature arc %s[%p] created with max_features: %u and indexes: %u", + feature_arc_name, (void *)arc, max_features, max_indexes); + return 0; + +err: + return -rte_errno; +} + +int +rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *_runs_after, const char *runs_before) +{ + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo = NULL; + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; + char feature_name[3*RTE_GRAPH_FEATURE_ARC_NAMELEN]; + const char *runs_after = NULL; + uint32_t num_feature = 0; + uint32_t slot, add_flag; + rte_edge_t edge = -1; + + /* sanity */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + graph_err("feature arc not created: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (feature_arc_lookup(_arc)) { + graph_err("invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -1; + } + + if (arc->runtime_enabled_features) { + graph_err("adding features after enabling any one of them is not supported"); + return -1; + } + + if ((_runs_after != NULL) && (runs_before != NULL) && + (_runs_after == runs_before)) { + graph_err("runs_after and runs_before are same '%s:%s]", _runs_after, + runs_before); + return -1; + } + + if (!feature_node) { + graph_err("feature_node: %p invalid", feature_node); + return -1; + } + + arc = rte_graph_feature_arc_get(_arc); + + if (feature_node->id == RTE_NODE_ID_INVALID) { + graph_err("Invalid node: %s", feature_node->name); + return -1; + } + + if (!feature_add_lookup(arc, feature_node->name, &finfo, &slot)) { + graph_err("%s feature already added", feature_node->name); + return -1; + } + + if (slot >= arc->max_features) { + graph_err("%s: Max features %u added to feature arc", + arc->feature_arc_name, slot); + return -1; + } + + if (strstr(feature_node->name, arc->start_node->name)) { + graph_err("Feature %s cannot point to itself: %s", feature_node->name, + arc->start_node->name); + return -1; + } + + feat_dbg("%s: adding feature node: %s at feature index: %u", arc->feature_arc_name, + feature_node->name, slot); + + if (connect_graph_nodes(arc->start_node, feature_node, &edge, arc->feature_arc_name)) { + graph_err("unable to connect %s -> %s", arc->start_node->name, feature_node->name); + return -1; + } + + snprintf(feature_name, sizeof(feature_name), "%s-%s-finfo", + arc->feature_arc_name, feature_node->name); + + finfo = rte_malloc(feature_name, sizeof(*finfo), 0); + if (!finfo) { + graph_err("%s/%s: rte_malloc failed", arc->feature_arc_name, feature_node->name); + return -1; + } + + memset(finfo, 0, sizeof(*finfo)); + + finfo->feature_arc = (void *)arc; + finfo->feature_node = feature_node; + finfo->edge_to_this_feature = edge; + arc->runtime_enabled_features = 0; + + /* + * if no constraints given and provided feature is not the first feature, + * explicitly set "runs_after" as last_feature. Handles the case: + * + * add(f1, NULL, NULL); + * add(f2, NULL, NULL); + */ + num_feature = rte_graph_feature_arc_num_features(_arc); + if (!_runs_after && !runs_before && num_feature) + runs_after = rte_graph_feature_arc_feature_to_name(_arc, num_feature - 1); + else + runs_after = _runs_after; + + /* Check for before and after constraints */ + if (runs_before) { + /* runs_before sanity */ + if (feature_lookup(arc, runs_before, &before_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid before feature name: %s", runs_before); + + if (!before_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_before %s does not exist", runs_before); + + /* + * Starting from 0 to runs_before, continue connecting edges + */ + add_flag = 1; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (!add_flag) + /* Nodes after seeing "runs_before", finfo connects to temp*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, + NULL, arc->feature_arc_name); + /* + * As soon as we see runs_before. stop adding edges + */ + if (!strncmp(temp->feature_node->name, runs_before, + RTE_GRAPH_NAMESIZE)) { + if (!connect_graph_nodes(finfo->feature_node, temp->feature_node, + &edge, arc->feature_arc_name)) + add_flag = 0; + } + + if (add_flag) + /* Nodes before seeing "run_before" are connected to finfo */ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + } + } + + if (runs_after) { + if (feature_lookup(arc, runs_after, &after_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid after feature_name %s", runs_after); + + if (!after_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_after %s does not exist", runs_after); + + /* Starting from runs_after to end continue connecting edges */ + add_flag = 0; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (add_flag) + /* We have already seen runs_after now */ + /* Add all features as next node to current feature*/ + connect_graph_nodes(finfo->feature_node, temp->feature_node, NULL, + arc->feature_arc_name); + else + /* Connect initial nodes to newly added node*/ + connect_graph_nodes(temp->feature_node, finfo->feature_node, NULL, + arc->feature_arc_name); + + /* as soon as we see runs_after. start adding edges + * from next iteration + */ + if (!strncmp(temp->feature_node->name, runs_after, RTE_GRAPH_NAMESIZE)) + add_flag = 1; + } + + /* add feature next to runs_after */ + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, next_feature); + } else { + if (before_finfo) { + /* add finfo before "before_finfo" element in the list */ + after_finfo = NULL; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (before_finfo == temp) { + if (after_finfo) + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, + finfo, next_feature); + else + STAILQ_INSERT_HEAD(&arc->all_features, finfo, + next_feature); + + return 0; + } + after_finfo = temp; + } + } else { + /* Very first feature just needs to be added to list */ + STAILQ_INSERT_TAIL(&arc->all_features, finfo, next_feature); + } + } + + return 0; + +finfo_free: + rte_free(finfo); + + return -1; +} + +int +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot; + + if (!feature_lookup(arc, feature_name, &finfo, &slot)) { + *feat = (rte_graph_feature_t) slot; + return 0; + } + + return -1; +} + +int +rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int is_enable_disable, bool emit_logs) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + rte_graph_feature_rt_list_t active_list; + struct rte_graph_feature *gf = NULL; + uint32_t slot; + + /* validate _arc */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + FEAT_COND_ERR(emit_logs, "invalid feature arc: 0x%016" PRIx64, (uint64_t)_arc); + return -EINVAL; + } + + /* validate index */ + if (index >= arc->max_indexes) { + FEAT_COND_ERR(emit_logs, "%s: Invalid provided index: %u >= %u configured", + arc->feature_arc_name, index, arc->max_indexes); + return -1; + } + + /* validate feature_name is already added or not */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) { + FEAT_COND_ERR(emit_logs, "%s: No feature %s added", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + if (!finfo) { + FEAT_COND_ERR(emit_logs, "%s: No feature: %s found", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + /* slot should be in valid range */ + if (slot >= arc->max_features) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid free slot %u(max=%u) for feature", + arc->feature_arc_name, feature_name, slot, arc->max_features); + return -EINVAL; + } + + /* slot should be in range of 0 - 63 */ + if (slot > (RTE_GRAPH_FEATURE_MAX_PER_ARC - 1)) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid slot: %u", arc->feature_arc_name, + feature_name, slot); + return -EINVAL; + } + + if (finfo->node_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s/%s: lookup slot mismatch for finfo idx: %u and lookup slot: %u", + arc->feature_arc_name, feature_name, finfo->node_index, slot); + return -1; + } + + active_list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + + /* Get feature from active list */ + gf = __rte_graph_feature_get(arc, slot, ARC_PASSIVE_LIST(active_list)); + if (gf->this_feature_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s: %s rcvd feature_idx: %u does not match with saved: %u", + arc->feature_arc_name, feature_name, slot, gf->this_feature_index); + return -1; + } + + if (is_enable_disable && (arc->feature_bit_mask_by_index[index] & + RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s already enabled on index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + if (!is_enable_disable && !arc->runtime_enabled_features) { + FEAT_COND_ERR(emit_logs, "%s: No feature enabled to disable", + arc->feature_arc_name); + return -1; + } + + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s not enabled in bitmask for index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + return 0; +} + +/* + * Before switch to passive list, user_data needs to be copied from active list to passive list + */ +static void +copy_fastpath_user_data(struct rte_graph_feature_arc *arc, uint16_t dest_list_index, + uint16_t src_list_index) +{ + rte_graph_feature_data_t *sgfd = NULL, *dgfd = NULL; + struct rte_graph_feature *sgf = NULL, *dgf = NULL; + uint32_t i, j; + + for (i = 0; i < arc->max_features; i++) { + sgf = __rte_graph_feature_get(arc, i, src_list_index); + dgf = __rte_graph_feature_get(arc, i, dest_list_index); + for (j = 0; j < arc->max_indexes; j++) { + sgfd = rte_graph_feature_data_get(arc, sgf, j); + dgfd = rte_graph_feature_data_get(arc, dgf, j); + dgfd->user_data = sgfd->user_data; + } + } +} +/* + * Fill fast path information like + * - next_edge + * - next_enabled_feature + */ +static void +refill_feature_fastpath_data(struct rte_graph_feature_arc *arc, uint16_t list_index) +{ + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; + uint32_t fi = UINT32_MAX, di = UINT32_MAX, prev_fi = UINT32_MAX; + struct rte_graph_feature *gf = NULL, *prev_gf = NULL; + rte_graph_feature_list_t *flist = NULL; + rte_edge_t edge = UINT16_MAX; + uint64_t bitmask = 0; + + flist = arc->feature_list[list_index]; + + for (di = 0; di < arc->max_indexes; di++) { + bitmask = arc->feature_bit_mask_by_index[di]; + prev_fi = RTE_GRAPH_FEATURE_INVALID; + /* for each feature set for index, set fast path data */ + while (rte_bsf64_safe(bitmask, &fi)) { + gf = __rte_graph_feature_get(arc, fi, list_index); + gfd = rte_graph_feature_data_get(arc, gf, di); + RTE_VERIFY(!feature_arc_node_info_lookup(arc, fi, &finfo, 1)); + + /* If previous feature_index was valid in last loop */ + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { + prev_gf = __rte_graph_feature_get(arc, prev_fi, list_index); + prev_gfd = rte_graph_feature_data_get(arc, prev_gf, di); + /* + * Get edge of previous feature node connecting + * to this feature node + */ + RTE_VERIFY(!feature_arc_node_info_lookup(arc, prev_fi, + &prev_finfo, 1)); + if (!get_existing_edge(arc->feature_arc_name, + prev_finfo->feature_node, + finfo->feature_node, &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (%u->%u)%s[%u] = %s", + arc->feature_arc_name, list_index, di, + prev_gfd->user_data, prev_fi, fi, + prev_finfo->feature_node->name, + edge, finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* + * Fill current feature as next enabled + * feature to previous one + */ + prev_gfd->next_enabled_feature = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* On first feature edge of the node to be added */ + if (fi == rte_bsf64(arc->feature_bit_mask_by_index[di])) { + if (!get_existing_edge(arc->feature_arc_name, arc->start_node, + finfo->feature_node, + &edge)) { + feat_dbg("\t[%s/%u/di:%2u,cookie:%u]: (->%u)%s[%u]=%s", + arc->feature_arc_name, list_index, di, + gfd->user_data, fi, + arc->start_node->name, edge, + finfo->feature_node->name); + /* Copy feature index for next iteration*/ + gfd->next_edge = edge; + prev_fi = fi; + /* Set first feature set array for index*/ + flist->first_enabled_feature_by_index[di] = + (rte_graph_feature_t)fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* Clear current feature index */ + bitmask &= ~RTE_BIT64(fi); + } + } +} + +int +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const + char *feature_name, int32_t user_data) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_rt_list_t passive_list, active_list; + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Enabling feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (!arc->runtime_enabled_features) + prepare_feature_arc_before_first_enable(arc); + + if (rte_graph_feature_validate(_arc, index, feature_name, 1, true)) + return -1; + + /** This should not fail as validate() has passed */ + if (feature_lookup(arc, feature_name, &finfo, &slot)) + RTE_VERIFY(0); + + active_list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + + passive_list = ARC_PASSIVE_LIST(active_list); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, active_list); + + /* Set current user-data */ + gfd->user_data = user_data; + + /* Set bitmask in control path bitmask */ + rte_bit_relaxed_set64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + refill_feature_fastpath_data(arc, passive_list); + + /* If first time feature getting enabled */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[active_list], + rte_memory_order_relaxed); + + /* On very first feature enable instance */ + if (!finfo->ref_count) + bitmask |= RTE_BIT64(slot); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], + bitmask, rte_memory_order_relaxed); + + /* Slow path updates */ + arc->runtime_enabled_features++; + + /* Increase feature node info reference count */ + finfo->ref_count++; + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After enable, switched active feature list to %u", + arc->feature_arc_name, feature_name, passive_list); + + return 0; +} + +int +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_rt_list_t passive_list, active_list; + struct rte_graph_feature_data *gfd = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature *gf = NULL; + uint64_t bitmask; + uint32_t slot; + + feat_dbg("%s: Disable feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (rte_graph_feature_validate(_arc, index, feature_name, 0, true)) + return -1; + + if (feature_lookup(arc, feature_name, &finfo, &slot)) + return -1; + + active_list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + + passive_list = ARC_PASSIVE_LIST(active_list); + + gf = __rte_graph_feature_get(arc, slot, passive_list); + gfd = rte_graph_feature_data_get(arc, gf, index); + + feat_dbg("\t%s/%s: index: %u, passive list: %u, feature index: %u", + arc->feature_arc_name, feature_name, index, passive_list, slot); + + rte_bit_relaxed_clear64(rte_graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + + /* Reset feature list */ + feature_arc_list_reset(arc, passive_list); + + /* Copy user-data */ + copy_fastpath_user_data(arc, passive_list, active_list); + + /* Reset current user-data */ + gfd->user_data = ~0; + + refill_feature_fastpath_data(arc, passive_list); + + finfo->ref_count--; + arc->runtime_enabled_features--; + + /* If no feature enabled, reset feature in u64 fast path bitmask */ + bitmask = rte_atomic_load_explicit(&arc->feature_enable_bitmask[active_list], + rte_memory_order_relaxed); + + /* When last feature is disabled */ + if (!finfo->ref_count) + bitmask &= ~(RTE_BIT64(slot)); + + rte_atomic_store_explicit(&arc->feature_enable_bitmask[passive_list], bitmask, + rte_memory_order_relaxed); + + /* Store release semantics for active_list update */ + rte_atomic_store_explicit(&arc->active_feature_list, passive_list, + rte_memory_order_release); + + feat_dbg("%s/%s: After disable, switched active feature list to %u", + arc->feature_arc_name, feature_name, passive_list); + + return 0; +} + +int +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_node_list *node_info = NULL; + + while (!STAILQ_EMPTY(&arc->all_features)) { + node_info = STAILQ_FIRST(&arc->all_features); + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); + rte_free(node_info); + } + feature_arc_list_destroy(arc, 0); + feature_arc_list_destroy(arc, 1); + + dm->feature_arcs[arc->feature_arc_index] = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + rte_free(arc); + + do_sanity_all_arcs(); + + return 0; +} + +int +rte_graph_feature_arc_cleanup(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + rte_graph_feature_arc_destroy((rte_graph_feature_arc_t)dm->feature_arcs[iter]); + } + rte_free(dm); + + __rte_graph_feature_arc_main = NULL; + + return 0; +} + +int +rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_arc *arc = NULL; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + if (_arc) + *_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == RTE_GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + arc = rte_graph_feature_arc_get(dm->feature_arcs[iter]); + + if ((strstr(arc->feature_arc_name, arc_name)) && + (strlen(arc->feature_arc_name) == strlen(arc_name))) { + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc; + return 0; + } + } + + return -1; +} + +uint32_t +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + return arc->runtime_enabled_features; +} + +uint32_t +rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t count = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) + count++; + + return count; +} + +char * +rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node->name; + + return NULL; +} + +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!feature_arc_node_info_lookup(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_node; + + return NULL; + +} diff --git a/lib/graph/meson.build b/lib/graph/meson.build index 0cb15442ab..d916176fb7 100644 --- a/lib/graph/meson.build +++ b/lib/graph/meson.build @@ -14,11 +14,13 @@ sources = files( 'graph_debug.c', 'graph_stats.c', 'graph_populate.c', + 'graph_feature_arc.c', 'graph_pcap.c', 'rte_graph_worker.c', 'rte_graph_model_mcore_dispatch.c', ) headers = files('rte_graph.h', 'rte_graph_worker.h') +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') indirect_headers += files( 'rte_graph_model_mcore_dispatch.h', 'rte_graph_model_rtc.h', diff --git a/lib/graph/rte_graph_feature_arc.h b/lib/graph/rte_graph_feature_arc.h new file mode 100644 index 0000000000..1615f8e1c8 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc.h @@ -0,0 +1,431 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ +#define _RTE_GRAPH_FEATURE_ARC_H_ + +#include <assert.h> +#include <errno.h> +#include <signal.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <rte_common.h> +#include <rte_compat.h> +#include <rte_debug.h> +#include <rte_graph.h> +#include <rte_graph_worker.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @file + * + * rte_graph_feature_arc.h + * + * Define APIs and structures/variables with respect to feature arc + * + * - Feature arc(s) + * - Feature(s) + * + * A feature arc represents an ordered list of features/protocol-nodes at a + * given networking layer. Feature arc provides a high level abstraction to + * connect various *rte_graph* nodes, designated as *feature nodes*, and + * allowing steering of packets across these feature nodes fast path processing + * in a generic manner. In a typical network stack, often a protocol or feature + * must be first enabled on a given interface, before any packet is steered + * towards it for feature processing. For eg: incoming IPv4 packets are sent to + * routing sub-system only after a valid IPv4 address is assigned to the + * received interface. In other words, often packets needs to be steered across + * features not based on the packet content but based on whether a feature is + * enable or disable on a given incoming/outgoing interface. Feature arc + * provides mechanism to enable/disable feature(s) on each interface at runtime + * and allow seamless packet steering across runtime enabled feature nodes in + * fast path. + * + * Feature arc also provides a way to steer packets from standard nodes to + * custom/user-defined *feature nodes* without any change in standard node's + * fast path functions + * + * On a given interface multiple feature(s) might be enabled in a particular + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" + * features may be enabled on "eth0" interface in "L3-output" feature arc. + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" + * interface in same "L3-output" feature arc. + * + * When multiple features are present in a given feature arc, its imperative + * to allow each feature processing in a particular sequential order. For + * instance, in "L3-input" feature arc it may be required to run "IPsec + * input" feature first, for packet decryption, before "ip-lookup". So a + * sequential order must be maintained among features present in a feature arc. + * + * Features are enabled/disabled multiple times at runtime to some or all + * available interfaces present in the system. Enable/disabling features on one + * interface is independent of other interface. + * + * A given feature might consume packet (if it's configured to consume) or may + * forward it to next enabled feature. For instance, "IPsec input" feature may + * consume/drop all packets with "Protect" policy action while all packets with + * policy action as "Bypass" may be forwarded to next enabled feature (with in + * same feature arc) + * + * This library facilitates rte graph based applications to steer packets in + * fast path to different feature nodes with-in a feature arc and support all + * functionalities described above + * + * In order to use feature-arc APIs, applications needs to do following in + * control path: + * - Initialize feature arc library via rte_graph_feature_arc_init() + * - Create feature arc via rte_graph_feature_arc_create() + * - *Before calling rte_graph_create()*, features must be added to feature-arc + * via rte_graph_feature_add(). rte_graph_feature_add() allows adding + * features in a sequential order with "runs_after" and "runs_before" + * constraints. + * - Post rte_graph_create(), features can be enabled/disabled at runtime on + * any interface via rte_graph_feature_enable()/rte_graph_feature_disable() + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() + * + * In fast path, APIs are provided to steer packets towards feature path from + * - start_node (provided as an argument to rte_graph_feature_arc_create()) + * - feature nodes (which are added via rte_graph_feature_add()) + * + * For typical steering of packets across feature nodes, application required + * to know "rte_edges" which are saved in feature data object. Feature data + * object is unique for every interface per feature with in a feature arc. + * + * When steering packets from start_node to feature node: + * - rte_graph_feature_arc_first_feature_get() provides first enabled feature. + * - Next rte_edge from start_node to first enabled feature can be obtained via + * rte_graph_feature_arc_feature_set() + * + * rte_mbuf can carry [current feature, interface index] from start_node of an + * arc to other feature nodes + * + * At the time of feature enable(rte_graph_feature_enable), application can set + * 32-bit unique user_data specific to feature per interface. In fast path + * user_data can be retrieved via rte_graph_feature_user_data_get(). User data + * can hold application specific cookie like IPsec policy database index, FIB + * table index etc. + * + * If feature node is not consuming packet, next enabled feature and next + * rte_edge can be obtained via rte_graph_feature_arc_next_feature_get() + * + * It is application responsibility to ensure that at-least *last feature*(or + * sink feature) must be enabled from where packet can exit feature-arc path, + * if *NO* intermediate feature is consuming the packet and it has reached till + * the end of feature arc path + * + * It is recommended that all features *MUST* be added to feature arc before + * calling `rte_graph_create()`. Addition of features after + * `rte_graph_create()` may not work functionally. + * Although,rte_graph_feature_enable()/rte_graph_feature_disable() should be + * called after `rte_graph_create()` in control plane. + * + * Synchronization among cores + * --------------------------- + * Subsequent calls to rte_graph_feature_enable() is allowed while worker cores + * are processing in rte_graph_walk() loop. However, for + * rte_graph_feature_disable() application must use RCU based synchronization + */ + +/** Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER ((rte_graph_feature_arc_t)UINT64_MAX) + +/** Max number of feature arcs which can be created */ +#define RTE_GRAPH_FEATURE_ARC_MAX 64 + +/** Max number of features supported in a given feature arc */ +#define RTE_GRAPH_FEATURE_MAX_PER_ARC 64 + +/** Length of feature arc name */ +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE + +/** @internal */ +#define rte_graph_feature_cast(x) ((rte_graph_feature_t)x) + +/**< Initializer value for rte_graph_feature_arc_t */ +#define RTE_GRAPH_FEATURE_INVALID rte_graph_feature_cast(UINT8_MAX) + +/** rte_graph feature arc object */ +typedef uintptr_t rte_graph_feature_arc_t; + +/** rte_graph feature object */ +typedef uint8_t rte_graph_feature_t; + +/** runtime active feature list index with in feature arc*/ +typedef uint16_t rte_graph_feature_rt_list_t; + +/** per feature arc monotonically increasing counter to synchronize fast path APIs */ +typedef uint16_t rte_graph_feature_counter_t; + +/** + * Initialize feature arc subsystem + * + * @param max_feature_arcs + * Maximum number of feature arcs required to be supported + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_init(int max_feature_arcs); + +/** + * Create a feature arc + * + * @param feature_arc_name + * Feature arc name with max length of @ref RTE_GRAPH_FEATURE_ARC_NAMELEN + * @param max_features + * Maximum number of features to be supported in this feature arc + * @param max_indexes + * Maximum number of interfaces/ports/indexes to be supported + * @param start_node + * Base node where this feature arc's features are checked in fast path + * @param[out] _arc + * Feature arc object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_create(const char *feature_arc_name, int max_features, int max_indexes, + struct rte_node_register *start_node, + rte_graph_feature_arc_t *_arc); + +/** + * Get feature arc object with name + * + * @param arc_name + * Feature arc name provided to successful @ref rte_graph_feature_arc_create + * @param[out] _arc + * Feature arc object returned. Valid only when API returns SUCCESS + * + * @return + * 0: Success + * <0: Failure. + */ +__rte_experimental +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc); + +/** + * Add a feature to already created feature arc. For instance + * + * 1. Add first feature node: "ipv4-input" to input arc + * rte_graph_feature_add(ipv4_input_arc, "ipv4-input", NULL, NULL); + * + * 2. Add "ipsec-input" feature node after "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipsec-input", "ipv4-input", NULL); + * + * 3. Add "ipv4-pre-classify-input" node before "ipv4-input" feature + * rte_graph_feature_add(ipv4_input_arc, "ipv4-pre-classify-input"", NULL, "ipv4-input"); + * + * 4. Add "acl-classify-input" node after ipv4-input but before ipsec-input + * rte_graph_feature_add(ipv4_input_arc, "acl-classify-input", "ipv4-input", "ipsec-input"); + * + * @param _arc + * Feature arc handle returned from @ref rte_graph_feature_arc_create() + * @param feature_node + * Graph node representing feature. On success, feature_node is next_node of + * feature_arc->start_node + * @param runs_after + * Add this feature_node after already added "runs_after". Creates + * start_node -> runs_after -> this_feature sequence + * @param runs_before + * Add this feature_node before already added "runs_before". Creates + * start_node -> this_feature -> runs_before sequence + * + * <I> Must be called before rte_graph_create() </I> + * <I> rte_graph_feature_add() is not allowed after call to + * rte_graph_feature_enable() so all features must be added before they can be + * enabled </I> + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_add(rte_graph_feature_arc_t _arc, struct rte_node_register *feature_node, + const char *runs_after, const char *runs_before); + +/** + * Enable feature within a feature arc + * + * Must be called after @b rte_graph_create(). + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param user_data + * Application specific data which is retrieved in fast path + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + int32_t user_data); + +/** + * Validate whether subsequent enable/disable feature would succeed or not. + * API is thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param is_enable_disable + * If 1, validate whether subsequent @ref rte_graph_feature_enable would pass or not + * If 0, validate whether subsequent @ref rte_graph_feature_disable would pass or not + * @param emit_logs + * If passed true, emit error logs when failure is returned + * If passed false, do not emit error logs when failure is returned + * + * @return + * 0: Subsequent enable/disable API would pass + * <0: Subsequent enable/disable API would not pass + */ +__rte_experimental +int rte_graph_feature_validate(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, int is_enable_disable, bool emit_logs); + +/** + * Disable already enabled feature within a feature arc + * + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name); + +/** + * Get rte_graph_feature_t object from feature name + * + * @param arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param feature_name + * Feature name provided to @ref rte_graph_feature_add + * @param[out] feature + * Feature object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_lookup(rte_graph_feature_arc_t arc, const char *feature_name, + rte_graph_feature_t *feature); + +/** + * Delete feature_arc object + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); + +/** + * Cleanup all feature arcs + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_cleanup(void); + +/** + * Slow path API to know how many features are added (NOT enabled) within a + * feature arc + * + * @param _arc + * Feature arc object + * + * @return: Number of added features to arc + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to know how many features are currently enabled within a + * feature arc across all indexes. If a single feature is enabled on all interfaces, + * this API would return "number_of_interfaces" as count (but not "1") + * + * @param _arc + * Feature arc object + * + * @return: Number of enabled features across all indexes + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to get feature node name from rte_graph_feature_t object + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: Name of the feature node + */ +__rte_experimental +char *rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + +/** + * Slow path API to get corresponding struct rte_node_register * from + * rte_graph_feature_t + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: struct rte_node_register * of feature node on SUCCESS else NULL + */ +__rte_experimental +struct rte_node_register * +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/lib/graph/rte_graph_feature_arc_worker.h b/lib/graph/rte_graph_feature_arc_worker.h new file mode 100644 index 0000000000..9b720e366c --- /dev/null +++ b/lib/graph/rte_graph_feature_arc_worker.h @@ -0,0 +1,679 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ + +#include <stddef.h> +#include <rte_graph_feature_arc.h> +#include <rte_bitops.h> + +/** + * @file + * + * rte_graph_feature_arc_worker.h + * + * Defines fast path structure + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/** @internal + * + * Slow path feature node info list + */ +struct rte_graph_feature_node_list { + /** Next feature */ + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; + + /** node representing feature */ + struct rte_node_register *feature_node; + + /** How many indexes/interfaces using this feature */ + int32_t ref_count; + + /* node_index in list (after feature_enable())*/ + uint32_t node_index; + + /** Back pointer to feature arc */ + void *feature_arc; + + /** rte_edge_t to this feature node from feature_arc->start_node */ + rte_edge_t edge_to_this_feature; +}; + +/** + * Feature data object: + * + * Feature data stores information to steer packets for: + * - a feature with in feature arc + * - Index i.e. Port/Interface index + * + * Each feature data object holds + * - User data of current feature retrieved via rte_graph_feature_user_data_get() + * - next_edge is used in two conditions when packet to be steered from + * -- start_node to first enabled feature on an interface index + * -- current feature node to next enabled feature on an interface index + * - next_enabled_feature on interface index, if current feature is not + * consuming packet + * + * While user_data corresponds to current enabled feature node however + * next_edge and next_enabled_feature corresponds to next enabled feature + * node on an interface index + * + * First enabled feature on interface index can be retrieved via: + * - rte_graph_feature_first_feature_get() if arc's start_node is trying to + * steer packet from start_node to first enabled feature on interface index + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_next_feature_get() if current node is not arc's + * start_node. Input to rte_graph_feature_next_feature_get() is current + * enabled feature and interface index + */ +typedef struct __rte_packed rte_graph_feature_data { + /** edge from current node to next enabled feature */ + rte_edge_t next_edge; + + union { + uint16_t reserved; + struct { + /** next enabled feature on index from current feature */ + rte_graph_feature_t next_enabled_feature; + }; + }; + + /** user_data set by application in rte_graph_feature_enable() for + * - current feature + * - interface index + */ + int32_t user_data; +} rte_graph_feature_data_t; + +/** + * Feature object + * + * Feature object holds feature data object for every index/interface within + * feature + * + * Within a given arc and interface index, first feature object can be + * retrieved in arc's start_node via: + * - rte_graph_feature_arc_first_feature_get() + * + * Feature data information can be retrieved for first feature in start node via + * - rte_graph_feature_arc_feature_set() + * + * Next enabled feature on interface index can be retrieved via: + * - rte_graph_feature_arc_next_feature_get() + * + * Typically application stores rte_graph_feature_t object in rte_mbuf. + * rte_graph_feature_t can be translated to (struct rte_graph_feature *) via + * rte_graph_feature_get() in fast path. Further if needed, feature data for an + * index within a feature can be retrieved via rte_graph_feature_data_get() + */ +struct __rte_cache_aligned rte_graph_feature { + /** feature index or rte_graph_feature_t */ + uint16_t this_feature_index; + + /* + * Array of size arc->feature_data_size + * + * <----------------- Feature --------------------------> + * [data-index-0][data-index-1]...[data-index-max_index-1] + * + * sizeof(feature_data_by_index[0] == sizeof(rte_graph_feature_data_t) + * + */ + uint8_t feature_data_by_index[]; +}; + +/** + * Feature list object + * + * Feature list is required to decouple fast path APIs with control path APIs. + * + * There are two feature lists: active, passive + * Passive list is duplicate of active list in terms of memory. + * + * While fast path APIs always work on active list but control plane work on + * passive list. When control plane needs to enable/disable any feature, it + * populates passive list afresh and atomically switch passive list to active + * list to make it available for fast path APIs + * + * Each feature node in start of its fast path function, must grab active list from + * arc via + * - rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * + * Retrieved list must be provided to other feature arc fast path APIs so that + * any control plane changes of active list should not impact current node + * execution iteration. Active list change would be reflected to current node + * in next iteration + * + * With active/passive lists and RCU mechanism in graph worker + * loop, application can update features at runtime without stopping fast path + * cores. A RCU synchronization is required when a feature needs to be + * disabled via rte_graph_feature_disable(). On enabling a feature, RCU + * synchronization may not be required + * + */ +typedef struct __rte_cache_aligned rte_graph_feature_list { + /** + * fast path array holding per_feature data. + * Duplicate entry as feature-arc also hold this pointer + * arc->features[] + * + *<-------------feature-0 ---------><---------feature-1 -------------->... + *[index-0][index-1]...[max_index-1]<-ALIGN->[index-0][index-1] ...[max_index-1]... + */ + struct rte_graph_feature *indexed_by_features; + /* + * fast path array holding first enabled feature per index + * (Required in start_node. In non start_node, mbuf can hold next enabled + * feature) + */ + rte_graph_feature_t first_enabled_feature_by_index[]; +} rte_graph_feature_list_t; + +/** + * rte_graph Feature arc object + * + * Feature arc object holds control plane and fast path information for all + * features and all interface index information for steering packets across + * feature nodes + * + * Within a feature arc, only RTE_GRAPH_FEATURE_MAX_PER_ARC features can be + * added. If more features needs to be added, another feature arc can be + * created + * + * Application gets rte_graph_feature_arc_t object via + * - rte_graph_feature_arc_create() OR + * - rte_graph_feature_arc_lookup_by_name() + * + * In fast path, rte_graph_feature_arc_t can be translated to (struct + * rte_graph_feature_arc *) via rte_graph_feature_arc_get(). Later is needed to + * add as an input argument to all fast path feature arc APIs + */ +struct __rte_cache_aligned rte_graph_feature_arc { + /* First 64B is fast path variables */ + RTE_MARKER fast_path_variables; + + /** runtime active feature list */ + RTE_ATOMIC(rte_graph_feature_rt_list_t) active_feature_list; + + /** Actual Size of feature_list object */ + uint16_t feature_list_size; + + /** + * Size each feature in fastpath. + * Required to navigate from feature to another feature in fast path + */ + uint16_t feature_size; + + /** + * Size of all feature data for an index + * Required to navigate through various feature data within a feature + * in fast path + */ + uint16_t feature_data_size; + + /** + * Quick fast path bitmask indicating if any feature enabled or not on + * any of the indexes. Helps in optimally process packets for the case + * when features are added but not enabled + * + * Separate for active and passive list + */ + RTE_ATOMIC(uint64_t) feature_enable_bitmask[2]; + + /** + * Pointer to both active and passive feature list object + */ + rte_graph_feature_list_t *feature_list[2]; + + /** + * Feature objects for each list + */ + struct rte_graph_feature *features[2]; + + /** index in feature_arc_main */ + uint16_t feature_arc_index; + + uint16_t reserved[3]; + + /** Slow path variables follows*/ + RTE_MARKER slow_path_variables; + + /** feature arc name */ + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** All feature lists */ + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; + + /** control plane counter to track enabled features */ + uint32_t runtime_enabled_features; + + /** Back pointer to feature_arc_main */ + void *feature_arc_main; + + /** Arc's start/base node */ + struct rte_node_register *start_node; + + /** maximum number of features supported by this arc */ + uint32_t max_features; + + /** maximum number of index supported by this arc */ + uint32_t max_indexes; + + /** Slow path bit mask per feature per index */ + uint64_t feature_bit_mask_by_index[]; +}; + +/** + * Feature arc main object + * + * Holds all feature arcs created by application + * + * RTE_GRAPH_FEATURE_ARC_MAX number of feature arcs can be created by + * application via rte_graph_feature_arc_create() + */ +typedef struct feature_arc_main { + /** number of feature arcs created by application */ + uint32_t num_feature_arcs; + + /** max features arcs allowed */ + uint32_t max_feature_arcs; + + /** feature arcs */ + rte_graph_feature_arc_t feature_arcs[]; +} rte_graph_feature_arc_main_t; + +/** @internal Get feature arc pointer from object */ +#define rte_graph_feature_arc_get(arc) ((struct rte_graph_feature_arc *)arc) + +extern rte_graph_feature_arc_main_t *__feature_arc_main; + +/** + * API to know if feature is valid or not + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_is_valid(rte_graph_feature_t feature) +{ + return (feature != RTE_GRAPH_FEATURE_INVALID); +} + +/** + * Get rte_graph_feature object with no checks + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * @param feature_list + * active feature list retrieved from rte_graph_feature_arc_has_any_feature() + * or rte_graph_feature_arc_has_feature() + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +__rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature, + const rte_graph_feature_rt_list_t feature_list) +{ + return ((struct rte_graph_feature *)(((uint8_t *)arc->features[feature_list]) + + (feature * arc->feature_size))); +} + +/** + * Get rte_graph_feature object for a given interface/index from feature arc + * + * @param arc + * Feature arc pointer + * @param feature + * Feature index + * + * @return + * Internal feature object. + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature * +rte_graph_feature_get(struct rte_graph_feature_arc *arc, rte_graph_feature_t feature) +{ + rte_graph_feature_rt_list_t list; + + if (unlikely(feature >= arc->max_features)) + RTE_VERIFY(0); + + if (likely(rte_graph_feature_is_valid(feature))) { + list = rte_atomic_load_explicit(&arc->active_feature_list, + rte_memory_order_relaxed); + return __rte_graph_feature_get(arc, feature, list); + } + + return NULL; +} + +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + RTE_SET_USED(arc); + return ((rte_graph_feature_data_t *)(((uint8_t *)feature->feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Get rte_graph feature data object for a index in feature + * + * @param arc + * feature arc + * @param feature + * Pointer to feature object + * @param index + * Index of feature maintained in slow path linked list + * + * @return + * Valid feature data + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t * +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, struct rte_graph_feature *feature, + uint8_t index) +{ + if (likely(index < arc->max_indexes)) + return __rte_graph_feature_data_get(arc, feature, index); + + RTE_VERIFY(0); +} + +/** + * Fast path API to check if any feature enabled on a feature arc + * Typically from arc->start_node process function + * + * @param arc + * Feature arc object + * @param[out] plist + * Pointer to runtime active feature list which needs to be provided to other + * fast path APIs + * + * @return + * 0: If no feature enabled + * Non-Zero: Bitmask of features enabled. plist is valid + * + */ +__rte_experimental +static __rte_always_inline uint64_t +rte_graph_feature_arc_has_any_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_rt_list_t *plist) +{ + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Fast path API to check if provided feature is enabled on any interface/index + * or not + * + * @param arc + * Feature arc object + * @param feature + * Input rte_graph_feature_t that needs to be checked + * @param[out] plist + * Returns active list to caller which needs to be provided to other fast path + * APIs + * + * @return + * 1: If input [feature] is enabled in arc + * 0: If input [feature] is not enabled in arc + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_has_feature(struct rte_graph_feature_arc *arc, + rte_graph_feature_t feature, + rte_graph_feature_rt_list_t *plist) +{ + uint64_t bitmask = RTE_BIT64(feature); + + *plist = rte_atomic_load_explicit(&arc->active_feature_list, rte_memory_order_relaxed); + + return (bitmask & rte_atomic_load_explicit(arc->feature_enable_bitmask + (uint8_t)*plist, + rte_memory_order_relaxed)); +} + +/** + * Prefetch feature arc fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) +{ + rte_prefetch0((void *)&arc->fast_path_variables); +} + +/** + * Prefetch feature related fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_feature_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature) +{ + /* feature cache line */ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)__rte_graph_feature_get(arc, feature, list)); +} + +/** + * Prefetch feature data upfront. Perform sanity + * + * @param arc + * RTE_GRAPH feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Pointer to feature object returned from @ref + * rte_graph_feature_arc_first_feature_get() + * @param index + * Interface/index + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_data_prefetch(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, uint32_t index) +{ + if (likely(rte_graph_feature_is_valid(feature))) + rte_prefetch0((void *)((uint8_t *)arc->features[list] + + offsetof(struct rte_graph_feature, feature_data_by_index) + + (index * sizeof(rte_graph_feature_data_t)))); +} + +/** + * Fast path API to get first enabled feature on interface index + * Typically required in arc->start_node so that from returned feature, + * feature-data can be retrieved to steer packets + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * rte_graph_feature_arc_has_feature() + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. + * + * @return + * 1. Success. If first feature field is enabled and returned [feature] is valid + * 0. Failure. If first feature field is disabled in arc + * + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_first_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + + *feature = feature_list->first_enabled_feature_by_index[index]; + + return rte_graph_feature_is_valid(*feature); +} + +/** + * Fast path API to get next enabled feature on interface index with provided + * input feature + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from + * rte_graph_feature_arc_has_any_feature() or + * @param index + * Interface Index + * @param[out] feature + * Pointer to rte_graph_feature_t. API sets next enabled feature on [index] + * from provided input feature. Valid only if API returns Success + * @param[out] next_edge + * Edge from current feature to next feature. Valid only if next feature is valid + * + * @return + * 1. Success. first feature field is enabled/valid + * 0. Failure. first feature field is disabled/invalid + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_next_feature_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *feature, + rte_edge_t *next_edge) +{ + rte_graph_feature_data_t *feature_data = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(*feature))) { + f = __rte_graph_feature_get(arc, *feature, list); + feature_data = rte_graph_feature_data_get(arc, f, index); + *feature = feature_data->next_enabled_feature; + *next_edge = feature_data->next_edge; + return rte_graph_feature_is_valid(*feature); + } + + return 0; +} + +/** + * Set fields with respect to first enabled feature in an arc and return edge + * Typically returned feature and interface index must be saved in rte_mbuf + * structure to pass this information to next feature node + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param index + * Index (of interface) + * @param[out] gf + * Pointer to rte_graph_feature_t. Valid if API returns Success + * @param[out] edge + * Edge to steer packet from arc->start_node to first enabled feature. Valid + * only if API returns Success + * + * @return + * 0: If valid feature is enabled and set by API in *gf + * 1: If valid feature is NOT enabled + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_t +rte_graph_feature_arc_feature_set(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + uint32_t index, + rte_graph_feature_t *gf, + rte_edge_t *edge) +{ + struct rte_graph_feature_list *feature_list = arc->feature_list[list]; + struct rte_graph_feature_data *feature_data = NULL; + struct rte_graph_feature *feature = NULL; + rte_graph_feature_t f; + + f = feature_list->first_enabled_feature_by_index[index]; + + if (unlikely(rte_graph_feature_is_valid(f))) { + feature = __rte_graph_feature_get(arc, f, list); + feature_data = rte_graph_feature_data_get(arc, feature, index); + *gf = f; + *edge = feature_data->next_edge; + return 0; + } + + return 1; +} + +__rte_experimental +static __rte_always_inline int32_t +__rte_graph_feature_user_data_get(rte_graph_feature_data_t *fdata) +{ + return fdata->user_data; +} + +/** + * Get user data corresponding to current feature set by application in + * rte_graph_feature_enable() + * + * @param arc + * Feature arc object + * @param list + * Pointer to runtime active feature list from rte_graph_feature_arc_has_any_feature(); + * @param feature + * Feature index + * @param index + * Interface index + * + * @return + * -1: Failure + * Valid user data: Success + */ +__rte_experimental +static __rte_always_inline int32_t +rte_graph_feature_user_data_get(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t list, + rte_graph_feature_t feature, + uint32_t index) +{ + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature *f = NULL; + + if (likely(rte_graph_feature_is_valid(feature))) { + f = __rte_graph_feature_get(arc, feature, list); + fdata = rte_graph_feature_data_get(arc, f, index); + return __rte_graph_feature_user_data_get(fdata); + } + + return -1; +} +#ifdef __cplusplus +} +#endif +#endif diff --git a/lib/graph/version.map b/lib/graph/version.map index 2c83425ddc..3b7f475afd 100644 --- a/lib/graph/version.map +++ b/lib/graph/version.map @@ -52,3 +52,23 @@ DPDK_25 { local: *; }; + +EXPERIMENTAL { + global: + + # added in 24.11 + rte_graph_feature_arc_init; + rte_graph_feature_arc_create; + rte_graph_feature_arc_lookup_by_name; + rte_graph_feature_add; + rte_graph_feature_enable; + rte_graph_feature_validate; + rte_graph_feature_disable; + rte_graph_feature_lookup; + rte_graph_feature_arc_destroy; + rte_graph_feature_arc_cleanup; + rte_graph_feature_arc_num_enabled_features; + rte_graph_feature_arc_num_features; + rte_graph_feature_arc_feature_to_name; + rte_graph_feature_arc_feature_to_node; +}; -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v5 2/5] graph: add feature arc option in graph create 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 1/5] graph: add feature arc support Nitin Saxena @ 2024-10-14 14:33 ` Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 3/5] graph: add IPv4 output feature arc Nitin Saxena ` (3 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 14:33 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena, Pavan Nikhilesh Added option in graph create to call feature-specific process node functions. This removes extra overhead for checking feature arc status in nodes where application is not using feature arc processing Signed-off-by: Pavan Nikhilesh <pbhagavatula@marvell.com> Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/rel_notes/release_24_11.rst | 7 +++++++ lib/graph/graph.c | 1 + lib/graph/graph_populate.c | 7 ++++++- lib/graph/graph_private.h | 3 +++ lib/graph/node.c | 2 ++ lib/graph/rte_graph.h | 3 +++ 6 files changed, 22 insertions(+), 1 deletion(-) diff --git a/doc/guides/rel_notes/release_24_11.rst b/doc/guides/rel_notes/release_24_11.rst index 1299de886a..451627a331 100644 --- a/doc/guides/rel_notes/release_24_11.rst +++ b/doc/guides/rel_notes/release_24_11.rst @@ -251,6 +251,13 @@ ABI Changes * eventdev: Added ``preschedule_type`` field to ``rte_event_dev_config`` structure. +* graph: Added feature arc specific `feat_arc_proc` node callback function in + `struct rte_node_register`. If this function is not NULL and + `feature_arc_enable` is set to `true` in `struct rte_graph_param`, + rte_graph_walk() calls `feat_arc_proc` callback function instead of `process` + +* graph: Added `feature_arc_enable` parameter in `struct rte_graph_param` for + calling non-NULL `feat_arc_proc` callback function by `rte_graph_walk()` Known Issues ------------ diff --git a/lib/graph/graph.c b/lib/graph/graph.c index dff8e690a8..a764c5824e 100644 --- a/lib/graph/graph.c +++ b/lib/graph/graph.c @@ -455,6 +455,7 @@ rte_graph_create(const char *name, struct rte_graph_param *prm) graph->parent_id = RTE_GRAPH_ID_INVALID; graph->lcore_id = RTE_MAX_LCORE; graph->num_pkt_to_capture = prm->num_pkt_to_capture; + graph->feature_arc_enabled = prm->feature_arc_enable; if (prm->pcap_filename) rte_strscpy(graph->pcap_filename, prm->pcap_filename, RTE_GRAPH_PCAP_FILE_SZ); diff --git a/lib/graph/graph_populate.c b/lib/graph/graph_populate.c index ed596a7711..5d8aa7b903 100644 --- a/lib/graph/graph_populate.c +++ b/lib/graph/graph_populate.c @@ -79,8 +79,13 @@ graph_nodes_populate(struct graph *_graph) if (graph_pcap_is_enable()) { node->process = graph_pcap_dispatch; node->original_process = graph_node->node->process; - } else + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->original_process = graph_node->node->feat_arc_proc; + } else { node->process = graph_node->node->process; + if (_graph->feature_arc_enabled && graph_node->node->feat_arc_proc) + node->process = graph_node->node->feat_arc_proc; + } memcpy(node->name, graph_node->node->name, RTE_GRAPH_NAMESIZE); pid = graph_node->node->parent_id; if (pid != RTE_NODE_ID_INVALID) { /* Cloned node */ diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index d557d55f2d..58ba0abeff 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -56,6 +56,7 @@ struct node { unsigned int lcore_id; /**< Node runs on the Lcore ID used for mcore dispatch model. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arch process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Allocated identifier for the node. */ @@ -126,6 +127,8 @@ struct graph { /**< Number of packets to be captured per core. */ char pcap_filename[RTE_GRAPH_PCAP_FILE_SZ]; /**< pcap file name/path. */ + uint8_t feature_arc_enabled; + /**< Graph feature arc. */ STAILQ_HEAD(gnode_list, graph_node) node_list; /**< Nodes in a graph. */ }; diff --git a/lib/graph/node.c b/lib/graph/node.c index 99a9622779..d8fd273543 100644 --- a/lib/graph/node.c +++ b/lib/graph/node.c @@ -90,6 +90,7 @@ __rte_node_register(const struct rte_node_register *reg) goto free; node->flags = reg->flags; node->process = reg->process; + node->feat_arc_proc = reg->feat_arc_proc; node->init = reg->init; node->fini = reg->fini; node->nb_edges = reg->nb_edges; @@ -137,6 +138,7 @@ node_clone(struct node *node, const char *name) /* Clone the source node */ reg->flags = node->flags; reg->process = node->process; + reg->feat_arc_proc = node->feat_arc_proc; reg->init = node->init; reg->fini = node->fini; reg->nb_edges = node->nb_edges; diff --git a/lib/graph/rte_graph.h b/lib/graph/rte_graph.h index ecfec2068a..f07272b308 100644 --- a/lib/graph/rte_graph.h +++ b/lib/graph/rte_graph.h @@ -172,6 +172,8 @@ struct rte_graph_param { uint32_t mp_capacity; /**< Capacity of memory pool for dispatch model. */ } dispatch; }; + + bool feature_arc_enable; /**< Enable Graph feature arc. */ }; /** @@ -470,6 +472,7 @@ struct rte_node_register { uint64_t flags; /**< Node configuration flag. */ #define RTE_NODE_SOURCE_F (1ULL << 0) /**< Node type is source. */ rte_node_process_t process; /**< Node process function. */ + rte_node_process_t feat_arc_proc; /**< Node feature-arc specific process function. */ rte_node_init_t init; /**< Node init function. */ rte_node_fini_t fini; /**< Node fini function. */ rte_node_t id; /**< Node Identifier. */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v5 3/5] graph: add IPv4 output feature arc 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 1/5] graph: add feature arc support Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 2/5] graph: add feature arc option in graph create Nitin Saxena @ 2024-10-14 14:33 ` Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 4/5] test/graph_feature_arc: add functional tests Nitin Saxena ` (2 subsequent siblings) 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 14:33 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena add ipv4-output feature arc in ipv4-rewrite node to allow custom/standard nodes(like outbound IPsec policy node) in outgoing forwarding path Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/node/ip4_rewrite.c | 476 +++++++++++++++++++++++++++++------- lib/node/ip4_rewrite_priv.h | 15 +- lib/node/node_private.h | 20 +- lib/node/rte_node_ip4_api.h | 3 + 4 files changed, 417 insertions(+), 97 deletions(-) diff --git a/lib/node/ip4_rewrite.c b/lib/node/ip4_rewrite.c index 34a920df5e..824ef9a4cd 100644 --- a/lib/node/ip4_rewrite.c +++ b/lib/node/ip4_rewrite.c @@ -15,39 +15,156 @@ #include "ip4_rewrite_priv.h" #include "node_private.h" +#define ALL_PKT_MASK 0xf + struct ip4_rewrite_node_ctx { + rte_graph_feature_arc_t output_feature_arc; /* Dynamic offset to mbuf priv1 */ int mbuf_priv1_off; /* Cached next index */ uint16_t next_index; + uint16_t last_tx; }; +typedef struct rewrite_priv_vars { + union { + struct { + rte_xmm_t xmm1; + }; + struct __rte_packed { + uint16_t next0; + uint16_t next1; + uint16_t next2; + uint16_t next3; + uint16_t last_tx_interface; + uint16_t last_if_feature; + uint16_t actual_feat_mask; + uint16_t speculative_feat_mask; + }; + }; +} rewrite_priv_vars_t; + static struct ip4_rewrite_node_main *ip4_rewrite_nm; #define IP4_REWRITE_NODE_LAST_NEXT(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->next_index) +#define IP4_REWRITE_NODE_LAST_TX(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->last_tx) + #define IP4_REWRITE_NODE_PRIV1_OFF(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->mbuf_priv1_off) -static uint16_t -ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, - void **objs, uint16_t nb_objs) +#define IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->output_feature_arc) + +static __rte_always_inline void +prefetch_mbuf_and_dynfield(struct rte_mbuf *mbuf) { + /* prefetch first cache line required for accessing buf_addr */ + rte_prefetch0((void *)mbuf); +} + +static __rte_always_inline void +check_output_feature_x4(struct rte_graph_feature_arc *arc, + const rte_graph_feature_rt_list_t flist, + rewrite_priv_vars_t *pvar, struct node_mbuf_priv1 *priv0, + struct node_mbuf_priv1 *priv1, struct node_mbuf_priv1 *priv2, + struct node_mbuf_priv1 *priv3) +{ + uint32_t mask = 0; + uint16_t xor = 0; + + /* + * interface edge's start from 1 and not from 0 as "pkt_drop" + * is next node at 0th index + */ + priv0->if_index = pvar->next0 - 1; + priv1->if_index = pvar->next1 - 1; + priv2->if_index = pvar->next2 - 1; + priv3->if_index = pvar->next3 - 1; + + /* Find out if all packets are sent to last_tx_interface */ + xor = pvar->last_tx_interface ^ priv0->if_index; + xor += priv0->if_index ^ priv1->if_index; + xor += priv1->if_index ^ priv2->if_index; + xor += priv2->if_index ^ priv3->if_index; + + if (likely(!xor)) { + /* copy last interface feature and feature mask */ + priv0->current_feature = priv1->current_feature = + priv2->current_feature = priv3->current_feature = + pvar->last_if_feature; + pvar->actual_feat_mask = pvar->speculative_feat_mask; + } else { + /* create a mask for index which does not have feature + * Also override next edge and if feature enabled, get feature + */ + mask = rte_graph_feature_arc_feature_set(arc, flist, priv0->if_index, + &priv0->current_feature, + &pvar->next0); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv1->if_index, + &priv1->current_feature, + &pvar->next1)) << 1); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv2->if_index, + &priv2->current_feature, + &pvar->next2)) << 2); + + mask |= ((rte_graph_feature_arc_feature_set(arc, flist, priv3->if_index, + &priv3->current_feature, + &pvar->next3)) << 3); + + /* + * add last tx and last feature regardless even if feature is + * valid or not + */ + pvar->last_tx_interface = priv3->if_index; + pvar->last_if_feature = priv3->current_feature; + /* Set 0xf if invalid feature to last packet, else 0 */ + pvar->speculative_feat_mask = (priv3->current_feature == + RTE_GRAPH_FEATURE_INVALID) ? ALL_PKT_MASK : 0x0; + pvar->actual_feat_mask = mask; + } +} + +static __rte_always_inline uint16_t +__ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs, + const int dyn, const int check_enabled_features, + struct rte_graph_feature_arc *out_feature_arc, + const rte_graph_feature_rt_list_t flist) +{ + struct node_mbuf_priv1 *priv0 = NULL, *priv1 = NULL, *priv2 = NULL, *priv3 = NULL; struct rte_mbuf *mbuf0, *mbuf1, *mbuf2, *mbuf3, **pkts; struct ip4_rewrite_nh_header *nh = ip4_rewrite_nm->nh; - const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); - uint16_t next0, next1, next2, next3, next_index; - struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; uint16_t n_left_from, held = 0, last_spec = 0; + struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; + rewrite_priv_vars_t pvar; + int64_t fd0, fd1, fd2, fd3; + rte_edge_t fix_spec = 0; void *d0, *d1, *d2, *d3; void **to_next, **from; + uint16_t next_index; rte_xmm_t priv01; rte_xmm_t priv23; int i; - /* Speculative next as last next */ + RTE_SET_USED(fd0); + RTE_SET_USED(fd1); + RTE_SET_USED(fd2); + RTE_SET_USED(fd3); + + /* Initialize speculative variables.*/ + + /* Last interface */ + pvar.last_tx_interface = IP4_REWRITE_NODE_LAST_TX(node->ctx); + /*last next from node ctx*/ next_index = IP4_REWRITE_NODE_LAST_NEXT(node->ctx); + pvar.speculative_feat_mask = ALL_PKT_MASK; + pvar.actual_feat_mask = 0; + rte_prefetch0(nh); pkts = (struct rte_mbuf **)objs; @@ -55,20 +172,47 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, n_left_from = nb_objs; for (i = 0; i < 4 && i < n_left_from; i++) - rte_prefetch0(pkts[i]); + prefetch_mbuf_and_dynfield(pkts[i]); /* Get stream for the speculated next node */ to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + + /* prefetch speculative feature and corresponding data */ + if (check_enabled_features) { + /* + * Get first feature enabled, if any, on last_tx_interface + */ + if (unlikely(rte_graph_feature_arc_first_feature_get(out_feature_arc, + flist, + pvar.last_tx_interface, + (rte_graph_feature_t *) + &pvar.last_if_feature))) { + /* prefetch feature cache line */ + rte_graph_feature_arc_feature_prefetch(out_feature_arc, flist, + pvar.last_if_feature); + + /* prefetch feature data cache line */ + rte_graph_feature_arc_data_prefetch(out_feature_arc, flist, + pvar.last_if_feature, + pvar.last_tx_interface); + /* + * Set speculativa_feat mask to indicate, all 4 packets + * going to feature path + */ + pvar.speculative_feat_mask = 0; + } + } + /* Update Ethernet header of pkts */ while (n_left_from >= 4) { if (likely(n_left_from > 7)) { /* Prefetch only next-mbuf struct and priv area. * Data need not be prefetched as we only write. */ - rte_prefetch0(pkts[4]); - rte_prefetch0(pkts[5]); - rte_prefetch0(pkts[6]); - rte_prefetch0(pkts[7]); + prefetch_mbuf_and_dynfield(pkts[4]); + prefetch_mbuf_and_dynfield(pkts[5]); + prefetch_mbuf_and_dynfield(pkts[6]); + prefetch_mbuf_and_dynfield(pkts[7]); } mbuf0 = pkts[0]; @@ -78,66 +222,138 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 4; n_left_from -= 4; + + /* Copy mbuf private data into private variables */ priv01.u64[0] = node_mbuf_priv1(mbuf0, dyn)->u; priv01.u64[1] = node_mbuf_priv1(mbuf1, dyn)->u; priv23.u64[0] = node_mbuf_priv1(mbuf2, dyn)->u; priv23.u64[1] = node_mbuf_priv1(mbuf3, dyn)->u; - /* Increment checksum by one. */ - priv01.u32[1] += rte_cpu_to_be_16(0x0100); - priv01.u32[3] += rte_cpu_to_be_16(0x0100); - priv23.u32[1] += rte_cpu_to_be_16(0x0100); - priv23.u32[3] += rte_cpu_to_be_16(0x0100); - - /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, - nh[priv01.u16[0]].rewrite_len); - - next0 = nh[priv01.u16[0]].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - ip0->time_to_live = priv01.u16[1] - 1; - ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ - d1 = rte_pktmbuf_mtod(mbuf1, void *); - rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, - nh[priv01.u16[4]].rewrite_len); - - next1 = nh[priv01.u16[4]].tx_node; - ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + - sizeof(struct rte_ether_hdr)); - ip1->time_to_live = priv01.u16[5] - 1; - ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ - d2 = rte_pktmbuf_mtod(mbuf2, void *); - rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, - nh[priv23.u16[0]].rewrite_len); - next2 = nh[priv23.u16[0]].tx_node; - ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + - sizeof(struct rte_ether_hdr)); - ip2->time_to_live = priv23.u16[1] - 1; - ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; - - /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ - d3 = rte_pktmbuf_mtod(mbuf3, void *); - rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, - nh[priv23.u16[4]].rewrite_len); - - next3 = nh[priv23.u16[4]].tx_node; - ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + - sizeof(struct rte_ether_hdr)); - ip3->time_to_live = priv23.u16[5] - 1; - ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + /* Copy next edge from next hop */ + pvar.next0 = nh[priv01.u16[0]].tx_node; + pvar.next1 = nh[priv01.u16[4]].tx_node; + pvar.next2 = nh[priv23.u16[0]].tx_node; + pvar.next3 = nh[priv23.u16[4]].tx_node; + + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + priv1 = node_mbuf_priv1(mbuf1, dyn); + priv2 = node_mbuf_priv1(mbuf2, dyn); + priv3 = node_mbuf_priv1(mbuf3, dyn); + + /* If feature is enabled, override next edge for each mbuf + * and set node_mbuf_priv data appropriately + */ + check_output_feature_x4(out_feature_arc, flist, + &pvar, priv0, priv1, priv2, priv3); + + /* check_output_feature_x4() returns bit mask which indicates + * which packet is not following feature path, hence normal processing + * has to happen on them + */ + if (unlikely(pvar.actual_feat_mask)) { + if (pvar.actual_feat_mask & 0x1) { + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + } + if (pvar.actual_feat_mask & 0x2) { + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + } + if (pvar.actual_feat_mask & 0x4) { + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + } + if (pvar.actual_feat_mask & 0x8) { + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } + } + } else { + /* Case when no feature is enabled */ + + /* Increment checksum by one. */ + priv01.u32[1] += rte_cpu_to_be_16(0x0100); + priv01.u32[3] += rte_cpu_to_be_16(0x0100); + priv23.u32[1] += rte_cpu_to_be_16(0x0100); + priv23.u32[3] += rte_cpu_to_be_16(0x0100); + + /* Update ttl,cksum rewrite ethernet hdr on mbuf0 */ + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[priv01.u16[0]].rewrite_data, + nh[priv01.u16[0]].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + ip0->time_to_live = priv01.u16[1] - 1; + ip0->hdr_checksum = priv01.u16[2] + priv01.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf1 */ + d1 = rte_pktmbuf_mtod(mbuf1, void *); + rte_memcpy(d1, nh[priv01.u16[4]].rewrite_data, + nh[priv01.u16[4]].rewrite_len); + + ip1 = (struct rte_ipv4_hdr *)((uint8_t *)d1 + + sizeof(struct rte_ether_hdr)); + ip1->time_to_live = priv01.u16[5] - 1; + ip1->hdr_checksum = priv01.u16[6] + priv01.u16[7]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf2 */ + d2 = rte_pktmbuf_mtod(mbuf2, void *); + rte_memcpy(d2, nh[priv23.u16[0]].rewrite_data, + nh[priv23.u16[0]].rewrite_len); + ip2 = (struct rte_ipv4_hdr *)((uint8_t *)d2 + + sizeof(struct rte_ether_hdr)); + ip2->time_to_live = priv23.u16[1] - 1; + ip2->hdr_checksum = priv23.u16[2] + priv23.u16[3]; + + /* Update ttl,cksum rewrite ethernet hdr on mbuf3 */ + d3 = rte_pktmbuf_mtod(mbuf3, void *); + rte_memcpy(d3, nh[priv23.u16[4]].rewrite_data, + nh[priv23.u16[4]].rewrite_len); + + ip3 = (struct rte_ipv4_hdr *)((uint8_t *)d3 + + sizeof(struct rte_ether_hdr)); + ip3->time_to_live = priv23.u16[5] - 1; + ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + } /* Enqueue four to next node */ - rte_edge_t fix_spec = - ((next_index == next0) && (next0 == next1) && - (next1 == next2) && (next2 == next3)); + fix_spec = next_index ^ pvar.next0; + fix_spec += next_index ^ pvar.next1; + fix_spec += next_index ^ pvar.next2; + fix_spec += next_index ^ pvar.next3; - if (unlikely(fix_spec == 0)) { + if (unlikely(fix_spec != 0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -146,56 +362,56 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, last_spec = 0; /* next0 */ - if (next_index == next0) { + if (next_index == pvar.next0) { to_next[0] = from[0]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next0, + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); } /* next1 */ - if (next_index == next1) { + if (next_index == pvar.next1) { to_next[0] = from[1]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next1, + rte_node_enqueue_x1(graph, node, pvar.next1, from[1]); } /* next2 */ - if (next_index == next2) { + if (next_index == pvar.next2) { to_next[0] = from[2]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next2, + rte_node_enqueue_x1(graph, node, pvar.next2, from[2]); } /* next3 */ - if (next_index == next3) { + if (next_index == pvar.next3) { to_next[0] = from[3]; to_next++; held++; } else { - rte_node_enqueue_x1(graph, node, next3, + rte_node_enqueue_x1(graph, node, pvar.next3, from[3]); } from += 4; /* Change speculation if last two are same */ - if ((next_index != next3) && (next2 == next3)) { + if ((next_index != pvar.next3) && (pvar.next2 == pvar.next3)) { /* Put the current speculated node */ rte_node_next_stream_put(graph, node, next_index, held); held = 0; /* Get next speculated stream */ - next_index = next3; + next_index = pvar.next3; to_next = rte_node_next_stream_get( graph, node, next_index, nb_objs); } @@ -212,20 +428,41 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 1; n_left_from -= 1; - d0 = rte_pktmbuf_mtod(mbuf0, void *); - rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, - nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); - - next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; - ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + - sizeof(struct rte_ether_hdr)); - chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + - rte_cpu_to_be_16(0x0100); - chksum += chksum >= 0xffff; - ip0->hdr_checksum = chksum; - ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; - - if (unlikely(next_index ^ next0)) { + pvar.next0 = nh[node_mbuf_priv1(mbuf0, dyn)->nh].tx_node; + if (check_enabled_features) { + priv0 = node_mbuf_priv1(mbuf0, dyn); + if (pvar.next0 != (pvar.last_tx_interface + 1)) { + priv0->if_index = pvar.next0 - 1; + rte_graph_feature_arc_feature_set(out_feature_arc, flist, + priv0->if_index, + &priv0->current_feature, + &pvar.next0); + pvar.last_tx_interface = priv0->if_index; + pvar.last_if_feature = priv0->current_feature; + } else { + /* current mbuf index is same as last_tx_interface */ + priv0->if_index = pvar.last_tx_interface; + priv0->current_feature = pvar.last_if_feature; + } + } + /* Do the needful if either feature arc is disabled OR + * Invalid feature is present + */ + if (!check_enabled_features || + (priv0->current_feature == RTE_GRAPH_FEATURE_INVALID)) { + d0 = rte_pktmbuf_mtod(mbuf0, void *); + rte_memcpy(d0, nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_data, + nh[node_mbuf_priv1(mbuf0, dyn)->nh].rewrite_len); + + ip0 = (struct rte_ipv4_hdr *)((uint8_t *)d0 + + sizeof(struct rte_ether_hdr)); + chksum = node_mbuf_priv1(mbuf0, dyn)->cksum + + rte_cpu_to_be_16(0x0100); + chksum += chksum >= 0xffff; + ip0->hdr_checksum = chksum; + ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; + } + if (unlikely(next_index ^ pvar.next0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); from += last_spec; @@ -233,13 +470,15 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, held += last_spec; last_spec = 0; - rte_node_enqueue_x1(graph, node, next0, from[0]); + rte_node_enqueue_x1(graph, node, pvar.next0, from[0]); from += 1; } else { last_spec += 1; } } + IP4_REWRITE_NODE_LAST_TX(node->ctx) = pvar.last_tx_interface; + /* !!! Home run !!! */ if (likely(last_spec == nb_objs)) { rte_node_next_stream_move(graph, node, next_index); @@ -255,22 +494,78 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, return nb_objs; } +static uint16_t +ip4_rewrite_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = + rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + rte_graph_feature_rt_list_t flist; + + /* If any feature is enabled on this arc */ + if (unlikely(rte_graph_feature_arc_has_any_feature(arc, &flist))) { + if (flist) + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)1); + else + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, + dyn, + 1 /* check features */, arc, + (rte_graph_feature_rt_list_t)0); + } else { + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); + } + return 0; +} + +static uint16_t +ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, NULL, + 0/* don't care */); +} + static int ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) { + rte_graph_feature_arc_t feature_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; static bool init_once; RTE_SET_USED(graph); RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_node_ctx) > RTE_NODE_CTX_SZ); + RTE_BUILD_BUG_ON(sizeof(struct ip4_rewrite_nh_header) != RTE_CACHE_LINE_SIZE); if (!init_once) { node_mbuf_priv1_dynfield_offset = rte_mbuf_dynfield_register( &node_mbuf_priv1_dynfield_desc); if (node_mbuf_priv1_dynfield_offset < 0) return -rte_errno; - init_once = true; + + /* Create ipv4-output feature arc, if not created + */ + if (rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + NULL) < 0) { + if (rte_graph_feature_arc_create(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + RTE_GRAPH_FEATURE_MAX_PER_ARC, + RTE_MAX_ETHPORTS, + ip4_rewrite_node_get(), &feature_arc)) { + return -rte_errno; + } + init_once = true; + } } IP4_REWRITE_NODE_PRIV1_OFF(node->ctx) = node_mbuf_priv1_dynfield_offset; + IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx) = feature_arc; + IP4_REWRITE_NODE_LAST_TX(node->ctx) = UINT16_MAX; node_dbg("ip4_rewrite", "Initialized ip4_rewrite node initialized"); @@ -329,6 +624,7 @@ rte_node_ip4_rewrite_add(uint16_t next_hop, uint8_t *rewrite_data, static struct rte_node_register ip4_rewrite_node = { .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, .name = "ip4_rewrite", /* Default edge i.e '0' is pkt drop */ .nb_edges = 1, diff --git a/lib/node/ip4_rewrite_priv.h b/lib/node/ip4_rewrite_priv.h index 5105ec1d29..52f39601bd 100644 --- a/lib/node/ip4_rewrite_priv.h +++ b/lib/node/ip4_rewrite_priv.h @@ -5,9 +5,11 @@ #define __INCLUDE_IP4_REWRITE_PRIV_H__ #include <rte_common.h> +#include <rte_graph_feature_arc.h> #define RTE_GRAPH_IP4_REWRITE_MAX_NH 64 -#define RTE_GRAPH_IP4_REWRITE_MAX_LEN 56 +#define RTE_GRAPH_IP4_REWRITE_MAX_LEN (sizeof(struct rte_ether_hdr) + \ + (2 * sizeof(struct rte_vlan_hdr))) /** * @internal @@ -15,11 +17,9 @@ * Ipv4 rewrite next hop header data structure. Used to store port specific * rewrite data. */ -struct ip4_rewrite_nh_header { - uint16_t rewrite_len; /**< Header rewrite length. */ +struct __rte_cache_aligned ip4_rewrite_nh_header { uint16_t tx_node; /**< Tx node next index identifier. */ - uint16_t enabled; /**< NH enable flag */ - uint16_t rsvd; + uint16_t rewrite_len; /**< Header rewrite length. */ union { struct { struct rte_ether_addr dst; @@ -30,8 +30,13 @@ struct ip4_rewrite_nh_header { uint8_t rewrite_data[RTE_GRAPH_IP4_REWRITE_MAX_LEN]; /**< Generic rewrite data */ }; + /* used in control path */ + uint8_t enabled; /**< NH enable flag */ }; +_Static_assert(sizeof(struct ip4_rewrite_nh_header) <= (size_t)RTE_CACHE_LINE_SIZE, + "ip4_rewrite_nh_header size must be less or equal to cache line"); + /** * @internal * diff --git a/lib/node/node_private.h b/lib/node/node_private.h index 1de7306792..25db04a9a6 100644 --- a/lib/node/node_private.h +++ b/lib/node/node_private.h @@ -12,6 +12,9 @@ #include <rte_mbuf.h> #include <rte_mbuf_dyn.h> +#include <rte_graph_worker_common.h> +#include <rte_graph_feature_arc_worker.h> + extern int rte_node_logtype; #define RTE_LOGTYPE_NODE rte_node_logtype @@ -29,15 +32,28 @@ extern int rte_node_logtype; */ struct node_mbuf_priv1 { union { - /* IP4/IP6 rewrite */ + /** + * IP4/IP6 rewrite + * only used to pass lookup data from + * ip4-lookup to ip4-rewrite + */ struct { uint16_t nh; uint16_t ttl; uint32_t cksum; }; - uint64_t u; }; + /** + * Feature arc data + */ + struct { + /** interface index */ + uint16_t if_index; + /** feature that current mbuf holds */ + rte_graph_feature_t current_feature; + uint8_t rsvd; + }; }; static const struct rte_mbuf_dynfield node_mbuf_priv1_dynfield_desc = { diff --git a/lib/node/rte_node_ip4_api.h b/lib/node/rte_node_ip4_api.h index 950751a525..da4995662a 100644 --- a/lib/node/rte_node_ip4_api.h +++ b/lib/node/rte_node_ip4_api.h @@ -19,6 +19,7 @@ #include <rte_compat.h> #include <rte_graph.h> +#include <rte_graph_feature_arc_worker.h> #ifdef __cplusplus extern "C" { @@ -67,6 +68,8 @@ struct rte_node_ip4_reassembly_cfg { /**< Node identifier to configure. */ }; +#define RTE_IP4_OUTPUT_FEATURE_ARC_NAME "ipv4-output" + /** * Add ipv4 route to lookup table. * -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v5 4/5] test/graph_feature_arc: add functional tests 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena ` (2 preceding siblings ...) 2024-10-14 14:33 ` [PATCH v5 3/5] graph: add IPv4 output feature arc Nitin Saxena @ 2024-10-14 14:33 ` Nitin Saxena 2024-10-14 19:54 ` Stephen Hemminger 2024-10-14 14:33 ` [PATCH v5 5/5] docs: add programming guide for feature arc Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena 5 siblings, 1 reply; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 14:33 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Added functional unit test case for verifying feature arc control plane and fast path APIs How to run: $ echo "graph_feature_arc_autotest" | ./bin/dpdk-test Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- app/test/meson.build | 1 + app/test/test_graph_feature_arc.c | 1410 +++++++++++++++++++++++++++++ 2 files changed, 1411 insertions(+) create mode 100644 app/test/test_graph_feature_arc.c diff --git a/app/test/meson.build b/app/test/meson.build index e29258e6ec..740fa1bfb4 100644 --- a/app/test/meson.build +++ b/app/test/meson.build @@ -90,6 +90,7 @@ source_file_deps = { 'test_func_reentrancy.c': ['hash', 'lpm'], 'test_graph.c': ['graph'], 'test_graph_perf.c': ['graph'], + 'test_graph_feature_arc.c': ['graph'], 'test_hash.c': ['net', 'hash'], 'test_hash_functions.c': ['hash'], 'test_hash_multiwriter.c': ['hash'], diff --git a/app/test/test_graph_feature_arc.c b/app/test/test_graph_feature_arc.c new file mode 100644 index 0000000000..58b676e215 --- /dev/null +++ b/app/test/test_graph_feature_arc.c @@ -0,0 +1,1410 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2024 Marvell International Ltd. + */ + +#include "test.h" + +#include <assert.h> +#include <inttypes.h> +#include <signal.h> +#include <stdalign.h> +#include <stdio.h> +#include <string.h> +#include <unistd.h> + +#include <rte_errno.h> + +#ifndef RTE_EXEC_ENV_WINDOWS +#include <rte_graph.h> +#include <rte_graph_worker.h> +#include <rte_mbuf.h> +#include <rte_mbuf_dyn.h> +#include <rte_random.h> +#include <rte_graph_feature_arc.h> +#include <rte_graph_feature_arc_worker.h> + +#define MBUFF_SIZE 512 +#define TEST_ARC1_NAME "arc1" +#define TEST_ARC2_NAME "arc2" +#define MAX_INDEXES 10 +#define MAX_FEATURES 5 + +#define SOURCE1 "test_node_arc_source1" +#define INPUT_STATIC "test_node_arc_input_static" +#define OUTPUT_STATIC "test_node_arc_output_static" +#define PKT_FREE_STATIC "test_node_arc_pkt_free_static" +#define ARC1_FEATURE1 "test_node_arc1_feature1" +#define ARC1_FEATURE2 "test_node_arc1_feature2" +#define ARC2_FEATURE1 "test_node_arc2_feature1" +#define ARC2_FEATURE2 "test_node_arc2_feature2" +#define ARC2_FEATURE3 "test_node_arc2_feature3" +#define DUMMY1_STATIC "test_node_arc_dummy1_static" +#define DUMMY2_STATIC "test_node_arc_dummy2_static" + +/* (Node index, Node Name, feature user data base */ +#define FOREACH_TEST_NODE_ARC { \ + R(0, SOURCE1, 64) \ + R(1, INPUT_STATIC, 128) \ + R(2, OUTPUT_STATIC, 256) \ + R(3, PKT_FREE_STATIC, 512) \ + R(4, ARC1_FEATURE1, 1024) \ + R(5, ARC1_FEATURE2, 2048) \ + R(6, ARC2_FEATURE1, 4096) \ + R(7, ARC2_FEATURE2, 8192) \ + R(8, ARC2_FEATURE3, 16384) \ + R(9, DUMMY1_STATIC, 32768) \ + R(10, DUMMY2_STATIC, 65536) \ + } + +/** + * ARC1: Feature arc on ingress interface + * ARC2: Feature arc on egress interface + * XX_static: Static nodes + * XX_featureX: Feature X on arc + * + * -----> ARC1_FEATURE1 + * | | | + * | | v + * | | ARC1_FEATURE2 + * | | | + * | v v + * SOURCE1 ->-----> INPUT_STATIC --> OUTPUT_STATIC -----> PKT_FREE_STATIC + * | | | ^ ^ ^ + * | | | | | | + * | | --> ARC2_FEATURE1 | | + * | | ^ ^ | | + * | | | | | | + * | ----------c-> ARC2_FEATURE2 | + * | | ^ | + * | | | | + * ----------> ARC2_FEATURE3 ------- + */ +const char *node_names_feature_arc[] = { + SOURCE1, INPUT_STATIC, OUTPUT_STATIC, PKT_FREE_STATIC, + ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE2, ARC2_FEATURE3, + DUMMY1_STATIC, DUMMY2_STATIC +}; + +#define MAX_NODES RTE_DIM(node_names_feature_arc) + +/* Function declarations */ +static uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs); +static int +common_node_init(const struct rte_graph *graph, struct rte_node *node); + +typedef struct test_node_priv { + /* index from 0 - MAX_NODES -1 */ + uint8_t node_index; + + /* feature */ + rte_graph_feature_t feature; + + /* rte_graph node id */ + uint32_t node_id; + + rte_graph_feature_arc_t arc; +} test_node_priv_t; + +typedef struct { + rte_graph_feature_t feature; + uint16_t egress_interface; + uint16_t ingress_interface; +} graph_dynfield_t; + +static int graph_dynfield_offset = -1; +static rte_graph_feature_arc_t arcs[RTE_GRAPH_FEATURE_ARC_MAX + 128]; +static struct rte_mbuf mbuf[MAX_NODES + 1][MBUFF_SIZE]; +static void *mbuf_p[MAX_NODES + 1][MBUFF_SIZE]; +static rte_graph_t graph_id = RTE_GRAPH_ID_INVALID; + +const char *node_patterns_feature_arc[] = { + "test_node_arc*" +}; + +static int32_t +compute_unique_user_data(const char *parent, const char *child, uint32_t interface_index) +{ + uint32_t user_data = interface_index; + + RTE_SET_USED(parent); +#define R(idx, node, node_cookie) { \ + if (!strcmp(child, node)) { \ + user_data += node_cookie; \ + } \ + } + + FOREACH_TEST_NODE_ARC +#undef R + + return user_data; +} + +static int +get_edge(struct rte_node_register *parent_node, + struct rte_node_register *child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t count, i; + + count = rte_node_edge_get(parent_node->id, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node->id, next_edges); + for (i = 0; i < count; i++) { + if (strstr(child_node->name, next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + +int +common_node_init(const struct rte_graph *graph, struct rte_node *node) +{ + test_node_priv_t *priv = (test_node_priv_t *)node->ctx; + + RTE_SET_USED(graph); + + priv->node_id = node->id; + priv->feature = RTE_GRAPH_FEATURE_INVALID; + priv->arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + +#define R(idx, _name, user_data) { \ + if (!strcmp(node->name, _name)) { \ + priv->node_index = idx; \ + } \ + } + FOREACH_TEST_NODE_ARC +#undef R + + return 0; +} + +uint16_t +source1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register source1 = { + .name = SOURCE1, + .process = source1_fn, + .flags = RTE_NODE_SOURCE_F, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {INPUT_STATIC, DUMMY1_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(source1); + +uint16_t +input_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +input_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register input = { + .name = INPUT_STATIC, + .process = input_fn, + .feat_arc_proc = input_fa_fn, + .nb_edges = 2, + .init = common_node_init, + .next_nodes = {OUTPUT_STATIC, DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(input); + +uint16_t +output_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +output_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register output = { + .name = OUTPUT_STATIC, + .process = output_fn, + .feat_arc_proc = output_fa_fn, + .nb_edges = 3, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC, PKT_FREE_STATIC, DUMMY2_STATIC}, +}; +RTE_NODE_REGISTER(output); + +uint16_t +pkt_free_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +uint16_t +pkt_free_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register pkt_free = { + .name = PKT_FREE_STATIC, + .process = pkt_free_fn, + .feat_arc_proc = pkt_free_fa_fn, + .nb_edges = 1, + .init = common_node_init, + .next_nodes = {DUMMY1_STATIC}, +}; +RTE_NODE_REGISTER(pkt_free); + +uint16_t +dummy1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy1 = { + .name = DUMMY1_STATIC, + .process = dummy1_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(dummy1); + +uint16_t +dummy2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + return 0; +} + +static struct rte_node_register dummy2 = { + .name = DUMMY2_STATIC, + .process = dummy2_fn, + .nb_edges = 5, + .init = common_node_init, + .next_nodes = { ARC1_FEATURE1, ARC1_FEATURE2, ARC2_FEATURE1, + ARC2_FEATURE2, ARC2_FEATURE3}, +}; +RTE_NODE_REGISTER(dummy2); + +uint16_t +arc1_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature1 = { + .name = ARC1_FEATURE1, + .process = arc1_feature1_fn, + .feat_arc_proc = arc1_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature1); + +uint16_t +arc1_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc1_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc1_feature2 = { + .name = ARC1_FEATURE2, + .process = arc1_feature2_fn, + .feat_arc_proc = arc1_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc1_feature2); + +uint16_t +arc2_feature1_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature1_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature1 = { + .name = ARC2_FEATURE1, + .process = arc2_feature1_fn, + .feat_arc_proc = arc2_feature1_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature1); + +uint16_t +arc2_feature2_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature2_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature2 = { + .name = ARC2_FEATURE2, + .process = arc2_feature2_fn, + .feat_arc_proc = arc2_feature2_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature2); + +uint16_t +arc2_feature3_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +uint16_t +arc2_feature3_fa_fn(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + RTE_SET_USED(graph); + RTE_SET_USED(node); + RTE_SET_USED(objs); + RTE_SET_USED(nb_objs); + + return 0; +} + +static struct rte_node_register arc2_feature3 = { + .name = ARC2_FEATURE3, + .process = arc2_feature3_fn, + .feat_arc_proc = arc2_feature3_fa_fn, + .nb_edges = 0, + .init = common_node_init, +}; +RTE_NODE_REGISTER(arc2_feature3); + +static int +create_graph(void) +{ + struct rte_graph_param gconf = { + .socket_id = SOCKET_ID_ANY, + .nb_node_patterns = 1, + .node_patterns = node_patterns_feature_arc, + }; + + graph_id = rte_graph_create("worker0", &gconf); + if (graph_id == RTE_GRAPH_ID_INVALID) { + printf("Graph creation failed with error = %d\n", rte_errno); + return TEST_FAILED; + } + + return TEST_SUCCESS; +} + +static int +__test_create_feature_arc(rte_graph_feature_arc_t *arcs, int max_arcs) +{ + rte_graph_feature_arc_t arc; + const char *sample_arc_name = "sample_arc"; + char arc_name[256]; + int n_arcs; + + /* Create max number of feature arcs first */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + if (rte_graph_feature_arc_create(arc_name, MAX_FEATURES, + MAX_INDEXES, &dummy1, &arcs[n_arcs])) { + printf("Feature arc creation failed for %u\n", n_arcs); + return TEST_FAILED; + } + } + /* Verify feature arc created more than max_arcs must fail */ + if (!rte_graph_feature_arc_create("negative_test_create_arc", MAX_FEATURES, + MAX_INDEXES, &dummy2, &arc)) { + printf("Feature arc creation success for more than max configured: %u\n", n_arcs); + return TEST_FAILED; + } + /* Make sure lookup passes for all feature arcs */ + for (n_arcs = 0; n_arcs < max_arcs; n_arcs++) { + snprintf(arc_name, sizeof(arc_name), "%s-%u", sample_arc_name, n_arcs); + arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + if (!rte_graph_feature_arc_lookup_by_name(arc_name, &arc)) { + if (arc != arcs[n_arcs]) { + printf("%s: Feature arc lookup mismatch for arc [%p, exp: %p]\n", + arc_name, (void *)arc, (void *)arcs[n_arcs]); + return TEST_FAILED; + } + } else { + printf("Feature arc lookup %s failed after creation\n", arc_name); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_create(void) +{ + int ret = 0, i; + + /* Create arcs with RTE_GRAPH_FEATURE_ARC_MAX */ + ret = __test_create_feature_arc(arcs, RTE_GRAPH_FEATURE_ARC_MAX); + if (ret) { + printf("Feature arc creation test failed for RTE_GRAPH_FEATURE_ARC_MAX arcs\n"); + return TEST_FAILED; + } + /* destroy all arcs via cleanup API*/ + ret = rte_graph_feature_arc_cleanup(); + if (ret) { + printf("Feature arc cleanup failed\n"); + return TEST_FAILED; + } + +#define NUM_FEAT_ARCS 128 + /* create 128 dummy feature arcs */ + ret = rte_graph_feature_arc_init(NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc init failed for NUM_FEAT_ARCS"); + return TEST_FAILED; + } + ret = __test_create_feature_arc(arcs, NUM_FEAT_ARCS); + if (ret) { + printf("Feature arc creation test failed for NUM_FEAT_ARCS\n"); + return TEST_FAILED; + } + /* destroy all of them*/ + for (i = 0; i < NUM_FEAT_ARCS; i++) { + if (rte_graph_feature_arc_destroy(arcs[i])) { + printf("Feature arc destroy failed for %u\n", i); + return TEST_FAILED; + } + } + rte_graph_feature_arc_cleanup(); + + /* Create two arcs as per test plan */ + /* First arc start/source node is node: SOURCE1 */ + if (rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[0])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + /* Duplicate name should fail */ + if (!rte_graph_feature_arc_create(TEST_ARC1_NAME, MAX_FEATURES, + MAX_INDEXES, &source1, &arcs[1])) { + printf("Duplicate feature arc %s creation is not caught\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Second arc start/source node is node: OUTPUT_STATIC */ + if (rte_graph_feature_arc_create(TEST_ARC2_NAME, MAX_FEATURES, + MAX_INDEXES, &output, &arcs[1])) { + printf("Feature arc creation failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_features_add(void) +{ + rte_graph_feature_t temp; + + /* First feature to SOURCE1 start node -> ARC1_FEATURE1 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature1, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1); + return TEST_FAILED; + } + /* Second feature to SOURCE1 -> ARC1_FEATURE2 */ + if (rte_graph_feature_add(arcs[0], &arc1_feature2, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + /* adding statically connected INPUT_STATIC as a last feature */ + if (rte_graph_feature_add(arcs[0], &input, ARC1_FEATURE2, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC1_NAME, INPUT_STATIC, ARC1_FEATURE2); + return TEST_FAILED; + } + /* First feature to OUTPUT_STATIC start node -> ARC2_FEATURE3 */ + if (rte_graph_feature_add(arcs[1], &arc2_feature3, NULL, NULL)) { + printf("%s: Feature add failed for adding feature %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Second feature to OUTPUT_STATIC -> ARC2_FEATURE1 and before feature to + * ARC2_FEATURE3 + */ + if (rte_graph_feature_add(arcs[1], &arc2_feature1, NULL, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Add PKT_FREE node as last feature, next to arc2_feature3 */ + if (rte_graph_feature_add(arcs[1], &pkt_free, ARC2_FEATURE3, NULL)) { + printf("%s: Feature add failed for adding feature %s after %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Adding feature ARC2_FEATURE2 between ARC2_FEATURE1 and ARC2_FEATURE3. */ + if (rte_graph_feature_add(arcs[1], &arc2_feature2, ARC2_FEATURE1, ARC2_FEATURE3)) { + printf("%s: Feature add failed for adding feature %s between [%s - %s]\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE1, ARC2_FEATURE3); + return TEST_FAILED; + } + /* Now check feature sequencing is correct for both ARCS */ + + /* arc1_feature1 must be first feature to arcs[0] */ + if (!strstr(ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc1_feature2 must be second feature to arcs[0] */ + if (!strstr(ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC1_NAME, ARC1_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[0], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* Make sure INPUT_STATIC is the last feature in arcs[0] */ + temp = rte_graph_feature_arc_num_features(arcs[0]); + if (!strstr(INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC1_NAME, INPUT_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[0], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_feature1 must be first feature to arcs[1] */ + if (!strstr(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(0)))) { + printf("%s: %s is not the first feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(0))); + return TEST_FAILED; + } + + /* arc2_feature2 must be second feature to arcs[1] */ + if (!strstr(ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(1)))) { + printf("%s: %s is not the second feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE2, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + /* arc2_feature3 must be third feature to arcs[1] */ + if (!strstr(ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], + rte_graph_feature_cast(2)))) { + printf("%s: %s is not the third feature instead %s\n", + TEST_ARC2_NAME, ARC2_FEATURE3, + rte_graph_feature_arc_feature_to_name(arcs[1], rte_graph_feature_cast(2))); + return TEST_FAILED; + } + + /* Make sure PKT_FREE is the last feature in arcs[1] */ + temp = rte_graph_feature_arc_num_features(arcs[1]); + if (!strstr(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1)))) { + printf("%s: %s is not the last feature instead %s\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], + temp - rte_graph_feature_cast(1))); + return TEST_FAILED; + } + + if (get_edge(&arc2_feature1, &pkt_free, NULL)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, ARC2_FEATURE1, PKT_FREE_STATIC); + return TEST_FAILED; + } + + return create_graph(); +} + +static int +test_graph_feature_arc_first_feature_enable(void) +{ + uint32_t n_indexes, n_features, count = 0; + rte_graph_feature_rt_list_t feature_list, temp = 0; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + char *feature_name = NULL; + int32_t user_data; + rte_edge_t edge = ~0; + + arc = rte_graph_feature_arc_get(arcs[0]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 0, + * On interface 1, enable feature 1 and so on so forth + * + * later verify first feature on every interface index is unique + * and check [rte_edge, user_data] retrieved via fast path APIs + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = n_indexes % 3 /* 3 features added to arc1 */; + feature_name = rte_graph_feature_arc_feature_to_name(arcs[0], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_validate(arcs[0], n_indexes, feature_name, 1, true)) { + printf("%s: Feature validate failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* negative test case. enable feature on invalid index */ + if (!n_indexes && !rte_graph_feature_enable(arcs[0], MAX_INDEXES, feature_name, + (int32_t)user_data)) { + printf("%s: Feature %s should not be enabled on invalid index\n", + TEST_ARC1_NAME, feature_name); + return TEST_FAILED; + } + if (rte_graph_feature_enable(arcs[0], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC1_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + if (temp == feature_list) { + printf("%s: Activer feature list not switched from %u -> %u\n", + TEST_ARC1_NAME, temp, feature_list); + return TEST_FAILED; + } + temp = feature_list; + if ((count + 1) != rte_graph_feature_arc_num_enabled_features(arcs[0])) { + printf("%s: Number of enabled mismatches [found: %u, exp: %u]\n", + TEST_ARC1_NAME, + rte_graph_feature_arc_num_enabled_features(arcs[0]), + count + 1); + return TEST_FAILED; + } + count++; + } + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC1_NAME); + return TEST_FAILED; + } + /* Negative test case */ + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, 1); + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC2_FEATURE1, user_data)) { + printf("%s: Invalid feature %s is enabled on index 1\n", + TEST_ARC1_NAME, ARC2_FEATURE1); + return TEST_FAILED; + } + /* Duplicate enable */ + if (!rte_graph_feature_enable(arcs[0], 1 /* index */, ARC1_FEATURE2, user_data)) { + printf("%s: Duplicate feature %s shouldn't be enabled again on index 1\n", + TEST_ARC1_NAME, ARC1_FEATURE2); + return TEST_FAILED; + } + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC1_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc1_feature1; + else if (1 == (n_indexes % 3)) + child = &arc1_feature2; + else + child = &input; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC1_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC1_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +verify_feature_sequencing(struct rte_graph_feature_arc *arc) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + rte_graph_feature_t feature; + uint32_t n_indexes; + rte_edge_t edge = ~0; + int32_t user_data; + + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: feature_list can't be obtained\n", + arc->feature_arc_name); + return TEST_FAILED; + } + /* Verify next features on interface 0 and interface 1*/ + for (n_indexes = 0; n_indexes < 2; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: 0\n", + arc->feature_arc_name); + return TEST_FAILED; + } + parent = arc->start_node; + child = rte_graph_feature_arc_feature_to_node(arcs[1], feature); + /* until fast path API reaches last feature i.e pkt_free */ + while (child != &pkt_free) { + fdata = rte_graph_feature_data_get(arc, + rte_graph_feature_get(arc, feature), + n_indexes); + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + arc->feature_arc_name, parent->name, child->name); + return TEST_FAILED; + } + user_data = compute_unique_user_data(parent->name, child->name, n_indexes); + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != user_data) { + printf("%s: Udata mismatch for %s->%s on index %u [%u, exp: %u]\n", + arc->feature_arc_name, parent->name, child->name, n_indexes, + fdata->user_data, user_data); + return TEST_FAILED; + } + + feature = fdata->next_enabled_feature; + + parent = child; + child = rte_graph_feature_arc_feature_to_node(arcs[1], + fdata->next_enabled_feature); + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_enable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_node_register *parent, *child; + rte_graph_feature_data_t *fdata = NULL; + struct rte_graph_feature_arc *arc; + uint32_t n_indexes, n_features; + rte_graph_feature_t feature; + char *feature_name = NULL; + rte_edge_t edge = ~0; + int32_t user_data; + + arc = rte_graph_feature_arc_get(arcs[1]); + + if (rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should not have any feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_num_enabled_features(arcs[1])) { + printf("%s: Feature arc should not have any_feature() enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + /* + * On interface 0, enable feature 2, skip feature 1 for later + * On interface 1, enable feature 3 + * On interface 2, enable pkt_free feature + * On interface 3, continue as interface 0 + * + * later enable next feature sequence for interface 0 from feature2 -> pkt_free + * later enable next feature sequence for interface 1 from feature3 -> pkt_free + * + * also later enable feature-1 and see first feature changes for all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + n_features = (n_indexes % 3) + 1; /* feature2 to pkt_free are 3 features */ + feature_name = rte_graph_feature_arc_feature_to_name(arcs[1], n_features); + user_data = compute_unique_user_data(arc->start_node->name, feature_name, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, feature_name, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + /* has any feature should be valid */ + if (!rte_graph_feature_arc_has_any_feature(arc, &feature_list)) { + printf("%s: Feature arc should have any_feature enabled by now\n", + TEST_ARC2_NAME); + return TEST_FAILED; + } + } + /* Retrieve latest feature_list */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + /* verify first feature */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: No first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + /* Get first feature data and ensure edge and user_data are correct */ + fdata = rte_graph_feature_data_get(arc, rte_graph_feature_get(arc, feature), + n_indexes); + parent = arc->start_node; + if (0 == (n_indexes % 3)) + child = &arc2_feature2; + else if (1 == (n_indexes % 3)) + child = &arc2_feature3; + else + child = &pkt_free; + + if (get_edge(parent, child, &edge)) { + printf("%s: Edge not found between %s and %s\n", + TEST_ARC2_NAME, parent->name, child->name); + return TEST_FAILED; + } + if (fdata->next_edge != edge) { + printf("%s: Edge mismatch for first feature on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->next_edge, edge); + return TEST_FAILED; + } + if (fdata->user_data != compute_unique_user_data(parent->name, child->name, + n_indexes)) { + printf("%s: First feature user data mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, fdata->user_data, + compute_unique_user_data(parent->name, child->name, n_indexes)); + return TEST_FAILED; + } + } + /* add next_features now + * On interface 0, enable feature-3 and pkt_free + * On interface 1, enable pkt_free + * Skip interface 2 + * On interface 3, same as interface 0 + * On interface 4, same as interface 1 + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (0 == (n_indexes % 3)) { + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE3, + compute_unique_user_data(ARC2_FEATURE2, + ARC2_FEATURE3, + n_indexes))) { + printf("%s: Feature enable failed for %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + /* pkt_free on interface-0, 1, 3, 4 and so on */ + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_enable(arcs[1], n_indexes, PKT_FREE_STATIC, + compute_unique_user_data(ARC2_FEATURE3, + PKT_FREE_STATIC, + n_indexes))) { + printf("%s: Feature enable failed %s -> (%s) on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + /* Enable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + user_data = compute_unique_user_data(arc->start_node->name, ARC2_FEATURE1, + n_indexes); + if (rte_graph_feature_enable(arcs[1], n_indexes, ARC2_FEATURE1, + (int32_t)user_data)) { + printf("%s: Feature enable failed for %s on index %u\n", + TEST_ARC2_NAME, feature_name, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: None first feature enabled on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature != rte_graph_feature_cast(0)) { + printf("%s: First feature mismatch on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(0)); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_first_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* Disable feature-1 on all interfaces and check first feature changes */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE1)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE1, n_indexes); + return TEST_FAILED; + } + } + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (feature == rte_graph_feature_cast(0)) { + printf("%s: First feature not disabled on index %u [%u, exp: %u]\n", + TEST_ARC2_NAME, n_indexes, feature, rte_graph_feature_cast(1)); + return TEST_FAILED; + } + if (!strncmp(ARC2_FEATURE1, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(ARC2_FEATURE1))) { + printf("%s: First feature mismatch on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + ARC2_FEATURE2); + return TEST_FAILED; + } + } + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_next_feature_disable(void) +{ + rte_graph_feature_rt_list_t feature_list; + struct rte_graph_feature_arc *arc; + rte_graph_feature_t feature; + uint32_t n_indexes; + + arc = rte_graph_feature_arc_get(arcs[1]); + + /* + * On interface 0, disable feature 2, keep feature3 and pkt_free enabled + * On interface 1, skip interface 1 where feature3 and pkt_free are enabled + * skip interface 2 as only pkt_free is enabled + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!(n_indexes % 3)) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE2)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE2, n_indexes); + return TEST_FAILED; + } + } + + if (verify_feature_sequencing(arc) == TEST_FAILED) + return TEST_FAILED; + } + + /** + * Disable feature 3 on all interface 0 and 1 and check first feature + * is pkt_free on all indexes + */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if ((0 == (n_indexes % 3)) || (1 == (n_indexes % 3))) { + if (rte_graph_feature_disable(arcs[1], n_indexes, ARC2_FEATURE3)) { + printf("%s: Feature disable failed for %s on index %u\n", + TEST_ARC2_NAME, ARC2_FEATURE3, n_indexes); + return TEST_FAILED; + } + } + } + /* Make sure pkt_free is first feature for all indexes */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (!rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: First feature get failed on index: %u\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + if (strncmp(PKT_FREE_STATIC, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + strlen(PKT_FREE_STATIC))) { + printf("%s: %s is not first feature found on index %u [%s, exp: %s]\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes, + rte_graph_feature_arc_feature_to_name(arcs[1], feature), + PKT_FREE_STATIC); + return TEST_FAILED; + } + } + + /* Disable PKT_FREE_STATIC from all indexes with no feature enabled on any interface */ + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_disable(arcs[1], n_indexes, PKT_FREE_STATIC)) { + printf("%s: Feat disable failed for %s on index %u\n", + TEST_ARC2_NAME, PKT_FREE_STATIC, n_indexes); + return TEST_FAILED; + } + } + /* Make sure no feature is enabled now on any interface */ + rte_graph_feature_arc_has_any_feature(arc, &feature_list); + for (n_indexes = 0; n_indexes < MAX_INDEXES; n_indexes++) { + if (rte_graph_feature_arc_first_feature_get(arc, feature_list, n_indexes, + &feature)) { + printf("%s: Index: %u should not have first feature enabled\n", + TEST_ARC2_NAME, n_indexes); + return TEST_FAILED; + } + } + return TEST_SUCCESS; +} + +static int +test_graph_feature_arc_destroy(void) +{ + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC1_NAME, &arc)) { + printf("Feature arc lookup failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (arc != arcs[0]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC1_NAME, (void *)arc, (void *)arcs[0]); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC1_NAME); + return TEST_FAILED; + } + + if (rte_graph_feature_arc_lookup_by_name(TEST_ARC2_NAME, &arc)) { + printf("Feature arc lookup success after destroy for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + + if (arc != arcs[1]) { + printf("Feature arc lookup mismatch for %s [%p, exp: %p]\n", + TEST_ARC2_NAME, (void *)arc, (void *)arcs[1]); + return TEST_FAILED; + } + if (rte_graph_feature_arc_destroy(arc)) { + printf("Feature arc destroy failed for %s\n", TEST_ARC2_NAME); + return TEST_FAILED; + } + return TEST_SUCCESS; +} + +static int +graph_feature_arc_setup(void) +{ + unsigned long i, j; + + static const struct rte_mbuf_dynfield graph_dynfield_desc = { + .name = "test_graph_dynfield", + .size = sizeof(graph_dynfield_t), + .align = alignof(graph_dynfield_t), + }; + + graph_dynfield_offset = + rte_mbuf_dynfield_register(&graph_dynfield_desc); + if (graph_dynfield_offset < 0) { + printf("Cannot register mbuf field\n"); + return TEST_FAILED; + } + RTE_SET_USED(graph_dynfield_offset); + + for (i = 0; i <= MAX_NODES; i++) { + for (j = 0; j < MBUFF_SIZE; j++) + mbuf_p[i][j] = &mbuf[i][j]; + } + + return TEST_SUCCESS; + +} + +static void +graph_feature_arc_teardown(void) +{ + if (graph_id != RTE_GRAPH_ID_INVALID) + rte_graph_destroy(graph_id); + + rte_graph_feature_arc_cleanup(); +} + +static struct unit_test_suite graph_feature_arc_testsuite = { + .suite_name = "Graph Feature arc library test suite", + .setup = graph_feature_arc_setup, + .teardown = graph_feature_arc_teardown, + .unit_test_cases = { + TEST_CASE(test_graph_feature_arc_create), + TEST_CASE(test_graph_feature_arc_features_add), + TEST_CASE(test_graph_feature_arc_first_feature_enable), + TEST_CASE(test_graph_feature_arc_next_feature_enable), + TEST_CASE(test_graph_feature_arc_first_feature_disable), + TEST_CASE(test_graph_feature_arc_next_feature_disable), + TEST_CASE(test_graph_feature_arc_destroy), + TEST_CASES_END(), /**< NULL terminate unit test array */ + }, +}; + +static int +graph_feature_arc_autotest_fn(void) +{ + return unit_test_suite_runner(&graph_feature_arc_testsuite); +} + +REGISTER_FAST_TEST(graph_feature_arc_autotest, true, true, graph_feature_arc_autotest_fn); +#endif /* !RTE_EXEC_ENV_WINDOWS */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [PATCH v5 4/5] test/graph_feature_arc: add functional tests 2024-10-14 14:33 ` [PATCH v5 4/5] test/graph_feature_arc: add functional tests Nitin Saxena @ 2024-10-14 19:54 ` Stephen Hemminger 0 siblings, 0 replies; 55+ messages in thread From: Stephen Hemminger @ 2024-10-14 19:54 UTC (permalink / raw) To: Nitin Saxena Cc: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine, dev, Nitin Saxena On Mon, 14 Oct 2024 20:03:57 +0530 Nitin Saxena <nsaxena@marvell.com> wrote: > Added functional unit test case for verifying feature arc control plane > and fast path APIs > > How to run: > $ echo "graph_feature_arc_autotest" | ./bin/dpdk-test > > Signed-off-by: Nitin Saxena <nsaxena@marvell.com> With current upstream kernel checkpatch additional warnings: WARNING:MACRO_ARG_UNUSED: Argument 'idx' is not used in function-like macro #217: FILE: app/test/test_graph_feature_arc.c:186: +#define R(idx, node, node_cookie) { \ + if (!strcmp(child, node)) { \ + user_data += node_cookie; \ + } \ + } WARNING:MACRO_ARG_UNUSED: Argument 'user_data' is not used in function-like macro #272: FILE: app/test/test_graph_feature_arc.c:241: +#define R(idx, _name, user_data) { \ + if (!strcmp(node->name, _name)) { \ + priv->node_index = idx; \ + } \ + } Personally, using macros to generate tests like this can get confusing. ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v5 5/5] docs: add programming guide for feature arc 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena ` (3 preceding siblings ...) 2024-10-14 14:33 ` [PATCH v5 4/5] test/graph_feature_arc: add functional tests Nitin Saxena @ 2024-10-14 14:33 ` Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena 5 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2024-10-14 14:33 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Updated graph library guide with feature arc Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/guides/prog_guide/graph_lib.rst | 288 +++++++++++ doc/guides/prog_guide/img/feature_arc-1.svg | 277 +++++++++++ doc/guides/prog_guide/img/feature_arc-2.svg | 511 ++++++++++++++++++++ doc/guides/prog_guide/img/feature_arc-3.svg | 318 ++++++++++++ 4 files changed, 1394 insertions(+) create mode 100644 doc/guides/prog_guide/img/feature_arc-1.svg create mode 100644 doc/guides/prog_guide/img/feature_arc-2.svg create mode 100644 doc/guides/prog_guide/img/feature_arc-3.svg diff --git a/doc/guides/prog_guide/graph_lib.rst b/doc/guides/prog_guide/graph_lib.rst index ad09bdfe26..d98b48b0e5 100644 --- a/doc/guides/prog_guide/graph_lib.rst +++ b/doc/guides/prog_guide/graph_lib.rst @@ -547,3 +547,291 @@ on success packet is enqueued to ``udp4_input`` node. Hash lookup is performed in ``udp4_input`` node with registered destination port and destination port in UDP packet , on success packet is handed to ``udp_user_node``. + +Feature Arc +----------- +`Feature arc` represents an ordered list of `protocols/features` at a given +networking layer. It is a high level abstraction to connect various `feature` +nodes in `rte_graph` instance and allows seamless packets steering based on the +sequence of enabled features at runtime on each interface. + +`Features` (or feature nodes) are nodes which handle partial or complete +protocol processing in a given direction. For instance, `ipv4-rewrite` and +`IPv4 IPsec encryption` are outbound features while `ipv4-lookup` and `IPv4 +IPsec decryption` are inbound features. Further, `ipv4-rewrite` and `IPv4 +IPsec encryption` can collectively represent a `feature arc` towards egress +direction with ordering constraints that `IPv4 IPsec encryption` must be +performed before `ipv4-rewrite`. Similarly, `IPv4 IPsec decryption` and +`ipv4-lookup` can represent a `feature arc` in an ingress direction. Both of +these `feature arc` can co-exist at an IPv4 layer in egress and ingress +direction respectively. + +A `feature` can be represented by a single node or collection of multiple nodes +performing feature processing collectively. + +.. figure:: img/feature_arc-1.* + :alt: feature-arc-1 + :width: 350px + :align: center + + Feature Arc overview + +Each `feature arc` is associated with a `Start` node from which all features in +a feature arc are connected. A `start` node itself is not a `feature` node but +it is where `first enabled feature` is checked in fast path. In above figure, +`Node-A` represents a `start node`. There may be a `Sink` node as well which +is child node for every feature in an arc. 'Sink` node is responsible of +consuming those packets which are not consumed by intermediate enabled features +between `start` and `sink` node. `Sink` node, if present, is the last enabled +feature in a feature arc. A `feature` node statically connected to `start` node +must also be added via feature arc API, `rte_graph_feature_add()``. Here `Node-B` +acts as a `sink` node which is statically linked to `Node A`. `Feature` nodes +are connected via `rte_graph_feature_add()` which takes care of connecting +all `feature` nodes with each other and start node. + +.. code-block:: bash + :linenos: + :emphasize-lines: 8 + :caption: Node-B statically linked to Node-A + + static struct rte_node_register node_A_node = { + .process = node_A_process_func, + ... + ... + .name = "Node-A", + .next_nodes = + { + [0] = "Node-B", + }, + .nb_edges = 1, + }; + +When multiple features are enabled on an interface, it may be required to steer +packets across `features` in a given order. For instance, if `Feature 1` and +`Feature 2` both are enabled on an interface ``X``, it may be required to send +packets to `Feature-1` before `Feature-2`. Such ordering constraints can be +easily expressed with `feature arc`. In this case, `Feature 1` is called as +``First Feature`` and `Feature 2` is called as ``Next Feature`` to `Feature 1`. + +.. figure:: img/feature_arc-2.* + :alt: feature-arc-2 + :width: 600px + :align: center + + First and Next features and their ordering + +In similar manner, even application specific ``custom features`` can be hooked +to standard nodes. It is to be noted that this `custom feature` hooking to +`feature arc` aware node does not require any code changes. + +It may be obvious by now that `features` enabled on one interface does not +affect packets on other interfaces. In above example, if no feature is +enabled on an interface ``X``, packets destined to interface ``X`` would be +directly sent to `Node-B` from `Node-A`. + +.. figure:: img/feature_arc-3.* + :alt: feature-arc-3 + :width: 550px + :align: center + + Feature-2 consumed/non-consumed packet path + +When a `Feature-X` node receives packets via feature arc, it may decide whether +to ``consume packet`` or send to `next enabled feature`. A node can consume +packet by freeing it, sending it on wire or enqueuing it to hardware queue. If a +packet is not consumed by a `Feature-X` node, it may send to `next enabled +feature` on an interface. In above figure, `Feature-2` nodes are represented to +consume packets. Classic example for a node performing consume and non-consume +operation on packets would be IPsec policy node where all packets with +``protect`` actions are consumed while remaining packets with ``bypass`` +actions are sent to next enabled feature. + +In fast path feature node may require to lookup local data structures for each +interface. For example, retrieving policy database per interface for IPsec +processing. ``rte_graph_feature_enable`` API allows to set application +specific cookie per feature per interface. `Feature data` object maintains this +cookie in fast path for each interface. + +`Feature arc design` allows to enable subsequent features in a control plane +without stopping workers which are accessing feature arc's fast path APIs in +``rte_graph_walk()`` context. However for disabling features require RCU like +scheme for synchronization. + +Programming model +~~~~~~~~~~~~~~~~~ +Feature Arc Objects +^^^^^^^^^^^^^^^^^^^ +Control plane and fast path APIs deals with following objects: + +Feature arc +*********** +``rte_graph_feature_arc_t`` is a handle to feature arc which is created via +``rte_graph_feature_arc_create()``. It is a `uint64_t` size object which can be +saved in feature node's context. This object can be translated to fast path +feature arc object ``struct rte_graph_feature_arc`` which is an input +argument to all fast path APIs. Control plane APIs majorly takes +`rte_graph_feature_arc_t` object as an input. + +Feature List +************ +Each feature arc holds two feature lists: `active` and `passive`. While worker +cores uses `active` list, control plane APIs uses `passive` list for +enabling/disabling a feature on any interface with in a arc. After successful +feature enable/disable, ``rte_graph_feature_enable()``/ +``rte_graph_feature_disable()`` atomically switches passive list to active list +and vice-versa. Most of the fast path APIs takes active list as an argument +(``rte_graph_feature_rt_list_t``), which feature node can obtain in start of +it's `process_func()` via ``rte_graph_feature_arc_has_any_feature()`` (in `start` +node) or ``rte_graph_feature_arc_has_feature()`` (in next feature nodes). + +Each feature list holds RTE_GRAPH_MAX_FEATURES number of features and +associated feature data for every interface index + +Feature +******** +Feature is a data structure which holds `feature data` object for every +interface. It is represented via ``rte_graph_feature_t`` which is a `uint8_t` +size object. Fast path internal structure ``struct rte_graph_feature`` can be +obtained from ``rte_graph_feature_t`` via ``rte_graph_feature_get()`` API. + +In `start` node `rte_graph_feature_arc_first_feature_get()` can be used to get +first enabled `rte_graph_feature_t` object for an interface. `rte_edge` from +`start` node to first enabled feature is provided by +``rte_graph_feature_arc_feature_set()`` API. + +In `feature nodes`, next enabled feature is obtained by providing current feature +as an input to ``rte_graph_feature_arc_next_feature_get()`` API. + +Feature data +************ +Feature data object is maintained per feature per interface which holds +following information in fast path + +- ``rte_edge_t`` to send packet to next enabled feature +- ``Next enabled feature`` on current interface +- ``User_data`` per feature per interface set by application via `rte_graph_feature_enable()` + +Enabling Feature Arc processing +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +By default, feature arc processing is disabled in `rte_graph_create()`. To +enable feature arc processing in fast path, `rte_graph_create()` API should be +invoked with `feature_arc_enable` flag set as `true` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Enabling feature are processing in rte_graph_create() + + struct rte_graph_param graph_conf; + + graph_conf.feature_arc_enable = true; + struct rte_graph *graph = rte_graph_create("graph_name", &graph_conf); + +Further as an optimization technique, `rte_graph_walk()` would call newly added +``feat_arc_proc()`` node callback function (if non-NULL) instead of +``process`` + +.. code-block:: bash + :linenos: + :emphasize-lines: 3 + :caption: Feature arc specific node callback function + + static struct rte_node_register ip4_rewrite_node = { + .process = ip4_rewrite_node_process, + .feat_arc_proc = ip4_rewrite_feature_node_process, + .name = "ip4_rewrite", + ... + ... + }; + +If `feat_arc_proc` is not provided in node registration, `process_func` would +be called by `rte_graph_walk()` + +Sample Usage +^^^^^^^^^^^^ +.. code-block:: bash + :linenos: + :caption: Feature arc sample usage + + #define MAX_FEATURES 10 + #define MAX_INDEXES 5 + + static uint16_t + feature2_feature_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* features may be enabled */ + } + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc is disabled in rte_graph_create() */ + } + + static uint16_t + feature2_node_process (struct rte_graph *graph, struct + rte_node *node, void **objs, uint16_t nb_objs) + { + /* Feature arc may be enabled or disabled as this process_func() would + * be called for the case when feature arc is enabled in rte_graph_create() + * and also the case when it is disabled + */ + } + + static struct rte_node_register feature2_node = { + .process = feature2_node_process, + .feat_arc_proc = feature2_feature_node_process, + .name = "feature2", + .init = feature2_init_func, + ... + ... + }; + + static struct rte_node_register feature1_node = { + .process = feature1_node_process, + .feat_arc_proc = NULL, + .name = "feature1", + ... + ... + }; + + int worker_cb(void *_em) + { + rte_graph_feature_arc_t arc; + uint32_t user_data; + + rte_graph_feature_arc_lookup_by_name("sample_arc", &arc); + + /* From control thread context (like CLII): + * Enable feature 2 on interface index 4 + */ + if (rte_lcore_id() == rte_get_main_lcore) { + user_data = 0x1234; + rte_graph_feature_enable(arc, 4 /* interface index */, "feature2", user_data); + } else { + while(1) + rte_graph_walk); + } + } + + int main(void) + { + struct rte_graph_param graph_conf; + rte_graph_feature_arc_t arc; + + if (rte_graph_feature_arc_create("sample_arc", MAX_FEATURES, MAX_INDEXES, &arc)) + return -1; + + rte_graph_feature_add(arc, "feature1", NULL, NULL); + rte_graph_feature_add(arc, "feature2", "feature1" /* add feature2 after feature 1*/, NULL); + + /* create graph*/ + ... + ... + graph_conf.feature_arc_enable = true; + + struct rte_graph *graph = rte_graph_create("sample_graph", &graph_conf); + + rte_eal_mp_remote_launch(worker_cb, arg, CALL_MAIN); + } diff --git a/doc/guides/prog_guide/img/feature_arc-1.svg b/doc/guides/prog_guide/img/feature_arc-1.svg new file mode 100644 index 0000000000..dada169e11 --- /dev/null +++ b/doc/guides/prog_guide/img/feature_arc-1.svg @@ -0,0 +1,277 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> +<!-- SPDX-License-Identifier: BSD-3-Clause --> +<!-- Copyright(C) 2024 Marvell International Ltd. --> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + id="svg4977" + version="1.1" + overflow="hidden" + xml:space="preserve" + height="1455" + width="1995"><metadata + id="metadata4983"><rdf:RDF><cc:Work + rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /><dc:title></dc:title></cc:Work></rdf:RDF></metadata><defs + id="defs4981" /><g + id="g4975" + transform="translate(-637 -359)"><path + id="path4901" + fill-rule="evenodd" + fill="#00B5E2" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M640.5 543.5C640.5 460.657 788.917 393.5 972 393.5 1155.08 393.5 1303.5 460.657 1303.5 543.5 1303.5 626.343 1155.08 693.5 972 693.5 788.917 693.5 640.5 626.343 640.5 543.5Z" /><text + id="text4903" + transform="matrix(1 0 0 1 831.669 574)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Node</text> +<text + id="text4905" + transform="matrix(1 0 0 1 1028.75 574)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text4907" + transform="matrix(1 0 0 1 1056.25 574)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">A</text> +<path + id="path4909" + fill-rule="evenodd" + fill="#00B5E2" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M640.5 1660.5C640.5 1577.66 788.917 1510.5 972 1510.5 1155.08 1510.5 1303.5 1577.66 1303.5 1660.5 1303.5 1743.34 1155.08 1810.5 972 1810.5 788.917 1810.5 640.5 1743.34 640.5 1660.5Z" /><text + id="text4911" + transform="matrix(1 0 0 1 831.669 1691)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Node</text> +<text + id="text4913" + transform="matrix(1 0 0 1 1028.75 1691)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text4915" + transform="matrix(1 0 0 1 1056.25 1691)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">B</text> +<path + id="path4917" + fill-rule="evenodd" + fill="#00B050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M1303.5 1132.5C1303.5 1049.66 1452.14 982.5 1635.5 982.5 1818.86 982.5 1967.5 1049.66 1967.5 1132.5 1967.5 1215.34 1818.86 1282.5 1635.5 1282.5 1452.14 1282.5 1303.5 1215.34 1303.5 1132.5Z" /><text + id="text4919" + transform="matrix(1 0 0 1 1455.56 1159)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Feature </text> +<text + id="text4921" + transform="matrix(1 0 0 1 1728.84 1159)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text4923" + transform="matrix(1 0 0 1 1774.1 1159)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">1</text> +<path + id="path4925" + fill="#63666A" + d="M977.875 693 977.875 1475.28 964.125 1475.28 964.125 693ZM991.625 1468.41 971 1509.66 950.375 1468.41Z" /><path + id="path4927" + fill="#003967" + d="M972.869 690.347 998.093 701.303 995.354 707.609 970.13 696.653ZM1017.01 709.52 1042.23 720.476 1039.49 726.782 1014.27 715.826ZM1061.15 728.693 1086.37 739.649 1083.64 745.955 1058.41 734.999ZM1105.29 747.866 1130.52 758.822 1127.78 765.127 1102.55 754.172ZM1149.43 767.039 1174.66 777.995 1171.92 784.3 1146.69 773.344ZM1193.57 786.211 1218.8 797.167 1216.06 803.473 1190.83 792.517ZM1237.71 805.384 1262.94 816.34 1260.2 822.646 1234.98 811.69ZM1281.86 824.557 1307.08 835.513 1304.34 841.819 1279.12 830.863ZM1326 843.73 1351.22 854.686 1348.48 860.992 1323.26 850.036ZM1370.14 862.903 1395.36 873.859 1392.62 880.165 1367.4 869.209ZM1414.28 882.076 1439.5 893.032 1436.76 899.338 1411.54 888.382ZM1458.42 901.249 1483.64 912.205 1480.9 918.511 1455.68 907.555ZM1502.56 920.422 1527.78 931.378 1525.04 937.684 1499.82 926.728ZM1546.7 939.595 1571.92 950.551 1569.18 956.857 1543.96 945.901ZM1590.84 958.768 1615.56 969.504 1612.82 975.81 1588.1 965.074ZM1615.46 958.219 1635.21 981.786 1604.51 983.442Z" /><path + id="path4929" + transform="matrix(-1 0 0 1 1635.21 1282.5)" + fill="#003967" + d="M1.11695-3.25097 27.1247 5.68467 24.8908 12.1866-1.11695 3.25097ZM46.6306 12.3864 72.6383 21.322 70.4044 27.824 44.3967 18.8883ZM92.1442 28.0238 118.152 36.9594 115.918 43.4613 89.9103 34.5257ZM137.658 43.6611 163.666 52.5968 161.432 59.0987 135.424 50.1631ZM183.171 59.2985 209.179 68.2341 206.945 74.7361 180.937 65.8004ZM228.685 74.9359 254.693 83.8715 252.459 90.3734 226.451 81.4378ZM274.199 90.5732 300.206 99.5089 297.972 106.011 271.965 97.0752ZM319.712 106.211 345.72 115.146 343.486 121.648 317.478 112.713ZM365.226 121.848 391.234 130.784 389 137.286 362.992 128.35ZM410.739 137.485 436.747 146.421 434.513 152.923 408.505 143.987ZM456.253 153.123 482.261 162.058 480.027 168.56 454.019 159.625ZM501.767 168.76 527.774 177.696 525.54 184.198 499.533 175.262ZM547.28 184.397 573.288 193.333 571.054 199.835 545.046 190.899ZM592.794 200.035 618.802 208.97 616.568 215.472 590.56 206.537ZM638.307 215.672 643.152 217.337 640.918 223.839 636.073 222.174ZM642.168 206.094 663.708 228.034 633.232 232.102Z" /><text + id="text4931" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 943.025 1309)" + font-size="55" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Static Packet Path</text> +<text + id="text4933" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 1009.03 1200)" + font-size="55" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0</text> +<text + id="text4935" + transform="matrix(0.916122 0.4009 -0.4009 0.916122 1103.93 809)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 1</text> +<text + id="text4937" + transform="matrix(1 0 0 1 1522.09 429)" + font-size="55" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Source/Start </text> +<text + id="text4939" + transform="matrix(1 0 0 1 1620.06 495)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Node</text> +<text + id="text4941" + transform="matrix(1 0 0 1 1618.96 1583)" + font-size="55" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">End/Sink</text> +<text + id="text4943" + transform="matrix(1 0 0 1 1671.66 1649)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Node</text> +<text + id="text4945" + transform="matrix(1 0 0 1 2196.79 1078)" + font-size="55" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Feature</text> +<text + id="text4947" + transform="matrix(1 0 0 1 2230.59 1144)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Node</text> +<path + id="path4949" + transform="matrix(-1 0 0 1 1486.34 450.5)" + fill="#63666A" + d="M0.259414-0.510821 162.664 81.9641 162.145 82.9858-0.259414 0.510821ZM164.544 68.1399 182.837 92.8515 152.092 92.6593Z" /><path + id="path4951" + transform="matrix(-1 0 0 1 1537.94 1604.5)" + fill="#63666A" + d="M0.133795-0.557075 212.291 50.3976 212.023 51.5117-0.133795 0.557075ZM210.911 36.5145 234.44 56.3064 204.489 63.2541Z" /><path + id="path4953" + transform="matrix(-1 0 0 1 2159.31 1099.5)" + fill="#63666A" + d="M0.0965444-0.564724 169.317 28.3649 169.124 29.4944-0.0965444 0.564724ZM167.019 14.6039 191.809 32.7914 162.385 41.7106Z" /><path + id="path4955" + transform="matrix(-1 0 0 1 1737.41 724.5)" + fill="#63666A" + d="M0.229683-0.524861 295.142 128.531 294.683 129.581-0.229683 0.524861ZM296.226 114.622 315.907 138.243 285.201 139.815Z" /><text + id="text4957" + transform="matrix(1 0 0 1 1833.68 703)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Packet path gets activated in </text> +<text + id="text4959" + transform="matrix(1 0 0 1 1865.62 769)" + text-decoration="underline" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Node</text> +<text + id="text4961" + transform="matrix(1 0 0 1 1996.25 769)" + text-decoration="underline" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text4963" + transform="matrix(1 0 0 1 2014.58 769)" + text-decoration="underline" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">A</text> +<text + id="text4965" + transform="matrix(1 0 0 1 2064.68 769)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">when Feature</text> +<text + id="text4967" + transform="matrix(1 0 0 1 2399.83 769)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text4969" + transform="matrix(1 0 0 1 2418.17 769)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">1 is </text> +<text + id="text4971" + transform="matrix(1 0 0 1 1898.42 835)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">enabled on </text> +<text + id="text4973" + transform="matrix(1 0 0 1 2184.3 835)" + text-decoration="underline" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">an interface</text> +</g></svg> diff --git a/doc/guides/prog_guide/img/feature_arc-2.svg b/doc/guides/prog_guide/img/feature_arc-2.svg new file mode 100644 index 0000000000..eb5fb37e04 --- /dev/null +++ b/doc/guides/prog_guide/img/feature_arc-2.svg @@ -0,0 +1,511 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> +<!-- SPDX-License-Identifier: BSD-3-Clause --> +<!-- Copyright(C) 2024 Marvell International Ltd. --> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + id="svg152" + version="1.1" + overflow="hidden" + xml:space="preserve" + height="1483" + width="3471"><metadata + id="metadata158"><rdf:RDF><cc:Work + rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /><dc:title></dc:title></cc:Work></rdf:RDF></metadata><defs + id="defs156" /><g + id="g150" + transform="translate(-637 -331)"><path + id="path2" + fill-rule="evenodd" + fill="#00B5E2" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M640.5 543.5C640.5 460.657 788.917 393.5 972 393.5 1155.08 393.5 1303.5 460.657 1303.5 543.5 1303.5 626.343 1155.08 693.5 972 693.5 788.917 693.5 640.5 626.343 640.5 543.5Z" /><text + id="text4" + transform="matrix(1 0 0 1 831.669 574)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Node</text> +<text + id="text6" + transform="matrix(1 0 0 1 1028.75 574)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text8" + transform="matrix(1 0 0 1 1056.25 574)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">A</text> +<path + id="path10" + fill-rule="evenodd" + fill="#00B5E2" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M640.5 1660.5C640.5 1577.66 788.917 1510.5 972 1510.5 1155.08 1510.5 1303.5 1577.66 1303.5 1660.5 1303.5 1743.34 1155.08 1810.5 972 1810.5 788.917 1810.5 640.5 1743.34 640.5 1660.5Z" /><text + id="text12" + transform="matrix(1 0 0 1 831.669 1691)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Node</text> +<text + id="text14" + transform="matrix(1 0 0 1 1028.75 1691)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text16" + transform="matrix(1 0 0 1 1056.25 1691)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">B</text> +<path + id="path18" + fill-rule="evenodd" + fill="#00B050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M1303.5 1132.5C1303.5 1049.66 1452.14 982.5 1635.5 982.5 1818.86 982.5 1967.5 1049.66 1967.5 1132.5 1967.5 1215.34 1818.86 1282.5 1635.5 1282.5 1452.14 1282.5 1303.5 1215.34 1303.5 1132.5Z" /><text + id="text20" + transform="matrix(1 0 0 1 1455.56 1159)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Feature </text> +<text + id="text22" + transform="matrix(1 0 0 1 1728.84 1159)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text24" + transform="matrix(1 0 0 1 1774.1 1159)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">1</text> +<path + id="path26" + fill="#63666A" + d="M974.937 693.5 974.938 1487.24 968.063 1487.24 968.062 693.5ZM985.25 1482.66 971.5 1510.16 957.75 1482.66Z" /><path + id="path28" + fill="#003967" + d="M972.869 690.347 998.093 701.303 995.354 707.609 970.13 696.653ZM1017.01 709.52 1042.23 720.476 1039.49 726.782 1014.27 715.826ZM1061.15 728.693 1086.37 739.649 1083.64 745.955 1058.41 734.999ZM1105.29 747.866 1130.52 758.822 1127.78 765.127 1102.55 754.172ZM1149.43 767.039 1174.66 777.995 1171.92 784.3 1146.69 773.344ZM1193.57 786.211 1218.8 797.167 1216.06 803.473 1190.83 792.517ZM1237.71 805.384 1262.94 816.34 1260.2 822.646 1234.98 811.69ZM1281.86 824.557 1307.08 835.513 1304.34 841.819 1279.12 830.863ZM1326 843.73 1351.22 854.686 1348.48 860.992 1323.26 850.036ZM1370.14 862.903 1395.36 873.859 1392.62 880.165 1367.4 869.209ZM1414.28 882.076 1439.5 893.032 1436.76 899.338 1411.54 888.382ZM1458.42 901.249 1483.64 912.205 1480.9 918.511 1455.68 907.555ZM1502.56 920.422 1527.78 931.378 1525.04 937.684 1499.82 926.728ZM1546.7 939.595 1571.92 950.551 1569.18 956.857 1543.96 945.901ZM1590.84 958.768 1615.56 969.504 1612.82 975.81 1588.1 965.074ZM1615.46 958.219 1635.21 981.786 1604.51 983.442Z" /><path + id="path30" + transform="matrix(-1 0 0 1 1400.55 1238.5)" + fill="#003967" + d="M1.84062-2.90319 25.0662 11.8217 21.3849 17.6281-1.84062 2.90319ZM42.4853 22.8654 65.7109 37.5903 62.0297 43.3967 38.8041 28.6718ZM83.1301 48.634 106.356 63.359 102.674 69.1653 79.4488 54.4404ZM123.775 74.4026 147 89.1276 143.319 94.934 120.094 80.209ZM164.42 100.171 187.645 114.896 183.964 120.703 160.738 105.978ZM205.064 125.94 228.29 140.665 224.609 146.471 201.383 131.746ZM245.709 151.708 268.935 166.433 265.253 172.24 242.028 157.515ZM286.354 177.477 309.579 192.202 305.898 198.008 282.672 183.283ZM326.998 203.246 350.224 217.971 346.543 223.777 323.317 209.052ZM367.643 229.014 390.869 243.739 387.188 249.546 363.962 234.821ZM408.288 254.783 411.537 256.843 407.856 262.649 404.607 260.589ZM413.188 245.68 429.052 272.017 398.464 268.905Z" /><text + id="text32" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 961.015 1340)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Static Packet</text> +<text + id="text34" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 961.015 945)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Path</text> +<text + id="text36" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 1040.01 1221)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0</text> +<text + id="text38" + transform="matrix(0.916122 0.4009 -0.4009 0.916122 1100.07 807)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 1</text> +<path + id="path40" + fill-rule="evenodd" + fill="#A6A6A6" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M3429.5 1088C3429.5 1004.88 3577.92 937.5 3761 937.5 3944.08 937.5 4092.5 1004.88 4092.5 1088 4092.5 1171.12 3944.08 1238.5 3761 1238.5 3577.92 1238.5 3429.5 1171.12 3429.5 1088Z" /><text + id="text42" + transform="matrix(1 0 0 1 3583.93 1068)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Custom </text> +<text + id="text44" + transform="matrix(1 0 0 1 3891.02 1068)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">–</text> +<text + id="text46" + transform="matrix(1 0 0 1 3618.31 1167)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Feature</text> +<path + id="path48" + fill="#003967" + d="M971.811 690.076 999.198 692.567 998.576 699.413 971.189 696.923ZM1019.74 694.434 1047.13 696.924 1046.5 703.771 1019.12 701.281ZM1067.67 698.792 1095.05 701.282 1094.43 708.129 1067.04 705.639ZM1115.59 703.15 1142.98 705.64 1142.36 712.487 1114.97 709.996ZM1163.52 707.507 1190.91 709.997 1190.28 716.844 1162.9 714.354ZM1211.45 711.865 1238.83 714.355 1238.21 721.202 1210.83 718.712ZM1259.38 716.223 1286.76 718.713 1286.14 725.56 1258.75 723.07ZM1307.3 720.581 1334.69 723.071 1334.07 729.917 1306.68 727.427ZM1355.23 724.938 1382.62 727.428 1381.99 734.275 1354.61 731.785ZM1403.16 729.296 1430.54 731.786 1429.92 738.633 1402.53 736.143ZM1451.08 733.654 1478.47 736.144 1477.85 742.991 1450.46 740.5ZM1499.01 738.011 1526.4 740.502 1525.78 747.348 1498.39 744.858ZM1546.94 742.369 1574.33 744.859 1573.7 751.706 1546.32 749.216ZM1594.87 746.727 1622.25 749.217 1621.63 756.064 1594.24 753.574ZM1642.79 751.085 1670.18 753.575 1669.56 760.421 1642.17 757.931ZM1690.72 755.442 1718.11 757.932 1717.49 764.779 1690.1 762.289ZM1738.65 759.8 1766.04 762.29 1765.41 769.137 1738.03 766.647ZM1786.58 764.158 1813.96 766.648 1813.34 773.495 1785.95 771.004ZM1834.5 768.515 1861.89 771.006 1861.27 777.852 1833.88 775.362ZM1882.43 772.873 1909.82 775.363 1909.19 782.21 1881.81 779.72ZM1930.36 777.231 1957.74 779.721 1957.12 786.568 1929.73 784.078ZM1978.28 781.589 2005.67 784.079 2005.05 790.925 1977.66 788.435ZM2026.21 785.946 2053.6 788.436 2052.98 795.283 2025.59 792.793ZM2074.14 790.304 2101.53 792.794 2100.9 799.641 2073.52 797.151ZM2122.07 794.662 2149.45 797.152 2148.83 803.999 2121.44 801.509ZM2169.99 799.019 2197.38 801.51 2196.76 808.356 2169.37 805.866ZM2217.92 803.377 2245.31 805.867 2244.69 812.714 2217.3 810.224ZM2265.85 807.735 2293.24 810.225 2292.61 817.072 2265.23 814.582ZM2313.78 812.093 2341.16 814.583 2340.54 821.43 2313.15 818.939ZM2361.7 816.45 2389.09 818.94 2388.47 825.787 2361.08 823.297ZM2409.63 820.808 2437.02 823.298 2436.4 830.145 2409.01 827.655ZM2457.56 825.166 2484.95 827.656 2484.32 834.503 2456.94 832.013ZM2505.49 829.523 2532.87 832.014 2532.25 838.86 2504.86 836.37ZM2553.41 833.881 2580.8 836.371 2580.18 843.218 2552.79 840.728ZM2601.34 838.239 2628.73 840.729 2628.1 847.576 2600.72 845.086ZM2649.27 842.597 2676.65 845.087 2676.03 851.934 2648.64 849.443ZM2697.19 846.954 2724.58 849.445 2723.96 856.291 2696.57 853.801ZM2745.12 851.312 2772.51 853.802 2771.89 860.649 2744.5 858.159ZM2793.05 855.67 2820.44 858.16 2819.81 865.007 2792.43 862.517ZM2840.98 860.028 2868.36 862.518 2867.74 869.364 2840.35 866.874ZM2888.9 864.385 2916.29 866.875 2915.67 873.722 2888.28 871.232ZM2936.83 868.743 2964.22 871.233 2963.59 878.08 2936.21 875.59ZM2984.76 873.101 3012.14 875.591 3011.52 882.438 2984.14 879.947ZM3032.68 877.458 3060.07 879.949 3059.45 886.795 3032.06 884.305ZM3080.61 881.816 3108 884.306 3107.38 891.153 3079.99 888.663ZM3128.54 886.174 3155.93 888.664 3155.3 895.511 3127.92 893.021ZM3176.47 890.532 3203.85 893.022 3203.23 899.869 3175.84 897.378ZM3224.39 894.889 3251.78 897.379 3251.16 904.226 3223.77 901.736ZM3272.32 899.247 3299.71 901.737 3299.09 908.584 3271.7 906.094ZM3320.25 903.605 3347.64 906.095 3347.01 912.942 3319.63 910.452ZM3368.18 907.963 3395.56 910.453 3394.94 917.299 3367.55 914.809ZM3416.1 912.32 3443.49 914.81 3442.87 921.657 3415.48 919.167ZM3464.03 916.678 3491.42 919.168 3490.8 926.015 3463.41 923.525ZM3511.96 921.036 3539.34 923.526 3538.72 930.373 3511.34 927.882ZM3559.89 925.393 3587.27 927.884 3586.65 934.73 3559.26 932.24ZM3607.81 929.751 3630.27 931.793 3629.65 938.64 3607.19 936.598ZM3626.64 921.108 3652.78 937.292 3624.15 948.495Z" /><text + id="text50" + transform="matrix(0.993978 0.109578 -0.109578 0.993978 1324.46 704)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 3</text> +<path + id="path52" + fill-rule="evenodd" + fill="#00B050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M2326.5 1136.5C2326.5 1053.66 2475.14 986.5 2658.5 986.5 2841.86 986.5 2990.5 1053.66 2990.5 1136.5 2990.5 1219.34 2841.86 1286.5 2658.5 1286.5 2475.14 1286.5 2326.5 1219.34 2326.5 1136.5Z" /><text + id="text54" + transform="matrix(1 0 0 1 2478.16 1163)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">Feature </text> +<text + id="text56" + transform="matrix(1 0 0 1 2751.44 1163)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">-</text> +<text + id="text58" + transform="matrix(1 0 0 1 2796.7 1163)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#FFFFFF">2</text> +<path + id="path60" + fill="#003967" + d="M972.27 690.15 999.071 696.312 997.53 703.012 970.73 696.85ZM1019.17 700.934 1045.97 707.096 1044.43 713.796 1017.63 707.634ZM1066.07 711.717 1092.87 717.88 1091.33 724.58 1064.53 718.417ZM1112.97 722.501 1139.77 728.663 1138.23 735.363 1111.43 729.201ZM1159.88 733.285 1186.68 739.447 1185.14 746.147 1158.33 739.985ZM1206.78 744.069 1233.58 750.231 1232.04 756.931 1205.24 750.769ZM1253.68 754.853 1280.48 761.015 1278.94 767.715 1252.14 761.553ZM1300.58 765.636 1327.38 771.799 1325.84 778.499 1299.04 772.336ZM1347.48 776.42 1374.28 782.582 1372.74 789.282 1345.94 783.12ZM1394.38 787.204 1421.18 793.366 1419.64 800.066 1392.84 793.904ZM1441.28 797.988 1468.08 804.15 1466.54 810.85 1439.74 804.688ZM1488.18 808.772 1514.98 814.934 1513.44 821.634 1486.64 815.472ZM1535.08 819.555 1561.89 825.718 1560.34 832.418 1533.54 826.255ZM1581.99 830.339 1608.79 836.501 1607.25 843.201 1580.45 837.039ZM1628.89 841.123 1655.69 847.285 1654.15 853.985 1627.35 847.823ZM1675.79 851.907 1702.59 858.069 1701.05 864.769 1674.25 858.607ZM1722.69 862.691 1749.49 868.853 1747.95 875.553 1721.15 869.391ZM1769.59 873.474 1796.39 879.637 1794.85 886.337 1768.05 880.175ZM1816.49 884.258 1843.29 890.42 1841.75 897.121 1814.95 890.958ZM1863.39 895.042 1890.19 901.204 1888.65 907.904 1861.85 901.742ZM1910.29 905.826 1937.1 911.988 1935.55 918.688 1908.75 912.526ZM1957.2 916.61 1984 922.772 1982.46 929.472 1955.66 923.31ZM2004.1 927.393 2030.9 933.556 2029.36 940.256 2002.56 934.094ZM2051 938.177 2077.8 944.339 2076.26 951.04 2049.46 944.877ZM2097.9 948.961 2124.7 955.123 2123.16 961.823 2096.36 955.661ZM2144.8 959.745 2171.6 965.907 2170.06 972.607 2143.26 966.445ZM2191.7 970.529 2218.5 976.691 2216.96 983.391 2190.16 977.229ZM2238.6 981.312 2265.4 987.475 2263.86 994.175 2237.06 988.013ZM2285.5 992.096 2312.31 998.258 2310.76 1004.96 2283.96 998.796ZM2332.41 1002.88 2359.21 1009.04 2357.67 1015.74 2330.87 1009.58ZM2379.31 1013.66 2406.06 1019.81 2404.52 1026.52 2377.77 1020.36ZM2403.91 1008.74 2427.62 1028.3 2397.74 1035.54Z" /><path + id="path62" + transform="matrix(-1 0 0 1 2658.15 1286.5)" + fill="#003967" + d="M0.62373-3.38044 27.6672 1.6094 26.4198 8.37028-0.62373 3.38044ZM47.9499 5.35178 74.9934 10.3416 73.7459 17.1025 46.7024 12.1127ZM95.276 14.084 122.32 19.0738 121.072 25.8347 94.0286 20.8449ZM142.602 22.8162 169.646 27.8061 168.398 34.5669 141.355 29.5771ZM189.928 31.5484 216.972 36.5383 215.724 43.2991 188.681 38.3093ZM237.254 40.2806 264.298 45.2705 263.051 52.0314 236.007 47.0415ZM284.581 49.0129 311.624 54.0027 310.377 60.7636 283.333 55.7737ZM331.907 57.7451 358.95 62.7349 357.703 69.4958 330.659 64.506ZM379.233 66.4773 406.276 71.4671 405.029 78.228 377.985 73.2382ZM426.559 75.2095 453.603 80.1994 452.355 86.9602 425.312 81.9704ZM473.885 83.9417 500.929 88.9316 499.681 95.6925 472.638 90.7026ZM521.211 92.674 548.255 97.6638 547.008 104.425 519.964 99.4348ZM568.538 101.406 595.581 106.396 594.334 113.157 567.29 108.167ZM615.864 110.138 642.907 115.128 641.66 121.889 614.616 116.899ZM663.19 118.871 690.233 123.86 688.986 130.621 661.942 125.631ZM710.516 127.603 737.56 132.593 736.312 139.354 709.269 134.364ZM757.842 136.335 784.886 141.325 783.638 148.086 756.595 143.096ZM805.168 145.067 832.212 150.057 830.964 156.818 803.921 151.828ZM852.494 153.799 879.538 158.789 878.29 165.55 851.247 160.56ZM899.82 162.532 926.864 167.522 925.617 174.282 898.573 169.293ZM947.147 171.264 974.19 176.254 972.943 183.015 945.899 178.025ZM994.473 179.996 1021.52 184.986 1020.27 191.747 993.225 186.757ZM1041.8 188.728 1068.84 193.718 1067.59 200.479 1040.55 195.489ZM1089.12 197.461 1116.17 202.45 1114.92 209.211 1087.88 204.221ZM1136.45 206.193 1163.49 211.183 1162.25 217.943 1135.2 212.954ZM1183.78 214.925 1210.82 219.915 1209.57 226.676 1182.53 221.686ZM1231.1 223.657 1258.15 228.647 1256.9 235.408 1229.86 230.418ZM1278.43 232.389 1305.47 237.379 1304.23 244.14 1277.18 239.15ZM1325.76 241.122 1352.8 246.111 1351.55 252.872 1324.51 247.883ZM1373.08 249.854 1400.13 254.844 1398.88 261.605 1371.83 256.615ZM1420.41 258.586 1429.74 260.308 1428.49 267.068 1419.16 265.347ZM1427.1 249.334 1451.65 267.846 1422.11 276.378Z" /><path + id="path64" + transform="matrix(-1 0 0 1 3760.78 1238.5)" + fill="#003967" + d="M0.422723-3.41141 27.714-0.0296238 26.8685 6.79319-0.422723 3.41141ZM48.1824 2.50672 75.4737 5.8885 74.6283 12.7113 47.337 9.32953ZM95.9422 8.42484 123.233 11.8066 122.388 18.6294 95.0967 15.2477ZM143.702 14.343 170.993 17.7248 170.148 24.5476 142.856 21.1658ZM191.462 20.2611 218.753 23.6429 217.907 30.4657 190.616 27.0839ZM239.221 26.1792 266.513 29.561 265.667 36.3838 238.376 33.002ZM286.981 32.0973 314.272 35.4791 313.427 42.3019 286.136 38.9202ZM334.741 38.0155 362.032 41.3972 361.187 48.2201 333.895 44.8383ZM382.501 43.9336 409.792 47.3154 408.946 54.1382 381.655 50.7564ZM430.26 49.8517 457.552 53.2335 456.706 60.0563 429.415 56.6745ZM478.02 55.7698 505.311 59.1516 504.466 65.9744 477.175 62.5927ZM525.78 61.688 553.071 65.0697 552.226 71.8926 524.934 68.5108ZM573.539 67.6061 600.831 70.9879 599.985 77.8107 572.694 74.4289ZM621.299 73.5242 648.59 76.906 647.745 83.7288 620.454 80.347ZM669.059 79.4423 696.35 82.8241 695.505 89.6469 668.213 86.2652ZM716.819 85.3605 744.11 88.7423 743.264 95.5651 715.973 92.1833ZM764.578 91.2786 791.87 94.6604 791.024 101.483 763.733 98.1014ZM812.338 97.1967 839.629 100.579 838.784 107.401 811.493 104.02ZM860.098 103.115 887.389 106.497 886.544 113.319 859.253 109.938ZM907.858 109.033 935.149 112.415 934.304 119.238 907.012 115.856ZM955.617 114.951 982.909 118.333 982.063 125.156 954.772 121.774ZM1003.38 120.869 1030.67 124.251 1029.82 131.074 1002.53 127.692ZM1051.14 126.787 1078.43 130.169 1077.58 136.992 1050.29 133.61ZM1098.9 132.705 1126.19 136.087 1125.34 142.91 1098.05 139.528ZM1146.66 138.624 1173.95 142.005 1173.1 148.828 1145.81 145.446ZM1194.42 144.542 1221.71 147.924 1220.86 154.746 1193.57 151.365ZM1242.18 150.46 1269.47 153.842 1268.62 160.664 1241.33 157.283ZM1289.94 156.378 1317.23 159.76 1316.38 166.583 1289.09 163.201ZM1337.7 162.296 1364.99 165.678 1364.14 172.501 1336.85 169.119ZM1385.45 168.214 1412.75 171.596 1411.9 178.419 1384.61 175.037ZM1433.21 174.132 1460.51 177.514 1459.66 184.337 1432.37 180.955ZM1480.97 180.051 1508.27 183.432 1507.42 190.255 1480.13 186.873ZM1528.73 185.969 1556.03 189.35 1555.18 196.173 1527.89 192.791ZM1576.49 191.887 1603.78 195.269 1602.94 202.091 1575.65 198.71ZM1624.25 197.805 1651.54 201.187 1650.7 208.009 1623.41 204.628ZM1672.01 203.723 1699.3 207.105 1698.46 213.928 1671.17 210.546ZM1719.77 209.641 1747.06 213.023 1746.22 219.846 1718.93 216.464ZM1767.53 215.559 1794.82 218.941 1793.98 225.764 1766.69 222.382ZM1815.29 221.477 1842.58 224.859 1841.74 231.682 1814.45 228.3ZM1863.05 227.395 1890.34 230.777 1889.5 237.6 1862.21 234.218ZM1910.81 233.314 1938.1 236.695 1937.26 243.518 1909.97 240.136ZM1958.57 239.232 1985.86 242.613 1985.02 249.436 1957.73 246.054ZM2006.33 245.15 2033.62 248.532 2032.78 255.354 2005.49 251.973ZM2054.09 251.068 2081.38 254.45 2080.54 261.273 2053.25 257.891ZM2101.85 256.986 2129.14 260.368 2128.3 267.191 2101 263.809ZM2149.61 262.904 2176.9 266.286 2176.06 273.109 2148.76 269.727ZM2197.37 268.822 2224.66 272.204 2223.82 279.027 2196.52 275.645ZM2245.13 274.74 2272.42 278.122 2271.58 284.945 2244.28 281.563ZM2292.89 280.659 2320.18 284.04 2319.34 290.863 2292.04 287.481ZM2340.65 286.577 2367.94 289.958 2367.09 296.781 2339.8 293.399ZM2388.41 292.495 2415.7 295.877 2414.85 302.699 2387.56 299.318ZM2436.17 298.413 2463.46 301.795 2462.61 308.618 2435.32 305.236ZM2483.93 304.331 2511.22 307.713 2510.37 314.536 2483.08 311.154ZM2531.69 310.249 2531.97 310.284 2531.12 317.107 2530.84 317.072ZM2528.68 299.485 2554.28 316.512 2525.29 326.776Z" /><text + id="text66" + transform="matrix(0.974105 0.226098 -0.226098 0.974105 1319.22 820)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 2</text> +<text + id="text68" + transform="matrix(1 0 0 1 1572.39 1639)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">First Feature</text> +<text + id="text70" + transform="matrix(1 0 0 1 1505.08 1716)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">For Interface </text> +<text + id="text72" + transform="matrix(1 0 0 1 1884.92 1716)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">–</text> +<text + id="text74" + transform="matrix(1 0 0 1 1938.2 1716)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">0,1</text> +<path + id="path76" + fill="#63666A" + d="M1764.3 1561.88 1643.18 1304.21 1647.32 1302.26 1768.45 1559.93ZM1634.76 1313.24 1635.5 1282.5 1659.64 1301.54Z" /><path + id="path78" + fill="#63666A" + d="M1967.54 1129.06 1995.04 1129.38 1994.96 1136.26 1967.46 1135.94ZM2015.66 1129.62 2043.16 1129.94 2043.08 1136.82 2015.58 1136.5ZM2063.78 1130.18 2091.28 1130.5 2091.2 1137.38 2063.7 1137.06ZM2111.91 1130.74 2139.4 1131.06 2139.32 1137.94 2111.83 1137.62ZM2160.03 1131.3 2187.53 1131.62 2187.45 1138.49 2159.95 1138.17ZM2208.15 1131.86 2235.65 1132.18 2235.57 1139.05 2208.07 1138.73ZM2256.27 1132.42 2283.77 1132.74 2283.69 1139.61 2256.19 1139.29ZM2299.05 1122.6 2326.39 1136.67 2298.73 1150.1Z" /><text + id="text80" + transform="matrix(1 0 0 1 2008.67 1117)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 1 </text> +<text + id="text82" + transform="matrix(1 0 0 1 2307.19 1624)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Next Feature to Feature</text> +<text + id="text84" + transform="matrix(1 0 0 1 3021.04 1624)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text86" + transform="matrix(1 0 0 1 3042.24 1624)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">1</text> +<text + id="text88" + transform="matrix(1 0 0 1 2493.66 1701)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">for Interface</text> +<text + id="text90" + transform="matrix(1 0 0 1 2834.55 1701)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text92" + transform="matrix(1 0 0 1 2855.74 1701)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">0</text> +<path + id="path94" + fill="#63666A" + d="M2690.94 1546.47 2659.26 1309.52 2663.81 1308.91 2695.48 1545.86ZM2648.51 1315.58 2658.5 1286.5 2675.77 1311.94Z" /><text + id="text96" + transform="matrix(1 0 0 1 3080.93 1096)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0</text> +<path + id="path98" + transform="matrix(1 0 0 -1 2990.5 1125.64)" + fill="#63666A" + d="M0.219926-3.43046 27.6636-1.67105 27.2237 5.18986-0.219926 3.43046ZM48.2463-0.351496 75.69 1.40791 75.2501 8.26883 47.8065 6.50942ZM96.2728 2.72747 123.716 4.48687 123.277 11.3478 95.8329 9.58838ZM144.299 5.80643 171.743 7.56583 171.303 14.4267 143.859 12.6673ZM192.326 8.88539 219.769 10.6448 219.329 17.5057 191.886 15.7463ZM240.352 11.9644 267.796 13.7238 267.356 20.5847 239.912 18.8253ZM288.378 15.0433 315.822 16.8027 315.382 23.6636 287.939 21.9042ZM336.405 18.1223 363.848 19.8817 363.409 26.7426 335.965 24.9832ZM384.431 21.2012 411.875 22.9606 411.435 29.8216 383.991 28.0622ZM412.357 12.6579 438.921 28.1391 410.598 40.1016Z" /><text + id="text100" + transform="matrix(1 0 0 1 3280.81 1634)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Next Feature to Feature</text> +<text + id="text102" + transform="matrix(1 0 0 1 3994.67 1634)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text104" + transform="matrix(1 0 0 1 4015.87 1634)" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">1</text> +<text + id="text106" + transform="matrix(1 0 0 1 3480.19 1711)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">for Interface</text> +<text + id="text108" + transform="matrix(1 0 0 1 3821.08 1711)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text110" + transform="matrix(1 0 0 1 3839.41 1711)" + font-size="55" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">1</text> +<path + id="path112" + transform="matrix(1 0 0 -1 3666.5 1556.2)" + fill="#63666A" + d="M2.19695-0.652044 89.9684 295.078 85.5745 296.382-2.19695 0.652044ZM99.6489 287.424 94.2918 317.699 73.2856 295.248Z" /><path + id="path114" + fill="#63666A" + d="M1867.07 1025.81 1867.42 1018.62 1868.49 1011.28 1870.26 1003.97 1872.27 997.989 1878.79 1000.18 1876.82 1006.02 1876.91 1005.73 1875.21 1012.74 1875.27 1012.43 1874.25 1019.45 1874.28 1019.12 1873.93 1026.15ZM1881.7 978.819 1883.97 975.115 1888.98 968.018 1894.59 960.969 1898.61 956.437 1903.75 960.998 1899.8 965.462 1899.91 965.323 1894.42 972.222 1894.54 972.063 1889.65 978.99 1889.77 978.808 1887.55 982.418ZM1913.37 941.594 1914.93 940.127 1922.84 933.284 1931.3 926.497 1934.85 923.839 1938.97 929.343 1935.46 931.966 1935.55 931.895 1927.19 938.606 1927.29 938.525 1919.48 945.281 1919.58 945.188 1918.07 946.609ZM1951.82 911.768 1959.83 906.488 1970.37 899.942 1975.26 897.066 1978.75 902.994 1973.89 905.849 1973.96 905.805 1963.5 912.303 1963.58 912.254 1955.6 917.508ZM1993.34 886.825 2004.89 880.719 2017.33 874.457 2017.91 874.18 2020.88 880.379 2020.34 880.642 2020.39 880.613 2008.01 886.844 2008.07 886.812 1996.55 892.902ZM2036.62 865.351 2043.57 862.173 2057.34 856.155 2061.9 854.247 2064.55 860.591 2060.01 862.488 2060.06 862.466 2046.35 868.46 2046.4 868.437 2039.48 871.603ZM2081.04 846.419 2086.12 844.388 2101.11 838.643 2106.81 836.551 2109.18 843.005 2103.51 845.089 2103.55 845.072 2088.6 850.799 2088.65 850.78 2083.59 852.803ZM2126.26 829.558 2132.27 827.446 2152.4 820.792 2154.55 827.32 2134.45 833.964 2134.52 833.942 2128.55 836.044ZM2172.11 814.467 2198.42 806.467 2200.42 813.044 2174.11 821.045ZM2218.36 800.882 2234.13 796.472 2244.98 793.688 2246.69 800.347 2235.88 803.122 2235.95 803.103 2220.21 807.503ZM2264.95 788.558 2270.62 787.103 2291.76 782.131 2293.34 788.824 2272.23 793.788 2272.3 793.771 2266.66 795.217ZM2311.92 777.466 2338.81 771.695 2340.25 778.417 2313.36 784.188ZM2359.09 767.584 2386.08 762.324 2387.39 769.072 2360.4 774.332ZM2406.45 758.735 2426.94 755.133 2433.62 754.084 2434.69 760.876 2428.04 761.92 2428.1 761.91 2407.64 765.506ZM2453.99 750.883 2468.18 748.653 2481.26 746.839 2482.2 753.649 2469.16 755.459 2469.22 755.45 2455.06 757.674ZM2501.69 744.003 2510.12 742.833 2529.03 740.551 2529.85 747.376 2510.97 749.655 2511.03 749.647 2502.63 750.813ZM2549.51 738.08 2552.66 737.699 2576.92 735.21 2577.62 742.049 2553.39 744.535 2553.45 744.529 2550.33 744.905ZM2597.5 733.13 2624.9 730.813 2625.48 737.664 2598.08 739.98ZM2645.52 729.185 2672.96 727.365 2673.42 734.224 2645.98 736.045ZM2693.62 726.19 2721.08 724.875 2721.41 731.742 2693.94 733.057ZM2741.76 724.16 2769.25 723.361 2769.45 730.233 2741.96 731.032ZM2789.94 723.116 2815.78 722.863 2817.49 722.875 2817.45 729.749 2815.76 729.738 2815.82 729.738 2790.01 729.991ZM2838.12 723.017 2860.13 723.169 2865.66 723.283 2865.52 730.156 2860.01 730.043 2860.06 730.044 2838.07 729.892ZM2886.28 723.705 2904.38 724.076 2913.82 724.394 2913.59 731.265 2904.17 730.948 2904.22 730.949 2886.14 730.579ZM2934.43 725.09 2948.46 725.563 2961.96 726.194 2961.64 733.061 2948.16 732.432 2948.21 732.433 2934.2 731.961ZM2982.56 727.157 2992.28 727.611 3010.06 728.671 3009.65 735.533 2991.9 734.476 2991.94 734.478 2982.24 734.024ZM3030.65 729.897 3035.76 730.202 3058.13 731.82 3057.63 738.677 3035.28 737.06 3035.33 737.063 3030.24 736.76ZM3078.7 733.307 3078.8 733.315 3106.14 735.64 3105.56 742.49 3078.24 740.167 3078.28 740.17 3078.2 740.164ZM3126.73 737.46 3154.1 740.137 3153.43 746.98 3126.06 744.302ZM3174.65 742.296 3201.99 745.322 3201.23 752.155 3173.9 749.129ZM3222.5 747.831 3244.91 750.607 3249.83 751.283 3248.9 758.094 3243.99 757.421 3244.04 757.427 3221.66 754.654ZM3270.26 754.089 3284.49 756.043 3297.53 758.016 3296.5 764.813 3283.48 762.844 3283.53 762.851 3269.33 760.9ZM3317.92 761.1 3323.12 761.886 3345.11 765.531 3343.99 772.314 3322.02 768.673 3322.07 768.68 3316.89 767.898ZM3365.5 768.983 3392.56 773.88 3391.33 780.645 3364.27 775.748ZM3412.86 777.806 3432.45 781.666 3439.87 783.257 3438.43 789.979 3431.04 788.394 3431.09 788.406 3411.53 784.551ZM3460.04 787.579 3466.42 788.945 3486.91 793.718 3485.35 800.414 3464.89 795.648 3464.95 795.661 3458.6 794.301ZM3507.01 798.523 3514.75 800.443 3530.13 804.42 3533.71 805.389 3531.92 812.026 3528.35 811.062 3528.39 811.071 3513.05 807.103 3513.08 807.111 3505.36 805.196ZM3553.64 810.869 3559.7 812.577 3573.87 816.753 3580.09 818.669 3578.07 825.239 3571.87 823.33 3571.91 823.342 3557.77 819.178 3557.81 819.189 3551.77 817.487ZM3599.8 824.913 3600.96 825.287 3613.84 829.641 3625.89 833.912 3623.59 840.391 3611.57 836.129 3611.62 836.146 3598.78 831.808 3598.82 831.823 3597.69 831.457ZM3645.28 841.269 3649.75 843.025 3660.77 847.587 3670.76 851.959 3668 858.257 3658.04 853.898 3658.11 853.926 3647.15 849.389 3647.21 849.413 3642.77 847.669ZM3689.52 860.9 3690.83 861.548 3699.81 866.289 3708.27 871.071 3713.79 874.438 3710.21 880.308 3704.74 876.97 3704.83 877.027 3696.47 872.297 3696.56 872.346 3687.66 867.65 3687.74 867.692 3686.47 867.064ZM3731.16 886.32 3736.55 890.61 3742.19 895.601 3747.24 900.64 3751.56 905.593 3746.37 910.111 3742.13 905.247 3742.3 905.421 3737.41 900.539 3737.56 900.68 3732.06 895.818 3732.2 895.935 3726.88 891.701ZM3764.7 906.97 3761.07 937.5 3738.83 916.273Z" /><text + id="text116" + transform="matrix(0.998741 -0.050158 0.050158 0.998741 2571.92 793)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0</text> +<text + id="text118" + transform="matrix(1 0 0 1 1933.46 420)" + text-decoration="underline" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Interface</text> +<text + id="text120" + transform="matrix(1 0 0 1 2199.86 420)" + text-decoration="underline" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text122" + transform="matrix(1 0 0 1 2221.06 420)" + text-decoration="underline" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">0</text> +<text + id="text124" + transform="matrix(1 0 0 1 1842.64 497)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Enabled Features</text> +<text + id="text126" + transform="matrix(1 0 0 1 2828.07 412)" + text-decoration="underline" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Interface</text> +<text + id="text128" + transform="matrix(1 0 0 1 3094.47 412)" + text-decoration="underline" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text130" + transform="matrix(1 0 0 1 3115.67 412)" + text-decoration="underline" + font-size="64" + font-weight="700" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">1</text> +<text + id="text132" + transform="matrix(1 0 0 1 2737.25 489)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Enabled Features</text> +<path + id="path134" + transform="matrix(-1 0 0 1 1993.37 532.5)" + fill="#63666A" + d="M0.513174-0.254726 224.189 450.366 223.163 450.875-0.513174 0.254726ZM233.954 440.402 233.865 471.148 209.322 452.629Z" /><path + id="path136" + fill="#63666A" + d="M2095.86 523.058 2640.92 971.389 2640.19 972.274 2095.14 523.943ZM2645.75 958.301 2658.26 986.389 2628.28 979.539Z" /><path + id="path138" + transform="matrix(-1 0 0 1 2834.66 536.5)" + fill="#63666A" + d="M0.307156-0.48362 857.122 543.695 856.508 544.663-0.307156 0.48362ZM860.318 530.115 876.16 556.465 845.574 553.329Z" /><path + id="path140" + fill="#63666A" + d="M2990.87 529.062 3509.46 966.033 3508.73 966.91 2990.13 529.938ZM3514.45 953.003 3526.62 981.238 3496.73 974.033Z" /><path + id="path142" + fill="#63666A" + d="M2381.7 431.962 3739.81 929.039 3739.42 930.115 2381.3 433.038ZM3740.03 915.088 3761.13 937.452 3730.58 940.913Z" /><text + id="text144" + transform="matrix(0.838284 -0.545233 0.545233 0.838284 1136.13 1461)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 2 </text> +<text + id="text146" + transform="matrix(0.98675 -0.162247 0.162247 0.98675 2069.25 1383)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0 </text> +<text + id="text148" + transform="matrix(0.994324 -0.106397 0.106397 0.994324 2994.86 1316)" + font-size="64" + font-weight="400" + font-style="italic" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0 </text> +</g></svg> diff --git a/doc/guides/prog_guide/img/feature_arc-3.svg b/doc/guides/prog_guide/img/feature_arc-3.svg new file mode 100644 index 0000000000..382f259e42 --- /dev/null +++ b/doc/guides/prog_guide/img/feature_arc-3.svg @@ -0,0 +1,318 @@ +<?xml version="1.0" encoding="UTF-8" standalone="no"?> +<!-- Created with Inkscape (http://www.inkscape.org/) --> +<!-- SPDX-License-Identifier: BSD-3-Clause --> +<!-- Copyright(C) 2024 Marvell International Ltd. --> +<svg + xmlns:dc="http://purl.org/dc/elements/1.1/" + xmlns:cc="http://creativecommons.org/ns#" + xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#" + xmlns:svg="http://www.w3.org/2000/svg" + xmlns="http://www.w3.org/2000/svg" + id="svg100" + version="1.1" + overflow="hidden" + xml:space="preserve" + height="2075" + width="3423"><metadata + id="metadata106"><rdf:RDF><cc:Work + rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type + rdf:resource="http://purl.org/dc/dcmitype/StillImage" /><dc:title></dc:title></cc:Work></rdf:RDF></metadata><defs + id="defs104" /><g + id="g98" + transform="translate(-590 -205)"><path + id="path2" + fill-rule="evenodd" + fill="#00B5E2" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M593.5 358.5C593.5 275.657 735.426 208.5 910.5 208.5 1085.57 208.5 1227.5 275.657 1227.5 358.5 1227.5 441.343 1085.57 508.5 910.5 508.5 735.426 508.5 593.5 441.343 593.5 358.5Z" /><text + id="text4" + transform="matrix(1 0 0 1 770.408 388)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Node</text> +<text + id="text6" + transform="matrix(1 0 0 1 967.492 388)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text8" + transform="matrix(1 0 0 1 994.992 388)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">A</text> +<path + id="path10" + fill-rule="evenodd" + fill="#00B5E2" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M596.5 1475.5C596.5 1392.66 738.65 1325.5 914 1325.5 1089.35 1325.5 1231.5 1392.66 1231.5 1475.5 1231.5 1558.34 1089.35 1625.5 914 1625.5 738.65 1625.5 596.5 1558.34 596.5 1475.5Z" /><text + id="text12" + transform="matrix(1 0 0 1 773.708 1505)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Node</text> +<text + id="text14" + transform="matrix(1 0 0 1 970.792 1505)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text16" + transform="matrix(1 0 0 1 998.292 1505)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">B</text> +<path + id="path18" + fill-rule="evenodd" + fill="#00B050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M1297.5 947C1297.5 863.881 1435.62 796.5 1606 796.5 1776.38 796.5 1914.5 863.881 1914.5 947 1914.5 1030.12 1776.38 1097.5 1606 1097.5 1435.62 1097.5 1297.5 1030.12 1297.5 947Z" /><text + id="text20" + transform="matrix(1 0 0 1 1425.81 974)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Feature </text> +<text + id="text22" + transform="matrix(1 0 0 1 1699.09 974)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text24" + transform="matrix(1 0 0 1 1744.35 974)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">1</text> +<path + id="path26" + fill="#63666A" + d="M913.938 508.486 917.145 1302.23 910.27 1302.25 907.063 508.514ZM927.439 1297.6 913.8 1325.16 899.939 1297.71Z" /><path + id="path28" + fill="#003967" + d="M911.378 506.383 928.313 513.406 926.557 517.639 909.622 510.617ZM941.014 518.672 957.949 525.695 956.194 529.929 939.259 522.906ZM970.651 530.962 987.586 537.984 985.83 542.218 968.895 535.195ZM1000.29 543.251 1017.22 550.274 1015.47 554.507 998.531 547.485ZM1029.92 555.54 1046.86 562.563 1045.1 566.797 1028.17 559.774ZM1059.56 567.83 1076.49 574.852 1074.74 579.086 1057.8 572.064ZM1089.2 580.119 1106.13 587.142 1104.38 591.375 1087.44 584.353ZM1118.83 592.409 1135.77 599.431 1134.01 603.665 1117.08 596.642ZM1148.47 604.698 1165.4 611.72 1163.65 615.954 1146.71 608.932ZM1178.1 616.987 1195.04 624.01 1193.28 628.244 1176.35 621.221ZM1207.74 629.277 1224.68 636.299 1222.92 640.533 1205.99 633.51ZM1237.38 641.566 1254.31 648.589 1252.56 652.822 1235.62 645.8ZM1267.01 653.855 1283.95 660.878 1282.19 665.112 1265.26 658.089ZM1296.65 666.145 1313.59 673.167 1311.83 677.401 1294.89 670.379ZM1326.29 678.434 1343.22 685.457 1341.47 689.69 1324.53 682.668ZM1355.92 690.724 1372.86 697.746 1371.1 701.98 1354.17 694.957ZM1385.56 703.013 1402.49 710.035 1400.74 714.269 1383.8 707.247ZM1415.2 715.302 1432.13 722.325 1430.38 726.558 1413.44 719.536ZM1444.83 727.592 1461.77 734.614 1460.01 738.848 1443.08 731.825ZM1474.47 739.881 1491.4 746.903 1489.65 751.137 1472.71 744.115ZM1504.1 752.17 1521.04 759.193 1519.28 763.427 1502.35 756.404ZM1533.74 764.46 1550.68 771.482 1548.92 775.716 1531.99 768.693ZM1563.38 776.749 1580.31 783.772 1578.56 788.005 1561.62 780.983ZM1585.58 773.552 1605.72 796.786 1575.05 798.954Z" /><path + id="path30" + transform="matrix(-1 0 0 1 1606.42 1097.5)" + fill="#003967" + d="M0.71731-2.17651 18.1294 3.56197 16.6948 7.91499-0.71731 2.17651ZM31.1885 7.86584 48.6006 13.6043 47.1659 17.9573 29.7538 12.2189ZM61.6596 17.9082 79.0717 23.6467 77.6371 27.9997 60.225 22.2612ZM92.1308 27.9505 109.543 33.689 108.108 38.042 90.6962 32.3035ZM122.602 37.9929 140.014 43.7314 138.579 48.0844 121.167 42.3459ZM153.073 48.0352 170.485 53.7737 169.051 58.1267 151.638 52.3882ZM183.544 58.0776 200.956 63.816 199.522 68.1691 182.11 62.4306ZM214.015 68.1199 231.428 73.8584 229.993 78.2114 212.581 72.4729ZM244.487 78.1622 261.899 83.9007 260.464 88.2538 243.052 82.5153ZM274.958 88.2046 292.37 93.9431 290.935 98.2961 273.523 92.5576ZM305.429 98.247 322.841 103.985 321.406 108.338 303.994 102.6ZM335.9 108.289 353.312 114.028 351.878 118.381 334.465 112.642ZM366.371 118.332 383.783 124.07 382.349 128.423 364.937 122.685ZM396.842 128.374 414.255 134.112 412.82 138.465 395.408 132.727ZM427.314 138.416 444.726 144.155 443.291 148.508 425.879 142.769ZM457.785 148.459 475.197 154.197 473.762 158.55 456.35 152.812ZM488.256 158.501 505.668 164.24 504.233 168.593 486.821 162.854ZM518.727 168.543 536.139 174.282 534.705 178.635 517.292 172.896ZM549.198 178.586 566.61 184.324 565.176 188.677 547.764 182.939ZM579.669 188.628 597.081 194.367 595.647 198.72 578.235 192.981ZM610.141 198.67 627.553 204.409 626.118 208.762 608.706 203.023ZM640.612 208.713 658.024 214.451 656.589 218.804 639.177 213.066ZM670.101 206.367 691.916 228.034 661.494 232.485Z" /><path + id="path32" + fill-rule="evenodd" + fill="#00B050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M3375.5 902.5C3375.5 819.657 3517.43 752.5 3692.5 752.5 3867.57 752.5 4009.5 819.657 4009.5 902.5 4009.5 985.343 3867.57 1052.5 3692.5 1052.5 3517.43 1052.5 3375.5 985.343 3375.5 902.5Z" /><text + id="text34" + transform="matrix(1 0 0 1 3513.57 933)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Feature</text> +<text + id="text36" + transform="matrix(1 0 0 1 3797.73 933)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text38" + transform="matrix(1 0 0 1 3825.23 933)" + font-size="83" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">3</text> +<path + id="path40" + fill="#003967" + d="M910.7 506.217 928.964 507.817 928.563 512.383 910.3 510.783ZM942.661 509.018 960.924 510.618 960.524 515.184 942.261 513.583ZM974.622 511.818 992.885 513.419 992.485 517.985 974.222 516.384ZM1006.58 514.619 1024.85 516.219 1024.45 520.785 1006.18 519.185ZM1038.54 517.42 1056.81 519.02 1056.41 523.586 1038.14 521.986ZM1070.5 520.22 1088.77 521.821 1088.37 526.387 1070.1 524.786ZM1102.47 523.021 1120.73 524.621 1120.33 529.187 1102.07 527.587ZM1134.43 525.822 1152.69 527.422 1152.29 531.988 1134.03 530.388ZM1166.39 528.622 1184.65 530.223 1184.25 534.789 1165.99 533.188ZM1198.35 531.423 1216.61 533.023 1216.21 537.589 1197.95 535.989ZM1230.31 534.224 1248.57 535.824 1248.17 540.39 1229.91 538.79ZM1262.27 537.024 1280.53 538.625 1280.13 543.191 1261.87 541.59ZM1294.23 539.825 1312.49 541.425 1312.09 545.991 1293.83 544.391ZM1326.19 542.626 1344.45 544.226 1344.05 548.792 1325.79 547.192ZM1358.15 545.426 1376.42 547.027 1376.02 551.593 1357.75 549.992ZM1390.11 548.227 1408.38 549.827 1407.98 554.393 1389.71 552.793ZM1422.07 551.028 1440.34 552.628 1439.94 557.194 1421.67 555.594ZM1454.03 553.828 1472.3 555.429 1471.9 559.995 1453.63 558.394ZM1486 556.629 1504.26 558.229 1503.86 562.795 1485.6 561.195ZM1517.96 559.43 1536.22 561.03 1535.82 565.596 1517.56 563.996ZM1549.92 562.23 1568.18 563.831 1567.78 568.397 1549.52 566.796ZM1581.88 565.031 1600.14 566.631 1599.74 571.197 1581.48 569.597ZM1613.84 567.832 1632.1 569.432 1631.7 573.998 1613.44 572.398ZM1645.8 570.632 1664.06 572.233 1663.66 576.799 1645.4 575.198ZM1677.76 573.433 1696.02 575.034 1695.62 579.599 1677.36 577.999ZM1709.72 576.234 1727.98 577.834 1727.58 582.4 1709.32 580.8ZM1741.68 579.034 1759.95 580.635 1759.55 585.201 1741.28 583.6ZM1773.64 581.835 1791.91 583.435 1791.51 588.001 1773.24 586.401ZM1805.6 584.636 1823.87 586.236 1823.47 590.802 1805.2 589.202ZM1837.56 587.436 1855.83 589.037 1855.43 593.603 1837.16 592.002ZM1869.53 590.237 1887.79 591.838 1887.39 596.403 1869.13 594.803ZM1901.49 593.038 1919.75 594.638 1919.35 599.204 1901.09 597.604ZM1933.45 595.839 1951.71 597.439 1951.31 602.005 1933.05 600.404ZM1965.41 598.639 1983.67 600.24 1983.27 604.805 1965.01 603.205ZM1997.37 601.44 2015.63 603.04 2015.23 607.606 1996.97 606.006ZM2029.33 604.24 2047.59 605.841 2047.19 610.407 2028.93 608.806ZM2061.29 607.041 2079.55 608.642 2079.15 613.207 2060.89 611.607ZM2093.25 609.842 2111.51 611.442 2111.11 616.008 2092.85 614.408ZM2125.21 612.643 2143.48 614.243 2143.08 618.809 2124.81 617.208ZM2157.17 615.443 2175.44 617.044 2175.04 621.609 2156.77 620.009ZM2189.13 618.244 2207.4 619.844 2207 624.41 2188.73 622.81ZM2221.09 621.045 2239.36 622.645 2238.96 627.211 2220.69 625.61ZM2253.06 623.845 2271.32 625.446 2270.92 630.011 2252.66 628.411ZM2285.02 626.646 2303.28 628.246 2302.88 632.812 2284.62 631.212ZM2316.98 629.447 2335.24 631.047 2334.84 635.613 2316.58 634.012ZM2348.94 632.247 2367.2 633.848 2366.8 638.413 2348.54 636.813ZM2380.9 635.048 2399.16 636.648 2398.76 641.214 2380.5 639.614ZM2412.86 637.849 2431.12 639.449 2430.72 644.015 2412.46 642.414ZM2444.82 640.649 2463.08 642.25 2462.68 646.815 2444.42 645.215ZM2476.78 643.45 2495.04 645.05 2494.64 649.616 2476.38 648.016ZM2508.74 646.251 2527.01 647.851 2526.6 652.417 2508.34 650.816ZM2540.7 649.051 2558.97 650.652 2558.57 655.217 2540.3 653.617ZM2572.66 651.852 2590.93 653.452 2590.53 658.018 2572.26 656.418ZM2604.62 654.653 2622.89 656.253 2622.49 660.819 2604.22 659.218ZM2636.58 657.453 2654.85 659.054 2654.45 663.62 2636.18 662.019ZM2668.55 660.254 2686.81 661.854 2686.41 666.42 2668.15 664.82ZM2700.51 663.055 2718.77 664.655 2718.37 669.221 2700.11 667.62ZM2732.47 665.855 2750.73 667.456 2750.33 672.021 2732.07 670.421ZM2764.43 668.656 2782.69 670.256 2782.29 674.822 2764.03 673.222ZM2796.39 671.457 2814.65 673.057 2814.25 677.623 2795.99 676.022ZM2828.35 674.257 2846.61 675.858 2846.21 680.424 2827.95 678.823ZM2860.31 677.058 2878.57 678.658 2878.17 683.224 2859.91 681.624ZM2892.27 679.859 2910.53 681.459 2910.13 686.025 2891.87 684.424ZM2924.23 682.659 2942.5 684.26 2942.1 688.826 2923.83 687.225ZM2956.19 685.46 2974.46 687.06 2974.06 691.626 2955.79 690.026ZM2988.15 688.261 3006.42 689.861 3006.02 694.427 2987.75 692.826ZM3020.11 691.061 3038.38 692.662 3037.98 697.228 3019.71 695.627ZM3052.08 693.862 3070.34 695.462 3069.94 700.028 3051.68 698.428ZM3084.04 696.663 3102.3 698.263 3101.9 702.829 3083.64 701.229ZM3116 699.463 3134.26 701.064 3133.86 705.63 3115.6 704.029ZM3147.96 702.264 3166.22 703.864 3165.82 708.43 3147.56 706.83ZM3179.92 705.065 3198.18 706.665 3197.78 711.231 3179.52 709.63ZM3211.88 707.865 3230.14 709.466 3229.74 714.031 3211.48 712.431ZM3243.84 710.666 3262.1 712.266 3261.7 716.832 3243.44 715.232ZM3275.8 713.467 3294.07 715.067 3293.67 719.633 3275.4 718.032ZM3307.76 716.267 3326.03 717.868 3325.63 722.433 3307.36 720.833ZM3339.72 719.068 3357.99 720.668 3357.59 725.234 3339.32 723.634ZM3371.69 721.869 3389.95 723.469 3389.55 728.035 3371.29 726.434ZM3403.65 724.669 3421.91 726.27 3421.51 730.836 3403.25 729.235ZM3435.61 727.47 3453.87 729.07 3453.47 733.636 3435.21 732.036ZM3467.57 730.271 3485.83 731.871 3485.43 736.437 3467.17 734.837ZM3499.53 733.071 3517.79 734.672 3517.39 739.238 3499.13 737.637ZM3531.49 735.872 3549.75 737.472 3549.35 742.038 3531.09 740.438ZM3563.45 738.673 3581.71 740.273 3581.31 744.839 3563.05 743.239ZM3595.41 741.473 3613.68 743.074 3613.28 747.64 3595.01 746.039ZM3627.37 744.274 3645.64 745.875 3645.24 750.44 3626.97 748.84ZM3659.33 747.075 3669.97 748.007 3669.57 752.573 3658.93 751.641ZM3666.42 736.194 3692.62 752.292 3664.02 763.589Z" /><path + id="path42" + fill-rule="evenodd" + fill="#92D050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M2273.5 951.5C2273.5 868.657 2409.16 801.5 2576.5 801.5 2743.84 801.5 2879.5 868.657 2879.5 951.5 2879.5 1034.34 2743.84 1101.5 2576.5 1101.5 2409.16 1101.5 2273.5 1034.34 2273.5 951.5Z" /><text + id="text44" + transform="matrix(1 0 0 1 2396.44 978)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Feature </text> +<text + id="text46" + transform="matrix(1 0 0 1 2669.72 978)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text48" + transform="matrix(1 0 0 1 2714.98 978)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">2</text> +<path + id="path50" + fill="#003967" + d="M910.896 506.243 928.954 509.413 928.161 513.927 910.104 510.757ZM942.496 511.79 960.554 514.961 959.761 519.475 941.704 516.305ZM974.096 517.338 992.154 520.508 991.361 525.023 973.304 521.853ZM1005.7 522.886 1023.75 526.056 1022.96 530.57 1004.9 527.4ZM1037.3 528.434 1055.35 531.604 1054.56 536.118 1036.5 532.948ZM1068.9 533.981 1086.95 537.152 1086.16 541.666 1068.1 538.496ZM1100.5 539.529 1118.55 542.699 1117.76 547.214 1099.7 544.043ZM1132.1 545.077 1150.15 548.247 1149.36 552.761 1131.3 549.591ZM1163.7 550.625 1181.75 553.795 1180.96 558.309 1162.9 555.139ZM1195.3 556.172 1213.35 559.343 1212.56 563.857 1194.5 560.687ZM1226.9 561.72 1244.95 564.89 1244.16 569.405 1226.1 566.234ZM1258.5 567.268 1276.55 570.438 1275.76 574.952 1257.7 571.782ZM1290.1 572.816 1308.15 575.986 1307.36 580.5 1289.3 577.33ZM1321.7 578.363 1339.75 581.534 1338.96 586.048 1320.9 582.878ZM1353.3 583.911 1371.35 587.081 1370.56 591.596 1352.5 588.425ZM1384.9 589.459 1402.95 592.629 1402.16 597.143 1384.1 593.973ZM1416.5 595.007 1434.55 598.177 1433.76 602.691 1415.7 599.521ZM1448.1 600.554 1466.15 603.724 1465.36 608.239 1447.3 605.069ZM1479.7 606.102 1497.75 609.272 1496.96 613.786 1478.9 610.616ZM1511.3 611.65 1529.35 614.82 1528.56 619.334 1510.5 616.164ZM1542.9 617.198 1560.95 620.368 1560.16 624.882 1542.1 621.712ZM1574.5 622.745 1592.55 625.915 1591.76 630.43 1573.7 627.26ZM1606.1 628.293 1624.15 631.463 1623.36 635.977 1605.3 632.807ZM1637.7 633.841 1655.75 637.011 1654.96 641.525 1636.91 638.355ZM1669.3 639.388 1687.35 642.559 1686.56 647.073 1668.5 643.903ZM1700.9 644.936 1718.95 648.106 1718.16 652.621 1700.1 649.451ZM1732.5 650.484 1750.55 653.654 1749.76 658.168 1731.7 654.998ZM1764.1 656.032 1782.15 659.202 1781.36 663.716 1763.3 660.546ZM1795.7 661.579 1813.75 664.75 1812.96 669.264 1794.91 666.094ZM1827.3 667.127 1845.35 670.297 1844.56 674.812 1826.51 671.641ZM1858.9 672.675 1876.95 675.845 1876.16 680.359 1858.1 677.189ZM1890.5 678.223 1908.55 681.393 1907.76 685.907 1889.71 682.737ZM1922.1 683.77 1940.15 686.941 1939.36 691.455 1921.3 688.285ZM1953.7 689.318 1971.75 692.488 1970.96 697.003 1952.91 693.832ZM1985.3 694.866 2003.35 698.036 2002.56 702.55 1984.51 699.38ZM2016.9 700.414 2034.95 703.584 2034.16 708.098 2016.11 704.928ZM2048.5 705.961 2066.55 709.131 2065.76 713.646 2047.71 710.476ZM2080.1 711.509 2098.15 714.679 2097.36 719.194 2079.31 716.023ZM2111.7 717.057 2129.75 720.227 2128.96 724.741 2110.91 721.571ZM2143.3 722.605 2161.35 725.775 2160.56 730.289 2142.51 727.119ZM2174.9 728.152 2192.96 731.322 2192.16 735.837 2174.11 732.667ZM2206.5 733.7 2224.56 736.87 2223.76 741.384 2205.71 738.214ZM2238.1 739.248 2256.16 742.418 2255.36 746.932 2237.31 743.762ZM2269.7 744.795 2287.76 747.966 2286.96 752.48 2268.91 749.31ZM2301.3 750.343 2319.36 753.513 2318.56 758.028 2300.51 754.858ZM2332.9 755.891 2350.96 759.061 2350.16 763.575 2332.11 760.405ZM2364.5 761.439 2382.56 764.609 2381.76 769.123 2363.71 765.953ZM2396.1 766.986 2414.16 770.157 2413.36 774.671 2395.31 771.501ZM2427.7 772.534 2445.76 775.704 2444.96 780.219 2426.91 777.048ZM2459.3 778.082 2477.36 781.252 2476.56 785.766 2458.51 782.596ZM2490.9 783.63 2508.96 786.8 2508.16 791.314 2490.11 788.144ZM2522.5 789.177 2540.56 792.348 2539.76 796.862 2521.71 793.692ZM2554.1 794.725 2554.18 794.739 2553.39 799.254 2553.31 799.24ZM2551.64 782.66 2576.35 800.958 2546.89 809.746Z" /><path + id="path52" + transform="matrix(-1 0 0 1 2577.05 1101.5)" + fill="#003967" + d="M0.305813-2.27117 18.4752 0.175336 17.8635 4.71768-0.305813 2.27117ZM32.1022 2.01022 50.2716 4.45672 49.6599 8.99906 31.4906 6.55256ZM63.8986 6.2916 82.0679 8.73811 81.4563 13.2804 63.287 10.8339ZM95.695 10.573 113.864 13.0195 113.253 17.5618 95.0833 15.1153ZM127.491 14.8544 145.661 17.3009 145.049 21.8432 126.88 19.3967ZM159.288 19.1358 177.457 21.5823 176.845 26.1246 158.676 23.6781ZM191.084 23.4171 209.253 25.8636 208.642 30.406 190.472 27.9595ZM222.88 27.6985 241.05 30.145 240.438 34.6874 222.269 32.2409ZM254.677 31.9799 272.846 34.4264 272.235 38.9688 254.065 36.5223ZM286.473 36.2613 304.643 38.7078 304.031 43.2501 285.862 40.8036ZM318.27 40.5427 336.439 42.9892 335.827 47.5315 317.658 45.085ZM350.066 44.8241 368.235 47.2706 367.624 51.8129 349.454 49.3664ZM381.862 49.1054 400.032 51.552 399.42 56.0943 381.251 53.6478ZM413.659 53.3868 431.828 55.8333 431.216 60.3757 413.047 57.9292ZM445.455 57.6682 463.624 60.1147 463.013 64.6571 444.843 62.2106ZM477.251 61.9496 495.421 64.3961 494.809 68.9384 476.64 66.4919ZM509.048 66.231 527.217 68.6775 526.606 73.2198 508.436 70.7733ZM540.844 70.5124 559.014 72.9589 558.402 77.5012 540.233 75.0547ZM572.641 74.7938 590.81 77.2403 590.198 81.7826 572.029 79.3361ZM604.437 79.0751 622.606 81.5217 621.995 86.064 603.825 83.6175ZM636.233 83.3565 654.403 85.803 653.791 90.3454 635.622 87.8989ZM668.03 87.6379 686.199 90.0844 685.587 94.6268 667.418 92.1803ZM699.826 91.9193 717.995 94.3658 717.384 98.9081 699.215 96.4616ZM731.622 96.2007 749.792 98.6472 749.18 103.19 731.011 100.743ZM763.419 100.482 781.588 102.929 780.977 107.471 762.807 105.024ZM795.215 104.763 813.385 107.21 812.773 111.752 794.604 109.306ZM827.012 109.045 845.181 111.491 844.569 116.034 826.4 113.587ZM858.808 113.326 876.978 115.773 876.366 120.315 858.197 117.869ZM890.605 117.608 908.774 120.054 908.162 124.596 889.993 122.15ZM922.401 121.889 940.57 124.335 939.959 128.878 921.789 126.431ZM954.198 126.17 972.367 128.617 971.755 133.159 953.586 130.713ZM985.994 130.452 1004.16 132.898 1003.55 137.441 985.382 134.994ZM1017.79 134.733 1035.96 137.18 1035.35 141.722 1017.18 139.275ZM1049.59 139.015 1067.76 141.461 1067.14 146.003 1048.98 143.557ZM1081.38 143.296 1099.55 145.742 1098.94 150.285 1080.77 147.838ZM1113.18 147.577 1131.35 150.024 1130.74 154.566 1112.57 152.12ZM1144.98 151.859 1163.15 154.305 1162.53 158.848 1144.36 156.401ZM1176.77 156.14 1194.94 158.587 1194.33 163.129 1176.16 160.682ZM1208.57 160.421 1226.74 162.868 1226.13 167.41 1207.96 164.964ZM1240.36 164.703 1258.53 167.149 1257.92 171.692 1239.75 169.245ZM1272.16 168.984 1290.33 171.431 1289.72 175.973 1271.55 173.527ZM1303.96 173.266 1322.13 175.712 1321.52 180.254 1303.35 177.808ZM1335.75 177.547 1353.92 179.993 1353.31 184.536 1335.14 182.089ZM1367.55 181.828 1385.72 184.275 1385.11 188.817 1366.94 186.371ZM1399.35 186.11 1417.52 188.556 1416.9 193.099 1398.74 190.652ZM1431.14 190.391 1449.31 192.838 1448.7 197.38 1430.53 194.934ZM1462.94 194.673 1481.11 197.119 1480.5 201.661 1462.33 199.215ZM1494.74 198.954 1512.91 201.4 1512.29 205.943 1494.12 203.496ZM1526.53 203.235 1544.7 205.682 1544.09 210.224 1525.92 207.778ZM1558.33 207.517 1576.5 209.963 1575.89 214.506 1557.72 212.059ZM1590.12 211.798 1608.29 214.245 1607.68 218.787 1589.51 216.34ZM1621.92 216.079 1640.09 218.526 1639.48 223.068 1621.31 220.622ZM1637.13 206.566 1662.55 223.862 1633.46 233.82Z" /><path + id="path54" + transform="matrix(-1 0 0 1 3693.32 1052.5)" + fill="#003967" + d="M0.223679-2.28072 18.4695-0.491293 18.0221 4.07016-0.223679 2.28072ZM32.1538 0.850781 50.3996 2.64021 49.9523 7.20166 31.7065 5.41223ZM64.084 3.98229 82.3298 5.77172 81.8824 10.3332 63.6366 8.54373ZM96.0141 7.11379 114.26 8.90322 113.813 13.4647 95.5667 11.6752ZM127.944 10.2453 146.19 12.0347 145.743 16.5962 127.497 14.8067ZM159.874 13.3768 178.12 15.1662 177.673 19.7277 159.427 17.9382ZM191.805 16.5083 210.05 18.2977 209.603 22.8592 191.357 21.0698ZM223.735 19.6398 241.98 21.4292 241.533 25.9907 223.287 24.2013ZM255.665 22.7713 273.911 24.5607 273.463 29.1222 255.217 27.3328ZM287.595 25.9028 305.841 27.6923 305.393 32.2537 287.148 30.4643ZM319.525 29.0343 337.771 30.8238 337.323 35.3852 319.078 33.5958ZM351.455 32.1658 369.701 33.9553 369.254 38.5167 351.008 36.7273ZM383.385 35.2973 401.631 37.0868 401.184 41.6482 382.938 39.8588ZM415.315 38.4288 433.561 40.2183 433.114 44.7797 414.868 42.9903ZM447.246 41.5603 465.491 43.3498 465.044 47.9112 446.798 46.1218ZM479.176 44.6919 497.422 46.4813 496.974 51.0427 478.728 49.2533ZM511.106 47.8234 529.352 49.6128 528.904 54.1742 510.659 52.3848ZM543.036 50.9549 561.282 52.7443 560.835 57.3058 542.589 55.5163ZM574.966 54.0864 593.212 55.8758 592.765 60.4373 574.519 58.6478ZM606.896 57.2179 625.142 59.0073 624.695 63.5688 606.449 61.7793ZM638.827 60.3494 657.072 62.1388 656.625 66.7003 638.379 64.9108ZM670.757 63.4809 689.003 65.2703 688.555 69.8318 670.309 68.0423ZM702.687 66.6124 720.933 68.4018 720.485 72.9633 702.24 71.1739ZM734.617 69.7439 752.863 71.5333 752.416 76.0948 734.17 74.3054ZM766.547 72.8754 784.793 74.6648 784.346 79.2263 766.1 77.4369ZM798.477 76.0069 816.723 77.7963 816.276 82.3578 798.03 80.5684ZM830.407 79.1384 848.653 80.9278 848.206 85.4893 829.96 83.6999ZM862.338 82.2699 880.583 84.0594 880.136 88.6208 861.89 86.8314ZM894.268 85.4014 912.513 87.1909 912.066 91.7523 893.82 89.9629ZM926.198 88.5329 944.444 90.3224 943.996 94.8838 925.75 93.0944ZM958.128 91.6644 976.374 93.4539 975.926 98.0153 957.681 96.2259ZM990.058 94.7959 1008.3 96.5854 1007.86 101.147 989.611 99.3574ZM1021.99 97.9274 1040.23 99.7169 1039.79 104.278 1021.54 102.489ZM1053.92 101.059 1072.16 102.848 1071.72 107.41 1053.47 105.62ZM1085.85 104.19 1104.09 105.98 1103.65 110.541 1085.4 108.752ZM1117.78 107.322 1136.02 109.111 1135.58 113.673 1117.33 111.883ZM1149.71 110.453 1167.95 112.243 1167.51 116.804 1149.26 115.015ZM1181.64 113.585 1199.88 115.374 1199.44 119.936 1181.19 118.146ZM1213.57 116.716 1231.81 118.506 1231.37 123.067 1213.12 121.278ZM1245.5 119.848 1263.74 121.637 1263.3 126.199 1245.05 124.409ZM1277.43 122.979 1295.68 124.769 1295.23 129.33 1276.98 127.541ZM1309.36 126.111 1327.61 127.9 1327.16 132.462 1308.91 130.672ZM1341.29 129.242 1359.54 131.032 1359.09 135.593 1340.84 133.804ZM1373.22 132.374 1391.47 134.163 1391.02 138.725 1372.77 136.935ZM1405.15 135.505 1423.4 137.295 1422.95 141.856 1404.7 140.067ZM1437.08 138.637 1455.33 140.426 1454.88 144.988 1436.63 143.198ZM1469.01 141.768 1487.26 143.558 1486.81 148.119 1468.56 146.33ZM1500.94 144.9 1519.19 146.689 1518.74 151.251 1500.49 149.461ZM1532.87 148.031 1551.12 149.821 1550.67 154.382 1532.42 152.593ZM1564.8 151.163 1583.05 152.952 1582.6 157.514 1564.35 155.724ZM1596.73 154.294 1614.98 156.084 1614.53 160.645 1596.28 158.856ZM1628.66 157.426 1646.91 159.215 1646.46 163.777 1628.21 161.987ZM1660.59 160.557 1678.84 162.347 1678.39 166.908 1660.14 165.119ZM1692.52 163.689 1710.77 165.478 1710.32 170.04 1692.07 168.25ZM1724.45 166.82 1742.7 168.61 1742.25 173.171 1724 171.382ZM1756.38 169.952 1774.63 171.741 1774.18 176.303 1755.93 174.513ZM1788.31 173.084 1806.56 174.873 1806.11 179.434 1787.86 177.645ZM1820.24 176.215 1838.49 178.004 1838.04 182.566 1819.79 180.776ZM1852.17 179.347 1870.42 181.136 1869.97 185.697 1851.72 183.908ZM1884.1 182.478 1902.35 184.267 1901.9 188.829 1883.65 187.039ZM1916.03 185.61 1934.28 187.399 1933.83 191.96 1915.58 190.171ZM1947.96 188.741 1966.21 190.53 1965.76 195.092 1947.51 193.302ZM1979.89 191.873 1998.14 193.662 1997.69 198.223 1979.44 196.434ZM2011.82 195.004 2030.07 196.793 2029.62 201.355 2011.37 199.566ZM2043.75 198.136 2062 199.925 2061.55 204.486 2043.31 202.697ZM2075.68 201.267 2093.93 203.057 2093.48 207.618 2075.24 205.829ZM2107.61 204.399 2125.86 206.188 2125.41 210.749 2107.17 208.96ZM2139.54 207.53 2157.79 209.32 2157.34 213.881 2139.1 212.092ZM2171.47 210.662 2189.72 212.451 2189.27 217.012 2171.03 215.223ZM2203.4 213.793 2221.65 215.583 2221.2 220.144 2202.96 218.355ZM2235.33 216.925 2253.58 218.714 2253.13 223.275 2234.89 221.486ZM2267.26 220.056 2285.51 221.846 2285.06 226.407 2266.82 224.618ZM2299.19 223.188 2317.44 224.977 2316.99 229.539 2298.75 227.749ZM2331.12 226.319 2349.37 228.109 2348.92 232.67 2330.68 230.881ZM2363.05 229.451 2381.3 231.24 2380.85 235.802 2362.61 234.012ZM2394.98 232.582 2413.23 234.372 2412.78 238.933 2394.54 237.144ZM2426.91 235.714 2445.16 237.503 2444.71 242.065 2426.47 240.275ZM2458.84 238.845 2477.09 240.635 2476.64 245.196 2458.4 243.407ZM2490.77 241.977 2509.02 243.766 2508.57 248.328 2490.33 246.538ZM2522.7 245.108 2540.95 246.898 2540.5 251.459 2522.26 249.67ZM2554.63 248.24 2572.88 250.029 2572.43 254.591 2554.19 252.801ZM2586.56 251.371 2604.81 253.161 2604.36 257.722 2586.12 255.933ZM2618.49 254.503 2636.74 256.292 2636.29 260.854 2618.05 259.064ZM2650.43 257.634 2668.67 259.424 2668.22 263.985 2649.98 262.196ZM2682.36 260.766 2700.6 262.555 2700.15 267.117 2681.91 265.327ZM2714.29 263.897 2732.53 265.687 2732.08 270.248 2713.84 268.459ZM2746.22 267.029 2756.22 268.01 2755.77 272.571 2745.77 271.59ZM2752.79 256.16 2778.82 272.529 2750.11 283.529Z" /><path + id="path56" + fill="#63666A" + d="M1914.53 945.208 1932.86 945.421 1932.81 950.004 1914.47 949.791ZM1946.61 945.581 1964.94 945.794 1964.89 950.377 1946.55 950.164ZM1978.69 945.954 1997.02 946.167 1996.97 950.75 1978.64 950.537ZM2010.77 946.327 2029.1 946.54 2029.05 951.123 2010.72 950.91ZM2042.85 946.7 2061.18 946.913 2061.13 951.496 2042.8 951.283ZM2074.93 947.073 2093.26 947.286 2093.21 951.869 2074.88 951.656ZM2107.01 947.446 2125.35 947.659 2125.29 952.242 2106.96 952.029ZM2139.09 947.819 2157.43 948.032 2157.37 952.615 2139.04 952.402ZM2171.18 948.192 2189.51 948.405 2189.45 952.988 2171.12 952.775ZM2203.26 948.564 2221.59 948.778 2221.54 953.361 2203.2 953.147ZM2235.34 948.937 2250.5 949.114 2250.45 953.697 2235.28 953.52ZM2246.05 937.603 2273.39 951.671 2245.73 965.101Z" /><text + id="text58" + transform="matrix(1 0 0 1 2945.81 935)" + font-size="55" + font-weight="700" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Not consumed</text> +<path + id="path60" + transform="matrix(1 0 0 -1 2879 950.275)" + fill="#63666A" + d="M0.117146-5.15492 41.3565-4.21775 41.1222 6.09209-0.117146 5.15492ZM72.286-3.51487 113.525-2.5777 113.291 7.73214 72.0517 6.79497ZM144.455-1.87482 185.694-0.937654 185.46 9.37218 144.221 8.43501ZM216.624-0.234776 257.863 0.702394 257.629 11.0122 216.389 10.0751ZM288.793 1.40527 330.032 2.34244 329.798 12.6523 288.558 11.7151ZM360.961 3.04532 402.201 3.98249 401.967 14.2923 360.727 13.3552ZM433.13 4.68537 470.489 5.53436 470.255 15.8442 432.896 14.9952ZM465.568-4.89263 496.147 11.275 464.866 26.0369Z" /><path + id="path62" + fill-rule="evenodd" + fill="none" + stroke-miterlimit="8" + stroke-width="4.58333" + stroke="#212322" + d="M2229.48 937.978 2673.33 630.043 3601.57 1967.99 3157.72 2275.92Z" /><path + id="path64" + fill-rule="evenodd" + fill="#92D050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M2653.5 1505.5C2653.5 1422.66 2789.38 1355.5 2957 1355.5 3124.62 1355.5 3260.5 1422.66 3260.5 1505.5 3260.5 1588.34 3124.62 1655.5 2957 1655.5 2789.38 1655.5 2653.5 1588.34 2653.5 1505.5Z" /><text + id="text66" + transform="matrix(1 0 0 1 2777.09 1532)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Feature</text> +<text + id="text68" + transform="matrix(1 0 0 1 3029.75 1532)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text70" + transform="matrix(1 0 0 1 3054.38 1532)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">2b</text> +<path + id="path72" + fill-rule="evenodd" + fill="#92D050" + stroke-miterlimit="8" + stroke-width="6.875" + stroke="#212322" + d="M2956.5 1942C2956.5 1858.88 3092.38 1791.5 3260 1791.5 3427.62 1791.5 3563.5 1858.88 3563.5 1942 3563.5 2025.12 3427.62 2092.5 3260 2092.5 3092.38 2092.5 2956.5 2025.12 2956.5 1942Z" /><text + id="text74" + transform="matrix(1 0 0 1 3082.34 1969)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Feature</text> +<text + id="text76" + transform="matrix(1 0 0 1 3334.99 1969)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">-</text> +<text + id="text78" + transform="matrix(1 0 0 1 3359.63 1969)" + font-size="73" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">2c</text> +<path + id="path80" + fill="#003967" + d="M2794.5 1054.49 2814.58 1090.53 2805.57 1095.55 2785.5 1059.51ZM2829.63 1117.56 2849.7 1153.59 2840.69 1158.61 2820.62 1122.57ZM2864.76 1180.62 2884.83 1216.66 2875.82 1221.68 2855.75 1185.64ZM2899.88 1243.68 2919.96 1279.72 2910.95 1284.74 2890.88 1248.7ZM2935.01 1306.75 2947.9 1329.89 2938.89 1334.91 2926 1311.77ZM2954.4 1320.37 2955.94 1354.92 2927.37 1335.42Z" /><path + id="path82" + fill="#003967" + d="M3175.63 1608.72 3193.86 1645.72 3184.61 1650.28 3166.37 1613.28ZM3207.53 1673.47 3225.76 1710.48 3216.51 1715.03 3198.28 1678.03ZM3239.44 1738.23 3253.05 1765.85 3243.8 1770.41 3230.19 1742.79ZM3260.02 1756.67 3259.82 1791.25 3232.27 1770.34Z" /><text + id="text84" + transform="matrix(0.494621 0.869109 -0.869109 0.494621 2827.66 1101)" + font-size="55" + font-weight="700" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Consumed</text> +<text + id="text86" + transform="matrix(0.993011 0.118018 -0.118018 0.993011 2156.49 595)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">rte_edge == 3</text> +<text + id="text88" + transform="matrix(0.998819 0.0485938 -0.0485938 0.998819 1925.41 932)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">edge == 0</text> +<text + id="text90" + transform="matrix(0.977645 0.210262 -0.210262 0.977645 1670.89 687)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">rte_edge == 2</text> +<text + id="text92" + transform="matrix(0.922018 0.387147 -0.387147 0.922018 1082.78 630)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">rte_edge == 1</text> +<text + id="text94" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 974.705 1097)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">rte_edge == 0</text> +<text + id="text96" + transform="matrix(-1.83697e-16 -1 1 -1.83697e-16 896.661 1024)" + font-size="64" + font-weight="400" + font-family="Arial,Arial_MSFontService,sans-serif" + fill="#212322">Static Path</text> +</g></svg> -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v6 0/4] add feature arc in rte_graph 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena ` (4 preceding siblings ...) 2024-10-14 14:33 ` [PATCH v5 5/5] docs: add programming guide for feature arc Nitin Saxena @ 2025-01-03 6:06 ` Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 1/4] graph: add API to override node process function Nitin Saxena ` (4 more replies) 5 siblings, 5 replies; 55+ messages in thread From: Nitin Saxena @ 2025-01-03 6:06 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on whether feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Changes in v6: - Rebased to latest main for DPDK-25.03 - Added constructor based feature arc/feature registration - Changed design to handle fast path synchronization via RCU mechanism when any feature is enabled or disabled - Added feature arc specific mbuf dynamic field to carry feature data across nodes - Added feature arc example in app/graph - Programming guide and functional test cases in future versions Nitin Saxena (4): graph: add API to override node process function graph: add feature arc abstraction ip4: add ip4 output feature arc app/graph: add custom feature nodes for ip4 output arc app/graph/commands.list | 6 + app/graph/feature.c | 141 ++ app/graph/feature.h | 13 + app/graph/graph.c | 4 + app/graph/ip4_output_hook.c | 169 ++ app/graph/main.c | 15 +- app/graph/meson.build | 2 + app/graph/module_api.h | 2 + doc/api/doxy-api-index.md | 2 + doc/guides/rel_notes/release_25_03.rst | 10 + lib/graph/graph_feature_arc.c | 1780 ++++++++++++++++++++++ lib/graph/graph_private.h | 15 + lib/graph/meson.build | 4 +- lib/graph/node.c | 23 + lib/graph/rte_graph_feature_arc.h | 552 +++++++ lib/graph/rte_graph_feature_arc_worker.h | 608 ++++++++ lib/graph/version.map | 20 + lib/node/ethdev_ctrl.c | 8 + lib/node/interface_tx_feature.c | 133 ++ lib/node/interface_tx_feature_priv.h | 33 + lib/node/ip4_rewrite.c | 298 +++- lib/node/meson.build | 1 + lib/node/node_private.h | 1 + lib/node/rte_node_ip4_api.h | 4 + 24 files changed, 3838 insertions(+), 6 deletions(-) create mode 100644 app/graph/feature.c create mode 100644 app/graph/feature.h create mode 100644 app/graph/ip4_output_hook.c create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h create mode 100644 lib/node/interface_tx_feature.c create mode 100644 lib/node/interface_tx_feature_priv.h -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v6 1/4] graph: add API to override node process function 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena @ 2025-01-03 6:06 ` Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 2/4] graph: add feature arc abstraction Nitin Saxena ` (3 subsequent siblings) 4 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2025-01-03 6:06 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena New API used by feature arc library to override node's original process() func. Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/graph/graph_private.h | 11 +++++++++++ lib/graph/node.c | 23 +++++++++++++++++++++++ 2 files changed, 34 insertions(+) diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index da48d73587..ceff0c8f50 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -198,6 +198,17 @@ struct node_head *node_list_head_get(void); */ struct node *node_from_name(const char *name); +/** + * @internal + * + * Override process func of a node. + * + * @return + * - 0: Success. + * - <0: Error + */ +int node_override_process_func(rte_node_t id, rte_node_process_t process); + /* Graph list functions */ STAILQ_HEAD(graph_head, graph); diff --git a/lib/graph/node.c b/lib/graph/node.c index 63db629da8..82834a6634 100644 --- a/lib/graph/node.c +++ b/lib/graph/node.c @@ -419,3 +419,26 @@ rte_node_max_count(void) { return node_id; } + +int +node_override_process_func(rte_node_t id, rte_node_process_t process) +{ + struct node *node; + + NODE_ID_CHECK(id); + graph_spinlock_lock(); + + STAILQ_FOREACH(node, &node_list, next) { + if (node->id == id) { + node->process = process; + graph_spinlock_unlock(); + return 0; + } + } + + graph_spinlock_unlock(); + + return 0; +fail: + return -1; +} -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v6 2/4] graph: add feature arc abstraction 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 1/4] graph: add API to override node process function Nitin Saxena @ 2025-01-03 6:06 ` Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 3/4] ip4: add ip4 output feature arc Nitin Saxena ` (2 subsequent siblings) 4 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2025-01-03 6:06 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Feature arc abstraction allows rte_graph based applications to - Hook feature nodes between start_node and end_node of an arc - Feature arc's are created via RTE_GRAPH_FEATURE_ARC_REGISTER() - Feature nodes are added to an arc via RTE_GRAPH_FEATURE_REGISTER() - If application explicitly calls rte_graph_feature_arc_init(), before rte_graph_create(), all features arcs and associated feature nodes are automatically connected - If rte_graph_feature_arc_init() is not called, feature arc module has no affect - Packet path towards feature node(s) is enabled/disabled at runtime on per interface basis. - More than one feature nodes can be added/enabled in an arc - If any feature node is enabled on any interface, feature arc fast path APIs provide next edge for each mbuf Once DPDK inbuilt nodes adopts feature arc abstraction, out-of-tree nodes can be hooked in a generic manner Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- doc/api/doxy-api-index.md | 2 + doc/guides/rel_notes/release_25_03.rst | 10 + lib/graph/graph_feature_arc.c | 1780 ++++++++++++++++++++++ lib/graph/graph_private.h | 4 + lib/graph/meson.build | 4 +- lib/graph/rte_graph_feature_arc.h | 552 +++++++ lib/graph/rte_graph_feature_arc_worker.h | 608 ++++++++ lib/graph/version.map | 20 + 8 files changed, 2979 insertions(+), 1 deletion(-) create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h diff --git a/doc/api/doxy-api-index.md b/doc/api/doxy-api-index.md index f0193502bc..b6a5dedee5 100644 --- a/doc/api/doxy-api-index.md +++ b/doc/api/doxy-api-index.md @@ -213,6 +213,8 @@ The public API headers are grouped by topics: [table_wm](@ref rte_swx_table_wm.h) * [graph](@ref rte_graph.h): [graph_worker](@ref rte_graph_worker.h) + [graph_feature_arc](@ref rte_graph_feature_arc.h) + [graph_feature_arc_worker](@ref rte_graph_feature_arc_worker.h) * graph_nodes: [eth_node](@ref rte_node_eth_api.h), [ip4_node](@ref rte_node_ip4_api.h), diff --git a/doc/guides/rel_notes/release_25_03.rst b/doc/guides/rel_notes/release_25_03.rst index 426dfcd982..205215b5de 100644 --- a/doc/guides/rel_notes/release_25_03.rst +++ b/doc/guides/rel_notes/release_25_03.rst @@ -55,6 +55,16 @@ New Features Also, make sure to start the actual text at the margin. ======================================================= +* **Added feature arc abstraction in graph library.** + + Feature arc abstraction helps ``rte_graph`` based applications to steer + packets across different node path(s) based on the features (or protocols) + enabled on interfaces. Different feature node paths can be enabled/disabled + at runtime on some or on all interfaces. This abstraction also help + applications to hook ``out-of-tree nodes`` in in-built DPDK node paths + in a generic manner. + + * Added ``ip4_output`` feature arc processing in ``ip4_rewrite`` node. Removed Items ------------- diff --git a/lib/graph/graph_feature_arc.c b/lib/graph/graph_feature_arc.c new file mode 100644 index 0000000000..895ec68f86 --- /dev/null +++ b/lib/graph/graph_feature_arc.c @@ -0,0 +1,1780 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#include "graph_private.h" +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> +#include <rte_string_fns.h> + +#define GRAPH_FEATURE_ARC_INITIALIZER UINT64_MAX +#define GRAPH_FEATURE_MAX_NUM_PER_ARC (64) + +#define connect_graph_nodes(node1, node2, edge, arc_name) \ + __connect_graph_nodes(node1, node2, edge, arc_name, __LINE__) + +#define FEATURE_ARC_MEMZONE_NAME "__rte_feature_arc_main_mz" + +#define graph_uint_cast(f) ((unsigned int)f) + +#define fdata_from_feat(arc, feat, index) \ + RTE_GRAPH_FEATURE_TO_FEATURE_DATA(arc, feat, index) + +#define feat_dbg graph_dbg + +#define FEAT_COND_ERR(cond, ...) \ + do { \ + if (cond) \ + graph_err(__VA_ARGS__); \ + } while (0) + +#define FEAT_ERR(fn, ln, ...) \ + GRAPH_LOG2(ERR, fn, ln, __VA_ARGS__) + +#define FEAT_ERR_JMP(_err, fn, ln, ...) \ + do { \ + FEAT_ERR(fn, ln, __VA_ARGS__); \ + rte_errno = _err; \ + } while (0) + +static struct rte_mbuf_dynfield rte_graph_feature_arc_mbuf_desc = { + .name = RTE_GRAPH_FEATURE_ARC_DYNFIELD_NAME, + .size = sizeof(struct rte_graph_feature_arc_mbuf_dynfields), + .align = alignof(struct rte_graph_feature_arc_mbuf_dynfields), +}; + +rte_graph_feature_arc_main_t *__rte_graph_feature_arc_main; +int __rte_graph_feature_arc_mbuf_dyn_offset = -1; + +/* global feature arc list */ +static STAILQ_HEAD(, rte_graph_feature_arc_register) feature_arc_list = + STAILQ_HEAD_INITIALIZER(feature_arc_list); + +/* global feature arc list */ +static STAILQ_HEAD(, rte_graph_feature_register) feature_list = + STAILQ_HEAD_INITIALIZER(feature_list); + +/* feature registration validate */ +static int +feature_registration_validate(struct rte_graph_feature_register *feat_entry, + const char *caller_name, int lineno, + int check_node_reg_id, /* check feature_node->id */ + int check_feat_reg_id /* check feature->feature_node_id */) +{ + if (!feat_entry) { + FEAT_ERR(caller_name, lineno, "NULL feature reg"); + return -1; + } + + if (!feat_entry->feature_name) { + FEAT_ERR(caller_name, lineno, + "NULL feature name %p", feat_entry); + return -1; + } + + if (!feat_entry->arc_name) { + FEAT_ERR(caller_name, lineno, + "No associated arc provided for feature: %s", + feat_entry->feature_name); + return -1; + } + + if (!feat_entry->feature_process_fn) { + FEAT_ERR(caller_name, lineno, + "No process function provided for feature: %s", + feat_entry->feature_name); + return -1; + } + + if (!feat_entry->feature_node) { + FEAT_ERR(caller_name, lineno, + "No feature node provided for feature: %s", + feat_entry->feature_name); + return -1; + } + + if (check_node_reg_id && (feat_entry->feature_node->id == RTE_NODE_ID_INVALID)) { + FEAT_ERR(caller_name, lineno, + "feature_node with invalid node id found for feature: %s", + feat_entry->feature_name); + return -1; + } + + if (check_feat_reg_id && (feat_entry->feature_node_id == RTE_NODE_ID_INVALID)) { + FEAT_ERR(caller_name, lineno, + "feature_node_id found invalid for feature: %s", + feat_entry->feature_name); + return -1; + } + + return 0; + +} + +/* validate arc registration */ +static int +arc_registration_validate(struct rte_graph_feature_arc_register *reg, + const char *caller_name, int lineno) +{ + if (!reg->arc_name) { + FEAT_ERR_JMP(EINVAL, caller_name, lineno, + "feature_arc name cannot be NULL"); + return -1; + } + + if (reg->max_features > GRAPH_FEATURE_MAX_NUM_PER_ARC) { + FEAT_ERR_JMP(EAGAIN, caller_name, lineno, + "arc: %s, number of features are more than max", + reg->arc_name); + return -1; + } + + if (!reg->max_indexes) { + FEAT_ERR_JMP(EINVAL, caller_name, lineno, + "Zero max_indexes found for arc: %s", + reg->arc_name); + return -1; + } + + if (!reg->start_node) { + FEAT_ERR_JMP(EINVAL, caller_name, lineno, + "start node cannot be NULL for arc: %s", + reg->arc_name); + return -1; + } + + if (!reg->start_node_feature_process_fn) { + FEAT_ERR_JMP(EINVAL, caller_name, lineno, + "start node feature_process_fn() cannot be NULL for arc: %s", + reg->arc_name); + return -1; + } + + return (feature_registration_validate(reg->end_feature, caller_name, lineno, 0, 0)); +} + +/* lookup arc registration by name */ +static int arc_registration_num(void) +{ + struct rte_graph_feature_arc_register *entry = NULL; + int num = 0; + + STAILQ_FOREACH(entry, &feature_arc_list, next_arc) + num++; + + return num; +} + + +/* lookup arc registration by name */ +static int arc_registration_lookup(const char *arc_name, + struct rte_graph_feature_arc_register **arc_entry) +{ + struct rte_graph_feature_arc_register *entry = NULL; + + STAILQ_FOREACH(entry, &feature_arc_list, next_arc) { + if (arc_registration_validate(entry, __func__, __LINE__) < 0) + continue; + + if (!strncmp(entry->arc_name, arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN)) { + if (arc_entry) + *arc_entry = entry; + return 1; + } + } + + return 0; +} + + +/* Number of features registered for an ARC + * + * i.e number of RTE_GRAPH_FEATURE_REGISTER() for an arc + */ +static int +arc_registered_features_num(const char *arc_name, uint32_t *num_features) +{ + struct rte_graph_feature_arc_register *arc_reg = NULL; + struct rte_graph_feature_register *feat_entry = NULL; + uint32_t num = 0; + + /* Check if arc is registered with end_feature */ + if (!arc_registration_lookup(arc_name, &arc_reg)) + return -1; + + if (arc_reg->end_feature) + num++; + + /* Calculate features other than end_feature added in arc */ + STAILQ_FOREACH(feat_entry, &feature_list, next_feature) { + if (feature_registration_validate(feat_entry, __func__, __LINE__, 1, 0) < 0) + continue; + + if (!strncmp(feat_entry->arc_name, arc_name, strlen(feat_entry->arc_name))) + num++; + } + + if (num_features) + *num_features = num; + + return 0; +} + +/* calculate arc size to be allocated */ +static int +feature_arc_reg_calc_size(struct rte_graph_feature_arc_register *reg, size_t *sz, + uint16_t *feat_off, uint16_t *fdata_off, uint32_t *fsz) +{ + size_t ff_size = 0, fdata_size = 0; + + /* first feature array per index */ + ff_size = RTE_ALIGN_CEIL(sizeof(rte_graph_feature_t) * reg->max_indexes, + RTE_CACHE_LINE_SIZE); + + /* fdata size per feature */ + *fsz = (uint32_t)RTE_ALIGN_CEIL(sizeof(struct rte_graph_feature_data) * reg->max_indexes, + RTE_CACHE_LINE_SIZE); + + /* Allocate feature_data extra by 1. Used in feature_disable */ + fdata_size = (*fsz) * (reg->max_features + 1); + + if (sz) + *sz = fdata_size + ff_size + sizeof(struct rte_graph_feature_arc); + if (feat_off) + *feat_off = sizeof(struct rte_graph_feature_arc); + if (fdata_off) + *fdata_off = ff_size + sizeof(struct rte_graph_feature_arc); + + return 0; +} + +static rte_graph_feature_t * +graph_first_feature_ptr_get(struct rte_graph_feature_arc *arc, + uint32_t index) +{ + return (rte_graph_feature_t *)((uint8_t *)arc + arc->fp_first_feature_offset + + (sizeof(rte_graph_feature_t) * index)); +} + +static int +feature_arc_data_reset(struct rte_graph_feature_arc *arc) +{ + rte_graph_feature_data_t first_fdata; + struct rte_graph_feature_data *fdata; + rte_graph_feature_t iter, *f = NULL; + uint16_t index; + + arc->runtime_enabled_features = 0; + + for (index = 0; index < arc->max_indexes; index++) { + f = graph_first_feature_ptr_get(arc, index); + *f = RTE_GRAPH_FEATURE_INVALID; + } + + for (iter = 0; iter < arc->max_features; iter++) { + first_fdata = fdata_from_feat(arc, iter, 0); + for (index = 0; index < arc->max_indexes; index++) { + fdata = rte_graph_feature_data_get(arc, first_fdata + index); + fdata->next_feature_data = RTE_GRAPH_FEATURE_INVALID; + fdata->app_cookie = UINT32_MAX; + fdata->next_edge = RTE_EDGE_ID_INVALID; + } + } + return 0; +} + +/* + * lookup feature name and get control path node_list as well as feature index + * at which it is inserted + */ +static int +nodeinfo_lkup_by_name(struct rte_graph_feature_arc *arc, const char *feat_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t fi = 0; + + if (!feat_name) + return -1; + + if (slot) + *slot = UINT32_MAX; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + if (!strncmp(finfo->feature_name, feat_name, strlen(finfo->feature_name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + fi++; + } + return -1; +} + +/* Lookup used only during rte_graph_feature_add() */ +static int +nodeinfo_add_lookup(struct rte_graph_feature_arc *arc, const char *feat_node_name, + struct rte_graph_feature_node_list **ffinfo, uint32_t *slot) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t fi = 0; + + if (!feat_node_name) + return -1; + + if (slot) + *slot = 0; + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + RTE_VERIFY(finfo->feature_arc == arc); + if (!strncmp(finfo->feature_name, feat_node_name, strlen(finfo->feature_name))) { + if (ffinfo) + *ffinfo = finfo; + if (slot) + *slot = fi; + return 0; + } + /* Update slot where new feature can be added */ + if (slot) + *slot = fi; + fi++; + } + + return -1; +} + +/* Get control path node info from provided input feature_index */ +static int +nodeinfo_lkup_by_index(struct rte_graph_feature_arc *arc, uint32_t feature_index, + struct rte_graph_feature_node_list **ppfinfo, + const int do_sanity_check) +{ + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0; + + if (!ppfinfo) + return -1; + + *ppfinfo = NULL; + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + /* Check sanity */ + if (do_sanity_check) + if (finfo->finfo_index != index) + RTE_VERIFY(0); + if (index == feature_index) { + *ppfinfo = finfo; + return 0; + } + index++; + } + return -1; +} + +/* get existing edge from parent_node -> child_node */ +static int +get_existing_edge(const char *arc_name, rte_node_t parent_node, + rte_node_t child_node, rte_edge_t *_edge) +{ + char **next_edges = NULL; + uint32_t i, count = 0; + + RTE_SET_USED(arc_name); + + count = rte_node_edge_get(parent_node, NULL); + + if (!count) + return -1; + + next_edges = malloc(count); + + if (!next_edges) + return -1; + + count = rte_node_edge_get(parent_node, next_edges); + for (i = 0; i < count; i++) { + if (strstr(rte_node_id_to_name(child_node), next_edges[i])) { + if (_edge) + *_edge = (rte_edge_t)i; + + free(next_edges); + return 0; + } + } + free(next_edges); + + return -1; +} + + +/* prepare feature arc after addition of all features */ +static int +prepare_feature_arc_before_first_enable(struct rte_graph_feature_arc *arc) +{ + struct rte_graph_feature_node_list *lfinfo = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t index = 0, iter; + rte_edge_t edge; + + STAILQ_FOREACH(lfinfo, &arc->all_features, next_feature) { + lfinfo->finfo_index = index; + index++; + } + if (!index) { + graph_err("No feature added to arc: %s", arc->feature_arc_name); + return -1; + } + + nodeinfo_lkup_by_index(arc, index - 1, &lfinfo, 0); + + /* lfinfo should be the info corresponding to end_feature + * Add edge from all features to end feature node to have exception path + * in fast path from all feature nodes to end feature node during enable/disable + */ + if (lfinfo->feature_node_id != arc->end_feature.feature_node_id) { + graph_err("end_feature node mismatch [found-%s: exp-%s]", + rte_node_id_to_name(lfinfo->feature_node_id), + rte_node_id_to_name(arc->end_feature.feature_node_id)); + return -1; + } + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) { + if (get_existing_edge(arc->feature_arc_name, arc->start_node->id, + finfo->feature_node_id, &edge)) { + graph_err("No edge found from %s to %s", + rte_node_id_to_name(arc->start_node->id), + rte_node_id_to_name(finfo->feature_node_id)); + return -1; + } + finfo->edge_to_this_feature = edge; + + if (finfo == lfinfo) + continue; + + if (get_existing_edge(arc->feature_arc_name, finfo->feature_node_id, + lfinfo->feature_node_id, &edge)) { + graph_err("No edge found from %s to %s", + rte_node_id_to_name(finfo->feature_node_id), + rte_node_id_to_name(lfinfo->feature_node_id)); + return -1; + } + finfo->edge_to_last_feature = edge; + } + /** + * Enable end_feature in control bitmask + * (arc->feature_bit_mask_by_index) but not in fast path bitmask + * arc->fp_feature_enable_bitmask. This is due to: + * 1. Application may not explicitly enabling end_feature node + * 2. However it should be enabled internally so that when a feature is + * disabled (say on an interface), next_edge of data should be + * updated to end_feature node hence packet can exit arc. + * 3. We do not want to set bit for end_feature in fast path bitmask as + * it will void the purpose of fast path APIs + * rte_graph_feature_arc_is_any_feature_enabled() and + * rte_graph_feature_arc_is_feature_enabled(). Since enabling + * end_feature would make these APIs to always return "true" + */ + for (iter = 0; iter < arc->max_indexes; iter++) + arc->feature_bit_mask_by_index[iter] |= (1 << lfinfo->finfo_index); + + return 0; +} + +/* feature arc sanity */ +static int +feature_arc_sanity(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + uint16_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + if (!arc) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (arc == rte_graph_feature_arc_get(iter)) { + if (arc->feature_arc_index != iter) + return -1; + if (arc->feature_arc_main != dm) + return -1; + + return 0; + } + } + return -1; +} + +/* create or retrieve already existing edge from parent_node -> child_node */ +static int +__connect_graph_nodes(rte_node_t parent_node, rte_node_t child_node, + rte_edge_t *_edge, char *arc_name, int lineno) +{ + const char *next_node = NULL; + rte_edge_t edge; + + if (!get_existing_edge(arc_name, parent_node, child_node, &edge)) { + feat_dbg("\t%s/%d: %s[%u]: \"%s\", edge reused", arc_name, lineno, + rte_node_id_to_name(parent_node), edge, rte_node_id_to_name(child_node)); + + if (_edge) + *_edge = edge; + + return 0; + } + + /* Node to be added */ + next_node = rte_node_id_to_name(child_node); + + edge = rte_node_edge_update(parent_node, RTE_EDGE_ID_INVALID, &next_node, 1); + + if (edge == RTE_EDGE_ID_INVALID) { + graph_err("edge invalid"); + return -1; + } + edge = rte_node_edge_count(parent_node) - 1; + + feat_dbg("\t%s/%d: %s[%u]: \"%s\", new edge added", arc_name, lineno, + rte_node_id_to_name(parent_node), edge, rte_node_id_to_name(child_node)); + + if (_edge) + *_edge = edge; + + return 0; +} + +/* feature arc initialization */ +static int +feature_arc_main_init(rte_graph_feature_arc_main_t **pfl, uint32_t max_feature_arcs) +{ + rte_graph_feature_arc_main_t *pm = NULL; + const struct rte_memzone *mz = NULL; + uint32_t i; + size_t sz; + + if (!pfl) { + graph_err("Invalid input"); + return -1; + } + + __rte_graph_feature_arc_mbuf_dyn_offset = + rte_mbuf_dynfield_register(&rte_graph_feature_arc_mbuf_desc); + + if (__rte_graph_feature_arc_mbuf_dyn_offset < 0) { + graph_err("rte_graph_feature_arc_dynfield_register failed"); + return -1; + } + + sz = sizeof(rte_graph_feature_arc_main_t) + + (sizeof(pm->feature_arcs[0]) * max_feature_arcs); + + mz = rte_memzone_reserve(FEATURE_ARC_MEMZONE_NAME, sz, SOCKET_ID_ANY, 0); + if (!mz) { + graph_err("memzone reserve failed for feature arc main"); + return -1; + } + + pm = mz->addr; + memset(pm, 0, sz); + + for (i = 0; i < max_feature_arcs; i++) + pm->feature_arcs[i] = GRAPH_FEATURE_ARC_INITIALIZER; + + pm->max_feature_arcs = max_feature_arcs; + + *pfl = pm; + + return 0; +} + +/* feature arc initialization, public API */ +int +rte_graph_feature_arc_init(void) +{ + struct rte_graph_feature_arc_register *arc_reg = NULL; + struct rte_graph_feature_register *feat_reg = NULL; + const struct rte_memzone *mz = NULL; + int max_feature_arcs; + int rc = -1; + + max_feature_arcs = arc_registration_num(); + + if (!max_feature_arcs) { + graph_err("No feature arcs registered"); + return -1; + } + + if (!__rte_graph_feature_arc_main) { + mz = rte_memzone_lookup(FEATURE_ARC_MEMZONE_NAME); + if (mz) { + __rte_graph_feature_arc_main = mz->addr; + __rte_graph_feature_arc_mbuf_dyn_offset = + rte_mbuf_dynfield_lookup(RTE_GRAPH_FEATURE_ARC_DYNFIELD_NAME, + &rte_graph_feature_arc_mbuf_desc); + } else { + rc = feature_arc_main_init(&__rte_graph_feature_arc_main, max_feature_arcs); + if (rc < 0) + return rc; + } + } + + STAILQ_FOREACH(arc_reg, &feature_arc_list, next_arc) { + /* validate end feature */ + if (feature_registration_validate(arc_reg->end_feature, + __func__, __LINE__, 1, 0) < 0) + continue; + + if (strncmp(arc_reg->arc_name, arc_reg->end_feature->arc_name, + RTE_GRAPH_FEATURE_ARC_NAMELEN)) { + feat_dbg("arc-%s: mismatch in arc_name for end_feature: %s", + arc_reg->arc_name, arc_reg->end_feature->arc_name); + continue; + } + + if (!arc_registration_lookup(arc_reg->arc_name, NULL)) + continue; + + /* If feature name not set, use node name as feature */ + if (!arc_reg->end_feature->feature_name) + arc_reg->end_feature->feature_name = + rte_node_id_to_name(arc_reg->end_feature->feature_node_id); + + /* If max_features not set, calculate number of static feature registrations */ + if (!arc_reg->max_features) + arc_registered_features_num(arc_reg->arc_name, &arc_reg->max_features); + + arc_reg->end_feature->feature_node_id = arc_reg->end_feature->feature_node->id; + + rc = rte_graph_feature_arc_create(arc_reg, NULL); + + if (rc < 0) + goto arc_cleanup; + + rc = rte_graph_feature_add(arc_reg->end_feature); + + if (rc < 0) + goto arc_cleanup; + } + + /* First add those features which has no runs_after and runs_before restriction */ + STAILQ_FOREACH(feat_reg, &feature_list, next_feature) { + if (feat_reg->runs_after || feat_reg->runs_before) + continue; + + if (feature_registration_validate(feat_reg, __func__, __LINE__, 1, 0) < 0) + continue; + + feat_reg->feature_node_id = feat_reg->feature_node->id; + + rc = rte_graph_feature_add(feat_reg); + + if (rc < 0) + goto arc_cleanup; + } + /* Add those features which has either runs_after or runs_before restrictions */ + STAILQ_FOREACH(feat_reg, &feature_list, next_feature) { + if (!feat_reg->runs_after && !feat_reg->runs_before) + continue; + + if (feat_reg->runs_after && feat_reg->runs_before) + continue; + + if (feature_registration_validate(feat_reg, __func__, __LINE__, 1, 0) < 0) + continue; + + feat_reg->feature_node_id = feat_reg->feature_node->id; + + rc = rte_graph_feature_add(feat_reg); + + if (rc < 0) + goto arc_cleanup; + } + /* Add those features with both runs_after and runs_before restrictions */ + STAILQ_FOREACH(feat_reg, &feature_list, next_feature) { + if (!feat_reg->runs_after && !feat_reg->runs_before) + continue; + + if ((feat_reg->runs_after && !feat_reg->runs_before) || + (!feat_reg->runs_after && feat_reg->runs_before)) + continue; + + if (feature_registration_validate(feat_reg, __func__, __LINE__, 1, 0) < 0) + continue; + + feat_reg->feature_node_id = feat_reg->feature_node->id; + + rc = rte_graph_feature_add(feat_reg); + + if (rc < 0) + goto arc_cleanup; + } + + return 0; + +arc_cleanup: + rte_graph_feature_arc_cleanup(); + + return rc; +} + +int +rte_graph_feature_arc_create(struct rte_graph_feature_arc_register *reg, + rte_graph_feature_arc_t *_arc) +{ + rte_graph_feature_arc_main_t *dfm = NULL; + struct rte_graph_feature_arc *arc = NULL; + uint16_t first_feat_off, fdata_off; + const struct rte_memzone *mz = NULL; + uint16_t iter, arc_index; + uint32_t feat_sz = 0; + size_t sz; + + if (arc_registration_validate(reg, __func__, __LINE__) < 0) + return -1; + + if (!reg->max_features) + graph_err("Zero features found for arc \"%s\" create", + reg->arc_name); + + if (!__rte_graph_feature_arc_main) { + mz = rte_memzone_lookup(FEATURE_ARC_MEMZONE_NAME); + if (mz) { + __rte_graph_feature_arc_main = mz->addr; + } else { + graph_err("Call to rte_graph_feature_arc_init() API missing"); + return -1; + } + } + + /* See if arc memory is already created */ + mz = rte_memzone_lookup(reg->arc_name); + if (mz) { + graph_warn("Feature arc %s already created", reg->arc_name); + arc = mz->addr; + if (_arc) + *_arc = arc->feature_arc_index; + + arc->process_ref_count++; + + return 0; + } + + dfm = __rte_graph_feature_arc_main; + + /* threshold check */ + if (dfm->num_feature_arcs > (dfm->max_feature_arcs - 1)) + SET_ERR_JMP(EAGAIN, arc_create_err, + "%s: max number (%u) of feature arcs reached", + reg->arc_name, dfm->max_feature_arcs); + + /* Find the free slot for feature arc */ + for (iter = 0; iter < dfm->max_feature_arcs; iter++) { + if (dfm->feature_arcs[iter] == GRAPH_FEATURE_ARC_INITIALIZER) + break; + } + arc_index = iter; + + if (arc_index >= dfm->max_feature_arcs) { + graph_err("No free slot found for num_feature_arc"); + return -1; + } + + /* This should not happen */ + if (dfm->feature_arcs[arc_index] != GRAPH_FEATURE_ARC_INITIALIZER) { + graph_err("Free arc_index: %u is not found free: %p", + arc_index, (void *)dfm->feature_arcs[arc_index]); + return -1; + } + + /* Calculate size of feature arc */ + feature_arc_reg_calc_size(reg, &sz, &first_feat_off, &fdata_off, &feat_sz); + + mz = rte_memzone_reserve(reg->arc_name, sz, SOCKET_ID_ANY, 0); + + if (!mz) { + graph_err("memzone reserve failed for arc: %s of size: %lu", + reg->arc_name, sz); + return -1; + } + + arc = mz->addr; + + memset(arc, 0, sz); + + arc->feature_bit_mask_by_index = rte_malloc(reg->arc_name, + sizeof(uint64_t) * reg->max_indexes, 0); + + if (!arc->feature_bit_mask_by_index) { + graph_err("%s: rte_malloc failed for feature_bit_mask_alloc", reg->arc_name); + rte_memzone_free(mz); + return -1; + } + + memset(arc->feature_bit_mask_by_index, 0, sizeof(uint64_t) * reg->max_indexes); + + /* override process function with start_node */ + if (node_override_process_func(reg->start_node->id, reg->start_node_feature_process_fn)) { + graph_err("node_override_process_func failed for %s", reg->start_node->name); + rte_free(arc->feature_bit_mask_by_index); + rte_memzone_free(mz); + return -1; + } + feat_dbg("arc-%s: node-%s process() overridden with %p", + reg->arc_name, reg->start_node->name, + reg->start_node_feature_process_fn); + + /* Initialize rte_graph port group fixed variables */ + STAILQ_INIT(&arc->all_features); + rte_strscpy(arc->feature_arc_name, reg->arc_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + arc->feature_arc_main = (void *)dfm; + arc->start_node = reg->start_node; + memcpy(&arc->end_feature, reg->end_feature, sizeof(arc->end_feature)); + arc->arc_start_process = reg->start_node_feature_process_fn; + arc->feature_arc_index = arc_index; + arc->arc_size = sz; + + /* reset fast path arc variables */ + arc->max_features = reg->max_features; + arc->max_indexes = reg->max_indexes; + arc->fp_first_feature_offset = first_feat_off; + arc->fp_feature_data_offset = fdata_off; + arc->fp_feature_size = feat_sz; + + arc->process_ref_count++; + + feature_arc_data_reset(arc); + + dfm->feature_arcs[arc->feature_arc_index] = (uintptr_t)arc; + dfm->num_feature_arcs++; + + if (_arc) + *_arc = (rte_graph_feature_arc_t)arc_index; + + feat_dbg("Feature arc %s[%p] created with max_features: %u and indexes: %u", + arc->feature_arc_name, (void *)arc, arc->max_features, arc->max_indexes); + + return 0; + +arc_create_err: + return -1; +} + +int +rte_graph_feature_add(struct rte_graph_feature_register *freg) +{ + struct rte_graph_feature_node_list *after_finfo = NULL, *before_finfo = NULL; + struct rte_graph_feature_node_list *temp = NULL, *finfo = NULL; + char feature_name[3 * RTE_GRAPH_FEATURE_ARC_NAMELEN]; + const char *runs_after = NULL, *runs_before = NULL; + struct rte_graph_feature_arc *arc = NULL; + uint32_t slot = UINT32_MAX, add_flag; + rte_graph_feature_arc_t _arc; + uint32_t num_features = 0; + const char *nodename = NULL; + rte_edge_t edge = -1; + int rc = 0; + + if (feature_registration_validate(freg, __func__, __LINE__, 0, 1) < 0) + return -1; + + /* arc is valid */ + if (rte_graph_feature_arc_lookup_by_name(freg->arc_name, &_arc)) { + graph_err("%s_add: feature arc %s not found", + freg->feature_name, freg->arc_name); + return -1; + } + + if (feature_arc_sanity(_arc)) { + graph_err("invalid feature arc: 0x%x", _arc); + return -1; + } + + arc = rte_graph_feature_arc_get(_arc); + + if (arc->runtime_enabled_features) { + graph_err("adding features after enabling any one of them is not supported"); + return -1; + } + + /* When application calls rte_graph_feature_add() directly*/ + if (freg->feature_node_id == RTE_NODE_ID_INVALID) { + graph_err("%s/%s: Invalid feature_node_id set for %s", + freg->arc_name, freg->feature_name, __func__); + return -1; + } + + if ((freg->runs_after != NULL) && (freg->runs_before != NULL) && + (freg->runs_after == freg->runs_before)) { + graph_err("runs_after and runs_before cannot be same '%s:%s]", freg->runs_after, + freg->runs_before); + return -1; + } + + num_features = rte_graph_feature_arc_num_features(_arc); + nodeinfo_lkup_by_index(arc, num_features - 1, &temp, 0); + + /* Check if feature is not added after end_feature */ + if (num_features && (freg->runs_after != NULL) && + (strncmp(freg->runs_after, temp->feature_name, + RTE_GRAPH_FEATURE_ARC_NAMELEN) == 0)) { + graph_err("Feature %s cannot be added after end_feature %s", + freg->feature_name, freg->runs_after); + return -1; + } + + if (!nodeinfo_add_lookup(arc, freg->feature_name, &finfo, &slot)) { + graph_err("%s/%s feature already added", arc->feature_arc_name, freg->feature_name); + return -1; + } + + if (slot >= arc->max_features) { + graph_err("%s: Max features %u added to feature arc", + arc->feature_arc_name, slot); + return -1; + } + + if (freg->feature_node_id == arc->start_node->id) { + graph_err("%s/%s: Feature node and start node are same %u", + freg->arc_name, freg->feature_name, freg->feature_node_id); + return -1; + } + + nodename = rte_node_id_to_name(freg->feature_node_id); + + feat_dbg("%s: adding feature node: %s at feature index: %u", arc->feature_arc_name, + nodename, slot); + + if (connect_graph_nodes(arc->start_node->id, freg->feature_node_id, &edge, + arc->feature_arc_name)) { + graph_err("unable to connect %s -> %s", arc->start_node->name, nodename); + return -1; + } + + snprintf(feature_name, sizeof(feature_name), "%s-%s-finfo", + arc->feature_arc_name, freg->feature_name); + + finfo = rte_malloc(feature_name, sizeof(*finfo), 0); + if (!finfo) { + graph_err("%s/%s: rte_malloc failed", arc->feature_arc_name, freg->feature_name); + return -1; + } + + memset(finfo, 0, sizeof(*finfo)); + + rte_strscpy(finfo->feature_name, freg->feature_name, RTE_GRAPH_FEATURE_ARC_NAMELEN - 1); + finfo->feature_arc = (void *)arc; + finfo->feature_node_id = freg->feature_node_id; + finfo->feature_node_process_fn = freg->feature_process_fn; + finfo->edge_to_this_feature = RTE_EDGE_ID_INVALID; + finfo->edge_to_last_feature = RTE_EDGE_ID_INVALID; + finfo->notifier_cb = freg->notifier_cb; + + runs_before = freg->runs_before; + runs_after = freg->runs_after; + + /* + * if no constraints given and provided feature is not the first feature, + * explicitly set "runs_before" as end_feature. + * + * Handles the case: + * arc_create(f1); + * add(f2, NULL, NULL); + */ + if (!runs_after && !runs_before && num_features) + runs_before = rte_graph_feature_arc_feature_to_name(_arc, num_features - 1); + + /* Check for before and after constraints */ + if (runs_before) { + /* runs_before sanity */ + if (nodeinfo_lkup_by_name(arc, runs_before, &before_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid before feature name: %s", runs_before); + + if (!before_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_before %s does not exist", runs_before); + + /* + * Starting from 0 to runs_before, continue connecting edges + */ + add_flag = 1; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (!add_flag) + /* Nodes after seeing "runs_before", finfo connects to temp*/ + connect_graph_nodes(finfo->feature_node_id, temp->feature_node_id, + NULL, arc->feature_arc_name); + /* + * As soon as we see runs_before. stop adding edges + */ + if (!strncmp(temp->feature_name, runs_before, RTE_GRAPH_NAMESIZE)) { + if (!connect_graph_nodes(finfo->feature_node_id, + temp->feature_node_id, + &edge, arc->feature_arc_name)) + add_flag = 0; + } + + if (add_flag) + /* Nodes before seeing "run_before" are connected to finfo */ + connect_graph_nodes(temp->feature_node_id, finfo->feature_node_id, + NULL, arc->feature_arc_name); + } + } + + if (runs_after) { + if (nodeinfo_lkup_by_name(arc, runs_after, &after_finfo, NULL)) + SET_ERR_JMP(EINVAL, finfo_free, + "Invalid after feature_name %s", runs_after); + + if (!after_finfo) + SET_ERR_JMP(EINVAL, finfo_free, + "runs_after %s does not exist", runs_after); + + /* Starting from runs_after to end continue connecting edges */ + add_flag = 0; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (add_flag) + /* We have already seen runs_after now */ + /* Add all features as next node to current feature*/ + connect_graph_nodes(finfo->feature_node_id, temp->feature_node_id, + NULL, arc->feature_arc_name); + else + /* Connect initial nodes to newly added node*/ + connect_graph_nodes(temp->feature_node_id, finfo->feature_node_id, + NULL, arc->feature_arc_name); + + /* as soon as we see runs_after. start adding edges + * from next iteration + */ + if (!strncmp(temp->feature_name, runs_after, RTE_GRAPH_NAMESIZE)) + add_flag = 1; + } + + /* add feature next to runs_after */ + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, finfo, next_feature); + } else { + if (before_finfo) { + /* add finfo before "before_finfo" element in the list */ + after_finfo = NULL; + STAILQ_FOREACH(temp, &arc->all_features, next_feature) { + if (before_finfo == temp) { + if (after_finfo) + STAILQ_INSERT_AFTER(&arc->all_features, after_finfo, + finfo, next_feature); + else + STAILQ_INSERT_HEAD(&arc->all_features, finfo, + next_feature); + + return 0; + } + after_finfo = temp; + } + } else { + /* Very first feature just needs to be added to list */ + STAILQ_INSERT_TAIL(&arc->all_features, finfo, next_feature); + } + } + /* override node_process_fn */ + rc = node_override_process_func(finfo->feature_node_id, freg->feature_process_fn); + if (rc < 0) { + graph_err("node_override_process_func failed for %s", freg->feature_name); + goto finfo_free; + } + + if (freg->feature_node) + feat_dbg("arc-%s: feature %s node %s process() overridden with %p", + freg->arc_name, freg->feature_name, freg->feature_node->name, + freg->feature_process_fn); + else + feat_dbg("arc-%s: feature %s nodeid %u process() overriding with %p", + freg->arc_name, freg->feature_name, + freg->feature_node_id, freg->feature_process_fn); + + return 0; +finfo_free: + rte_free(finfo); + + return -1; +} + +int +rte_graph_feature_lookup(rte_graph_feature_arc_t _arc, const char *feature_name, + rte_graph_feature_t *feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot; + + if (!arc) + return -1; + + if (!nodeinfo_lkup_by_name(arc, feature_name, &finfo, &slot)) { + *feat = (rte_graph_feature_t) slot; + return 0; + } + + return -1; +} + +static int +feature_enable_disable_validate(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, + int is_enable_disable, bool emit_logs) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot, last_end_feature; + + if (!arc) + return -EINVAL; + + /* validate _arc */ + if (arc->feature_arc_main != __rte_graph_feature_arc_main) { + FEAT_COND_ERR(emit_logs, "invalid feature arc: 0x%x", _arc); + return -EINVAL; + } + + /* validate index */ + if (index >= arc->max_indexes) { + FEAT_COND_ERR(emit_logs, "%s: Invalid provided index: %u >= %u configured", + arc->feature_arc_name, index, arc->max_indexes); + return -1; + } + + /* validate feature_name is already added or not */ + if (nodeinfo_lkup_by_name(arc, feature_name, &finfo, &slot)) { + FEAT_COND_ERR(emit_logs, "%s: No feature %s added", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + if (!finfo) { + FEAT_COND_ERR(emit_logs, "%s: No feature: %s found to enable/disable", + arc->feature_arc_name, feature_name); + return -EINVAL; + } + + /* slot should be in valid range */ + if (slot >= arc->max_features) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid free slot %u(max=%u) for feature", + arc->feature_arc_name, feature_name, slot, arc->max_features); + return -EINVAL; + } + + /* slot should be in range of 0 - 63 */ + if (slot > (GRAPH_FEATURE_MAX_NUM_PER_ARC - 1)) { + FEAT_COND_ERR(emit_logs, "%s/%s: Invalid slot: %u", arc->feature_arc_name, + feature_name, slot); + return -EINVAL; + } + + last_end_feature = rte_fls_u64(arc->feature_bit_mask_by_index[index]); + /* if enabled feature is not end feature node */ + if (is_enable_disable && + (arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot)) && + (RTE_BIT64(slot) != last_end_feature)) { + FEAT_COND_ERR(emit_logs, "%s: %s already enabled on index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + if (!is_enable_disable && !arc->runtime_enabled_features) { + FEAT_COND_ERR(emit_logs, "%s: No feature enabled to disable", + arc->feature_arc_name); + return -1; + } + + if (!is_enable_disable && !(arc->feature_bit_mask_by_index[index] & RTE_BIT64(slot))) { + FEAT_COND_ERR(emit_logs, "%s: %s not enabled in bitmask for index: %u", + arc->feature_arc_name, feature_name, index); + return -1; + } + + /* If no feature has been enabled, avoid extra sanity checks */ + if (!arc->runtime_enabled_features) + return 0; + + if (finfo->finfo_index != slot) { + FEAT_COND_ERR(emit_logs, + "%s/%s: lookup slot mismatch for finfo idx: %u and lookup slot: %u", + arc->feature_arc_name, feature_name, finfo->finfo_index, slot); + return -1; + } + + return 0; +} + +static int +refill_fastpath_data(struct rte_graph_feature_arc *arc, uint32_t feature_bit, + uint16_t index /* array index */, int is_enable_disable) +{ + struct rte_graph_feature_node_list *finfo = NULL, *prev_finfo = NULL; + struct rte_graph_feature_data *gfd = NULL, *prev_gfd = NULL; + RTE_ATOMIC(rte_graph_feature_t) * first_feat = NULL; + uint64_t bitmask = 0, prev_bitmask, next_bitmask; + uint32_t fi = 0, prev_fi = 0, next_fi = 0, ffi = 0; + rte_edge_t edge = UINT16_MAX; + + if (is_enable_disable) + bitmask = RTE_BIT64(feature_bit); + + /* set bit from (feature_bit + 1) to 64th bit */ + next_bitmask = UINT64_MAX << (feature_bit + 1); + + /* set bits from 0 to (feature_bit - 1) */ + prev_bitmask = ((UINT64_MAX & ~next_bitmask) & ~(RTE_BIT64(feature_bit))); + + next_bitmask &= arc->feature_bit_mask_by_index[index]; + prev_bitmask &= arc->feature_bit_mask_by_index[index]; + + /* Set next bit set in next_bitmask */ + if (rte_bsf64_safe(next_bitmask, &next_fi)) + bitmask |= RTE_BIT64(next_fi); + + /* Set prev bit set in prev_bitmask*/ + prev_fi = rte_fls_u64(prev_bitmask); + if (prev_fi) + bitmask |= RTE_BIT64(prev_fi - 1); + + /* for each feature set for index, set fast path data */ + prev_fi = RTE_GRAPH_FEATURE_INVALID; + while (rte_bsf64_safe(bitmask, &fi)) { + gfd = rte_graph_feature_data_get(arc, fdata_from_feat(arc, fi, index)); + + RTE_VERIFY(!nodeinfo_lkup_by_index(arc, fi, &finfo, 1)); + + /* Reset next edge to point to last feature node so that packet can exit from arc */ + rte_atomic_store_explicit(&gfd->next_edge, finfo->edge_to_last_feature, + rte_memory_order_relaxed); + + /* + * Reset next feature data + */ + rte_atomic_store_explicit(&gfd->next_feature_data, RTE_GRAPH_FEATURE_DATA_INVALID, + rte_memory_order_relaxed); + + /* If previous feature_index was valid in last loop */ + if (prev_fi != RTE_GRAPH_FEATURE_INVALID) { + prev_gfd = rte_graph_feature_data_get(arc, fdata_from_feat(arc, prev_fi, + index)); + + /* + * Get edge of previous feature node connecting + * to this feature node + */ + RTE_VERIFY(!nodeinfo_lkup_by_index(arc, prev_fi, &prev_finfo, 1)); + + if (!get_existing_edge(arc->feature_arc_name, + prev_finfo->feature_node_id, + finfo->feature_node_id, &edge)) { + feat_dbg("\t[%s/index:%2u,cookie:%u]: (%u->%u)%s[%u] = %s", + arc->feature_arc_name, index, + prev_gfd->app_cookie, prev_fi, fi, + rte_node_id_to_name(prev_finfo->feature_node_id), + edge, rte_node_id_to_name(finfo->feature_node_id)); + + /* + * In fastpath, nodes always should call + * rte_graph_feature_data_next_feature_get() to + * get_next_feature data + * + * So next feature data i.e. gfd should have edge going from + * prev_feature to current/next feature node + */ + rte_atomic_store_explicit(&gfd->next_edge, + edge, + rte_memory_order_relaxed); + + /* + * Fill current feature as next enabled + * feature to previous one + */ + rte_atomic_store_explicit(&prev_gfd->next_feature_data, + fdata_from_feat(arc, fi, index), + rte_memory_order_relaxed); + + prev_fi = fi; + } else { + /* Should not fail */ + RTE_VERIFY(0); + } + } + /* On first feature + * 1. Update fdata with next_edge from start_node to feature node + * 2. Update first enabled feature in its index array + */ + if (rte_bsf64_safe(arc->feature_bit_mask_by_index[index], &ffi)) { + /* If fi is first feature */ + if (ffi == fi) { + feat_dbg("\t[%s/index:%2u,cookie:%u]: (->%u)%s[%u]=%s", + arc->feature_arc_name, index, + gfd->app_cookie, fi, + arc->start_node->name, finfo->edge_to_this_feature, + rte_node_id_to_name(finfo->feature_node_id)); + + gfd = rte_graph_feature_data_get(arc, + fdata_from_feat(arc, fi, index)); + + /* add next edge into feature data + * First set feature data then first feature memory + */ + rte_atomic_store_explicit(&gfd->next_edge, + finfo->edge_to_this_feature, + rte_memory_order_relaxed); + + first_feat = graph_first_feature_ptr_get(arc, index); + + rte_atomic_store_explicit(first_feat, fi, + rte_memory_order_relaxed); + } + + prev_fi = fi; + } + /* Clear current feature index */ + bitmask &= ~RTE_BIT64(fi); + } + + return 0; +} + +int +rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, uint32_t app_cookie, + struct rte_rcu_qsbr *qsbr) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + struct rte_graph_feature_data *gfd = NULL; + uint64_t bitmask; + uint32_t slot; + + if (!arc) { + graph_err("Invalid feature arc: 0x%x", _arc); + return -1; + } + + feat_dbg("%s: Enabling feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if ((!arc->runtime_enabled_features && + (prepare_feature_arc_before_first_enable(arc) < 0))) + return -1; + + if (feature_enable_disable_validate(_arc, index, feature_name, 1 /* enable */, true)) + return -1; + + /** This should not fail as validate() has passed */ + if (nodeinfo_lkup_by_name(arc, feature_name, &finfo, &slot)) + RTE_VERIFY(0); + + gfd = rte_graph_feature_data_get(arc, fdata_from_feat(arc, slot, index)); + + /* Set current app_cookie */ + rte_atomic_store_explicit(&gfd->app_cookie, app_cookie, rte_memory_order_relaxed); + + /* Set bitmask in control path bitmask */ + rte_bit_relaxed_set64(graph_uint_cast(slot), &arc->feature_bit_mask_by_index[index]); + + refill_fastpath_data(arc, slot, index, 1 /* enable */); + + /* On very first feature enable instance */ + if (!finfo->ref_count) { + /* If first time feature getting enabled + */ + bitmask = rte_atomic_load_explicit(&arc->fp_feature_enable_bitmask, + rte_memory_order_relaxed); + + bitmask |= RTE_BIT64(slot); + + rte_atomic_store_explicit(&arc->fp_feature_enable_bitmask, + bitmask, rte_memory_order_relaxed); + } + + /* Slow path updates */ + arc->runtime_enabled_features++; + + /* Increase feature node info reference count */ + finfo->ref_count++; + + if (qsbr) + rte_rcu_qsbr_synchronize(qsbr, RTE_QSBR_THRID_INVALID); + + if (finfo->notifier_cb) + finfo->notifier_cb(arc->feature_arc_name, finfo->feature_name, index, + true /* enable */, gfd->app_cookie); + + return 0; +} + +int +rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, const char *feature_name, + struct rte_rcu_qsbr *qsbr) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_data *gfd = NULL, *dummy_gfd = NULL; + struct rte_graph_feature_node_list *finfo = NULL; + rte_graph_feature_data_t dummy_fdata; + uint32_t slot, last_end_feature; + uint64_t bitmask; + + if (!arc) { + graph_err("Invalid feature arc: 0x%x", _arc); + return -1; + } + feat_dbg("%s: Disable feature: %s for index: %u", + arc->feature_arc_name, feature_name, index); + + if (feature_enable_disable_validate(_arc, index, feature_name, 0, true)) + return -1; + + if (nodeinfo_lkup_by_name(arc, feature_name, &finfo, &slot)) + return -1; + + /* If feature is not last feature, unset in control plane bitmask */ + last_end_feature = rte_fls_u64(arc->feature_bit_mask_by_index[index]); + if (RTE_BIT64(slot) != last_end_feature) + rte_bit_relaxed_clear64(graph_uint_cast(slot), + &arc->feature_bit_mask_by_index[index]); + + /* we have allocated one extra feature data space. Get dummy feature data */ + dummy_fdata = fdata_from_feat(arc, last_end_feature + 1, index); + dummy_gfd = __rte_graph_feature_data_get(arc, dummy_fdata); + gfd = rte_graph_feature_data_get(arc, fdata_from_feat(arc, slot, index)); + + /* + * Packets may have reached to feature node which is getting disabled. + * We want to steer those packets to last feature node so that they can + * exit arc + * - First, reset next_edge of dummy feature data to point to last_feature_node + * - Secondly, reset next_feature_data of current feature getting disabled to dummy + * feature data + */ + rte_atomic_store_explicit(&dummy_gfd->next_edge, finfo->edge_to_last_feature, + rte_memory_order_relaxed); + rte_atomic_store_explicit(&gfd->next_feature_data, dummy_fdata, + rte_memory_order_relaxed); + rte_atomic_store_explicit(&dummy_gfd->next_feature_data, RTE_GRAPH_FEATURE_DATA_INVALID, + rte_memory_order_relaxed); + + /* Now we can unwire fast path*/ + refill_fastpath_data(arc, slot, index, 0 /* disable */); + + finfo->ref_count--; + + /* When last feature is disabled */ + if (!finfo->ref_count) { + /* If no feature enabled, reset feature in u64 fast path bitmask */ + bitmask = rte_atomic_load_explicit(&arc->fp_feature_enable_bitmask, + rte_memory_order_relaxed); + bitmask &= ~(RTE_BIT64(slot)); + rte_atomic_store_explicit(&arc->fp_feature_enable_bitmask, bitmask, + rte_memory_order_relaxed); + } + + if (qsbr) + rte_rcu_qsbr_synchronize(qsbr, RTE_QSBR_THRID_INVALID); + + /* Reset current gfd after rcu synchronization */ + rte_atomic_store_explicit(&gfd->next_edge, RTE_GRAPH_FEATURE_DATA_INVALID, + rte_memory_order_relaxed); + + /* Call notifier cb with valid app_cookie */ + if (finfo->notifier_cb) + finfo->notifier_cb(arc->feature_arc_name, finfo->feature_name, index, + false /* disable */, gfd->app_cookie); + + /* Reset app_cookie later after calling notifier_cb */ + rte_atomic_store_explicit(&gfd->app_cookie, UINT32_MAX, rte_memory_order_relaxed); + + arc->runtime_enabled_features--; + + return 0; +} + +int +rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_node_list *node_info = NULL; + int iter; + + if (!arc) { + graph_err("Invalid feature arc: 0x%x", _arc); + return -1; + } + arc->process_ref_count--; + + if (arc->process_ref_count) + return arc->process_ref_count; + + while (!STAILQ_EMPTY(&arc->all_features)) { + node_info = STAILQ_FIRST(&arc->all_features); + STAILQ_REMOVE_HEAD(&arc->all_features, next_feature); + if (node_info->notifier_cb) { + for (iter = 0; iter < arc->max_indexes; iter++) { + if (!(arc->feature_bit_mask_by_index[iter] & + RTE_BIT64(node_info->finfo_index))) + continue; + + node_info->notifier_cb(arc->feature_arc_name, + node_info->feature_name, + 0, false /* disable */, 0); + } + } + rte_free(node_info); + } + + dm->feature_arcs[arc->feature_arc_index] = GRAPH_FEATURE_ARC_INITIALIZER; + + rte_free(arc->feature_bit_mask_by_index); + + rte_memzone_free(rte_memzone_lookup(arc->feature_arc_name)); + + return 0; +} + +int +rte_graph_feature_arc_cleanup(void) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + int dont_free = 0; + uint32_t iter; + + if (!__rte_graph_feature_arc_main) + return -1; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + if (dm->feature_arcs[iter] == GRAPH_FEATURE_ARC_INITIALIZER) + continue; + + if (rte_graph_feature_arc_destroy(dm->feature_arcs[iter]) > 0) + dont_free++; + } + if (!dont_free) { + rte_memzone_free(rte_memzone_lookup(FEATURE_ARC_MEMZONE_NAME)); + __rte_graph_feature_arc_main = NULL; + } + + return 0; +} + +int +rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc) +{ + struct rte_graph_feature_arc *arc = NULL; + const struct rte_memzone *mz = NULL; + rte_graph_feature_arc_main_t *dm; + uint32_t iter; + + if (_arc) + *_arc = RTE_GRAPH_FEATURE_ARC_INITIALIZER; + + if (!__rte_graph_feature_arc_main) { + mz = rte_memzone_lookup(FEATURE_ARC_MEMZONE_NAME); + if (mz) + __rte_graph_feature_arc_main = mz->addr; + else + return -1; + } + + dm = __rte_graph_feature_arc_main; + + for (iter = 0; iter < dm->max_feature_arcs; iter++) { + arc = rte_graph_feature_arc_get(iter); + + if (!arc) + continue; + + if ((strstr(arc->feature_arc_name, arc_name)) && + (strlen(arc->feature_arc_name) == strlen(arc_name))) { + if (_arc) + *_arc = arc->feature_arc_index; + return 0; + } + } + + return -1; +} + +uint32_t +rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + + if (!arc) { + graph_err("Invalid feature arc: 0x%x", _arc); + return 0; + } + + return arc->runtime_enabled_features; +} + +uint32_t +rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t count = 0; + + if (!arc) { + graph_err("Invalid feature arc: 0x%x", _arc); + return 0; + } + + STAILQ_FOREACH(finfo, &arc->all_features, next_feature) + count++; + + return count; +} + +char * +rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (!arc) + return NULL; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return NULL; + } + if (!nodeinfo_lkup_by_index(arc, slot, &finfo, 0/* ignore sanity*/)) + return finfo->feature_name; + + return NULL; +} + +int +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, rte_graph_feature_t feat, + rte_node_t *node) +{ + struct rte_graph_feature_arc *arc = rte_graph_feature_arc_get(_arc); + struct rte_graph_feature_node_list *finfo = NULL; + uint32_t slot = feat; + + if (!arc) + return -1; + + if (node) + *node = RTE_NODE_ID_INVALID; + + if (feat >= rte_graph_feature_arc_num_features(_arc)) { + graph_err("%s: feature %u does not exist", arc->feature_arc_name, feat); + return -1; + } + if (!nodeinfo_lkup_by_index(arc, slot, &finfo, 0/* ignore sanity*/)) { + if (node) + *node = finfo->feature_node_id; + return 0; + } + return -1; +} + +void __rte_graph_feature_arc_register(struct rte_graph_feature_arc_register *reg, + const char *caller_name, int lineno) +{ + if (!reg) { + FEAT_ERR(caller_name, lineno, "NULL feature arc register"); + return; + } + + if (!reg->arc_name) { + FEAT_ERR(caller_name, lineno, "NULL feature arc name"); + return; + } + + if (!reg->max_indexes) { + FEAT_ERR(caller_name, lineno, "No indexes provided for arc %s", + reg->arc_name); + return; + } + + /* reg->max_features is calculated in rte_graph_feature_arc_init() */ + if (!reg->start_node) { + FEAT_ERR(caller_name, lineno, "No start node provided for arc %s", + reg->arc_name); + return; + } + + if (!reg->start_node_feature_process_fn) { + FEAT_ERR(caller_name, lineno, + "No start node process function provided for arc %s", + reg->arc_name); + return; + } + + if (!reg->end_feature) { + FEAT_ERR(caller_name, lineno, + "No end feature provided for arc %s", reg->arc_name); + return; + } + + /* No need to check end_feature->arc_name as it is implicit */ + if (!reg->end_feature->feature_name) { + FEAT_ERR(caller_name, lineno, + "No end feature provided for arc %s", reg->arc_name); + return; + } + + if (!reg->end_feature->feature_process_fn) { + FEAT_ERR(caller_name, lineno, + "No end feature process function provided for arc %s", + reg->arc_name); + return; + } + + if (!reg->end_feature->arc_name) { + FEAT_ERR(caller_name, lineno, + "No end feature arc name provided for arc %s", reg->arc_name); + return; + } + + STAILQ_INSERT_TAIL(&feature_arc_list, reg, next_arc); +} + +void __rte_graph_feature_register(struct rte_graph_feature_register *reg, + const char *caller_name, int lineno) +{ + if (feature_registration_validate(reg, caller_name, lineno, 0, 0) < 0) + return; + + /* Add to the feature_list*/ + STAILQ_INSERT_TAIL(&feature_list, reg, next_feature); +} + +uint32_t +rte_graph_feature_arc_names_get(char *arc_names[]) +{ + rte_graph_feature_arc_main_t *dm = __rte_graph_feature_arc_main; + struct rte_graph_feature_arc *arc = NULL; + uint32_t count, num_arcs; + + if (!__rte_graph_feature_arc_main) + return 0; + + for (count = 0, num_arcs = 0; count < dm->max_feature_arcs; count++) + if (dm->feature_arcs[count] != GRAPH_FEATURE_ARC_INITIALIZER) + num_arcs++; + + if (!num_arcs) + return 0; + + if (!arc_names) + return sizeof(char *) * num_arcs; + + for (count = 0, num_arcs = 0; count < dm->max_feature_arcs; count++) { + if (dm->feature_arcs[count] != GRAPH_FEATURE_ARC_INITIALIZER) { + arc = rte_graph_feature_arc_get(count); + arc_names[num_arcs] = arc->feature_arc_name; + num_arcs++; + } + } + return num_arcs; +} diff --git a/lib/graph/graph_private.h b/lib/graph/graph_private.h index ceff0c8f50..a94d7c867f 100644 --- a/lib/graph/graph_private.h +++ b/lib/graph/graph_private.h @@ -24,6 +24,10 @@ extern int rte_graph_logtype; RTE_LOG_LINE_PREFIX(level, GRAPH, \ "%s():%u ", __func__ RTE_LOG_COMMA __LINE__, __VA_ARGS__) +#define GRAPH_LOG2(level, _fname, _linenum, ...) \ + RTE_LOG_LINE_PREFIX(level, GRAPH, \ + "%s():%u ", _fname RTE_LOG_COMMA _linenum, __VA_ARGS__) + #define graph_err(...) GRAPH_LOG(ERR, __VA_ARGS__) #define graph_warn(...) GRAPH_LOG(WARNING, __VA_ARGS__) #define graph_info(...) GRAPH_LOG(INFO, __VA_ARGS__) diff --git a/lib/graph/meson.build b/lib/graph/meson.build index 0cb15442ab..5d137d326e 100644 --- a/lib/graph/meson.build +++ b/lib/graph/meson.build @@ -15,14 +15,16 @@ sources = files( 'graph_stats.c', 'graph_populate.c', 'graph_pcap.c', + 'graph_feature_arc.c', 'rte_graph_worker.c', 'rte_graph_model_mcore_dispatch.c', ) headers = files('rte_graph.h', 'rte_graph_worker.h') +headers += files('rte_graph_feature_arc.h', 'rte_graph_feature_arc_worker.h') indirect_headers += files( 'rte_graph_model_mcore_dispatch.h', 'rte_graph_model_rtc.h', 'rte_graph_worker_common.h', ) -deps += ['eal', 'pcapng', 'mempool', 'ring'] +deps += ['eal', 'pcapng', 'mempool', 'ring', 'rcu'] diff --git a/lib/graph/rte_graph_feature_arc.h b/lib/graph/rte_graph_feature_arc.h new file mode 100644 index 0000000000..80ec2f0f19 --- /dev/null +++ b/lib/graph/rte_graph_feature_arc.h @@ -0,0 +1,552 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_H_ +#define _RTE_GRAPH_FEATURE_ARC_H_ + +#include <assert.h> +#include <errno.h> +#include <signal.h> +#include <stddef.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <rte_common.h> +#include <rte_compat.h> +#include <rte_debug.h> +#include <rte_graph.h> +#include <rte_rcu_qsbr.h> + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @file + * + * rte_graph_feature_arc.h + * + * Define APIs and structures/variables with respect to feature arc + * + * - Feature arc(s) + * - Feature(s) + * + * A feature arc represents an ordered list of features/protocols nodes at a + * given networking layer. It provides a high level abstraction to + * enable/disable feature nodes on a given interface at runtime and steer packets + * across these feature nodes in a generic manner. + * + * A feature arc in a graph is represented via *start_node* and *end_node*. + * Feature nodes are added between start_node and end_node. Packets steering + * from start_node to feature nodes are controlled via + * rte_graph_feature_enable()/rte_graph_feature_disable(). + * + * In a typical network stack, often a protocol or feature must be first + * enabled on a given interface, before any packet is steered towards it for + * feature processing. For eg: incoming IPv4 packets are sent to + * routing sub-system only after a valid IPv4 address is assigned to the + * received interface. In other words, often packets needs to be steered across + * features not based on the packet content but based on whether a feature is + * enable or disable on a given incoming/outgoing interface. Feature arc + * provides mechanism to enable/disable feature(s) on each interface at runtime + * and allow seamless packet steering across runtime enabled feature nodes in + * fast path. + * + * Feature arc also provides a way to steer packets from in-built nodes to + * out-of-tree *feature nodes* without any change in in-built node's + * fast path functions + * + * On a given interface multiple feature(s) might be enabled in a particular + * feature arc. For instance, both "ipv4-output" and "IPsec policy output" + * features may be enabled on "eth0" interface in "L3-output" feature arc. + * Similarly, "ipv6-output" and "ipsec-output" may be enabled on "eth1" + * interface in same "L3-output" feature arc. + * + * When multiple features are present in a given feature arc, its imperative + * to allow each feature processing in a particular sequential order. For + * instance, in "L3-input" feature arc it may be required to run "IPsec + * input" feature first, for packet decryption, before "ip-lookup". So a + * sequential order must be maintained among features present in a feature arc. + * + * Features are enabled/disabled multiple times at runtime to some or all + * available interfaces present in the system. Enable/disabling features on one + * interface is independent of other interface. + * + * A given feature might consume packet (if it's configured to consume) or may + * forward it to next enabled feature. For instance, "IPsec input" feature may + * consume/drop all packets with "Protect" policy action while all packets with + * policy action as "Bypass" may be forwarded to next enabled feature (with in + * same feature arc) + * + * This library facilitates rte graph based applications to steer packets in + * fast path to different feature nodes with-in a feature arc and support all + * functionalities described above + * + * In order to use feature-arc APIs, applications needs to do following in + * control path: + * - Create feature arc's using RTE_GRAPH_FEATURE_ARC_REGISTER() + * - New features can be added to an arc via RTE_GRAPH_FEATURE_REGISTER() + * - Before calling rte_graph_create(), rte_graph_feature_arc_init() API must + * be called. If rte_graph_feature_arc_init() is not called by application, + * feature arc library is NOP + * - Features can be enabled/disabled on any interface via + * rte_graph_feature_enable()/rte_graph_feature_disable() + * - Feature arc can be destroyed via rte_graph_feature_arc_destroy() + * + * In fast path, APIs are provided to steer packets towards feature path from + * - start_node (@ref RTE_GRAPH_FEATURE_ARC_REGISTER()) + * - feature nodes added via RTE_GRAPH_FEATURE_REGISTER() + * + * For typical steering of packets across feature nodes, application required + * to know "rte_edges" which are saved in feature data object. Feature data + * object is unique for every interface per feature with in a feature arc. + * + * APIs used to steer packets from start_node to first enabled feature node are: + * - rte_graph_feature_data_first_feature_get(). Once valid feature data is + * returned, application can use + * - rte_graph_feature_data_edge_get() to get edge from start_node to first + * feature + * + * rte_mbuf can carry [feature_data] into feature arc specific mbuf dynamic + * field rte_graph_feature_arc_mbuf_dynfield_offset_get() + * + * APIs used to steer packets from one feature node to next enabled feature + * node + * - rte_graph_feature_data_app_cookie_get() to get application specific data + * set by application in rte_graph_feature_enable() + * - rte_graph_feature_data_edge_get() to get edge from current node to next + * feature node + * - mbuf->dynfield[feature_data] needs to be updated with new feature data + * via rte_graph_feature_data_next_feature_get() + * + * Fast path synchronization + * ------------------------- + * Any feature enable/disable in control plane does not require stopping of + * worker cores. + * + * rte_graph_feature_enable()/rte_graph_feature_disable() APIs accepts + * (rte_rcu_qsbr *) as an argument to allow application releasing + * resources associated with features it may have allocated for per feature + * per interface. + * + * After every successful enable/disable, API internally calls + * - rte_rcu_qsbr_synchronize(rte_rcu_qsbr *) to synchronize all worker cores + * - Calls RTE_GRAPH_FEATURE_REGISTER()->notifier_cb() with app_cookie, + * provided per feature per interface in rte_graph_feature_enable() + */ + +/** Length of feature arc name */ +#define RTE_GRAPH_FEATURE_ARC_NAMELEN RTE_NODE_NAMESIZE + +/** Initializer values for ARC, Feature, Feature data */ +#define RTE_GRAPH_FEATURE_ARC_INITIALIZER ((rte_graph_feature_arc_t)UINT16_MAX) +#define RTE_GRAPH_FEATURE_DATA_INVALID ((rte_graph_feature_data_t)UINT16_MAX) +#define RTE_GRAPH_FEATURE_INVALID ((rte_graph_feature_t)UINT8_MAX) + +/** rte_graph feature arc object */ +typedef uint16_t rte_graph_feature_arc_t; + +/** rte_graph feature object */ +typedef uint8_t rte_graph_feature_t; + +/** rte_graph feature data object */ +typedef uint16_t rte_graph_feature_data_t; + +/** feature notifier callback called when feature is enabled/disabled */ +typedef void (*rte_graph_feature_change_notifier_cb_t)(const char *arc_name, + const char *feature_name, + uint16_t index, + bool enable_disable, + uint32_t app_cookie); + +/** + * Feature registration structure provided to + * RTE_GRAPH_FEATURE_REGISTER() + */ +struct rte_graph_feature_register { + STAILQ_ENTRY(rte_graph_feature_register) next_feature; + + /** Name of the arc which is registered either via + * RTE_GRAPH_FEATURE_ARC_REGISTER() or via + * rte_graph_feature_arc_create() + */ + const char *arc_name; + + /* Name of the feature */ + const char *feature_name; + + /** + * Node id of feature_node. + * + * Setting this field can be skipped if registering feature via + * RTE_GRAPH_FEATURE_REGISTER() + */ + rte_node_t feature_node_id; + + /** + * Feature node process() function calling feature fast path APIs. + * + * If application calls rte_graph_feature_arc_init(), node->process() + * provided in RTE_NODE_REGISTER() is overwritten by this + * function. + */ + rte_node_process_t feature_process_fn; + + /* + * Pointer to Feature node registration + * + * Used when features are registered via + * RTE_GRAPH_FEATURE_REGISTER(). + */ + struct rte_node_register *feature_node; + + /** Feature ordering constraints + * runs_after: Name of the feature which must run before "this feature" + * runs_before: Name of the feature which must run after "this feature" + */ + const char *runs_after; + const char *runs_before; + + /** + * Callback for notifying any change in feature enable/disable state + */ + rte_graph_feature_change_notifier_cb_t notifier_cb; +}; + +/** Feature arc registration structure */ +struct rte_graph_feature_arc_register { + STAILQ_ENTRY(rte_graph_feature_arc_register) next_arc; + + /** Name of the feature arc */ + const char *arc_name; + + /** + * Maximum number of features supported in this feature arc. + * + * This field can be skipped for feature arc registration via + * RTE_GRAPH_FEATURE_ARC_REGISTER(). + * + * API internally sets this field by calculating number of + * RTE_GRAPH_FEATURE_REGISTER() for every arc registration via + * RTE_GRAPH_FEATURE_ARC_REGISTER() + */ + uint32_t max_features; + + /** + * Maximum number of indexes supported in this feature arc + * + * Typically number of interfaces or ethdevs (For eg: RTE_MAX_ETHPORTS) + */ + uint32_t max_indexes; + + /** Start node of this arc */ + struct rte_node_register *start_node; + + /** + * Feature arc specific process() function for Start node. + * If application calls rte_graph_feature_arc_init(), + * start_node->process() is replaced by this function + */ + rte_node_process_t start_node_feature_process_fn; + + /** End feature node registration */ + struct rte_graph_feature_register *end_feature; +}; + +/** constructor to register feature to an arc */ +#define RTE_GRAPH_FEATURE_REGISTER(reg) \ + RTE_INIT(__rte_graph_feature_register_##reg) \ + { \ + __rte_graph_feature_register(®, __func__, __LINE__); \ + } + +/** constructor to register a feature arc */ +#define RTE_GRAPH_FEATURE_ARC_REGISTER(reg) \ + RTE_INIT(__rte_graph_feature_arc_register_##reg) \ + { \ + __rte_graph_feature_arc_register(®, __func__, __LINE__); \ + } +/** + * Initialize feature arc subsystem + * + * This API + * - Initializes feature arc module and alloc associated memory + * - creates feature arc for every RTE_GRAPH_FEATURE_ARC_REGISTER() + * - Add feature node to a feature arc for every RTE_GRAPH_FEATURE_REGISTER() + * - Replaces all RTE_NODE_REGISTER()->process() functions for + * - Every start_node/end_node provided in arc registration + * - Every feature node provided in feature registration + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_init(void); + +/** + * Create a feature arc. + * + * This API can be skipped if RTE_GRAPH_FEATURE_ARC_REGISTER() is used + * + * @param reg + * Pointer to struct rte_graph_feature_arc_register + * @param[out] _arc + * Feature arc object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_create(struct rte_graph_feature_arc_register *reg, + rte_graph_feature_arc_t *_arc); + +/** + * Get feature arc object with name + * + * @param arc_name + * Feature arc name provided to successful @ref rte_graph_feature_arc_create + * @param[out] _arc + * Feature arc object returned. Valid only when API returns SUCCESS + * + * @return + * 0: Success + * <0: Failure. + */ +__rte_experimental +int rte_graph_feature_arc_lookup_by_name(const char *arc_name, rte_graph_feature_arc_t *_arc); + +/** + * Add a feature to already created feature arc. + * + * This API is not required in case RTE_GRAPH_FEATURE_REGISTER() is used + * + * @param feat_reg + * Pointer to struct rte_graph_feature_register + * + * <I> Must be called before rte_graph_create() </I> + * <I> rte_graph_feature_add() is not allowed after call to + * rte_graph_feature_enable() so all features must be added before they can be + * enabled </I> + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_add(struct rte_graph_feature_register *feat_reg); + +/** + * Enable feature within a feature arc + * + * Must be called after @b rte_graph_create(). + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param app_cookie + * Application specific data which is retrieved in fast path + * @param qsbr + * RCU QSBR object. After enabling feature, API calls + * rte_rcu_qsbr_synchronize() followed by call to struct + * rte_graph_feature_register::notifier_cb(), if it is set, to notify feature + * caller This object can be passed NULL as well if no RCU synchronization is + * required + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_enable(rte_graph_feature_arc_t _arc, uint32_t index, const + char *feature_name, uint32_t app_cookie, + struct rte_rcu_qsbr *qsbr); + +/** + * Disable already enabled feature within a feature arc + * + * Must be called after @b rte_graph_create(). API is *NOT* Thread-safe + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param index + * Application specific index. Can be corresponding to interface_id/port_id etc + * @param feature_name + * Name of the node which is already added via @ref rte_graph_feature_add + * @param qsbr + * RCU QSBR object. After disabling feature, API calls + * rte_rcu_qsbr_synchronize() followed by call to struct + * rte_graph_feature_register::notifier_cb(), if it is set, to notify feature + * caller This object can be passed NULL as well if no RCU synchronization is + * required + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_disable(rte_graph_feature_arc_t _arc, uint32_t index, + const char *feature_name, struct rte_rcu_qsbr *qsbr); + +/** + * Get rte_graph_feature_t object from feature name + * + * @param arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * @param feature_name + * Feature name provided to @ref rte_graph_feature_add + * @param[out] feature + * Feature object + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_lookup(rte_graph_feature_arc_t arc, const char *feature_name, + rte_graph_feature_t *feature); + +/** + * Delete feature_arc object + * + * @param _arc + * Feature arc object returned by @ref rte_graph_feature_arc_create or @ref + * rte_graph_feature_arc_lookup_by_name + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_destroy(rte_graph_feature_arc_t _arc); + +/** + * Cleanup all feature arcs + * + * @return + * 0: Success + * <0: Failure + */ +__rte_experimental +int rte_graph_feature_arc_cleanup(void); + +/** + * Slow path API to know how many features are added (NOT enabled) within a + * feature arc + * + * @param _arc + * Feature arc object + * + * @return: Number of added features to arc + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to know how many features are currently enabled within a + * feature arc across all indexes. If a single feature is enabled on all interfaces, + * this API would return "number_of_interfaces" as count (but not "1") + * + * @param _arc + * Feature arc object + * + * @return: Number of enabled features across all indexes + */ +__rte_experimental +uint32_t rte_graph_feature_arc_num_enabled_features(rte_graph_feature_arc_t _arc); + +/** + * Slow path API to get feature node name from rte_graph_feature_t object + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * + * @return: Name of the feature node + */ +__rte_experimental +char *rte_graph_feature_arc_feature_to_name(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature); + +/** + * Slow path API to get corresponding rte_node_t from + * rte_graph_feature_t + * + * @param _arc + * Feature arc object + * @param feature + * Feature object + * @param[out] node + * rte_node_t of feature node, Valid only when API returns SUCCESS + * + * @return: 0 on success, < 0 on failure + */ +__rte_experimental +int +rte_graph_feature_arc_feature_to_node(rte_graph_feature_arc_t _arc, + rte_graph_feature_t feature, + rte_node_t *node); + +/** + * Slow path API to dump valid feature arc names + * + * @param[out] arc_names + * Buffer to copy the arc names. The NULL value is allowed in that case, + * the function returns the size of the array that needs to be allocated. + * + * @return + * When next_nodes == NULL, it returns the size of the array else + * number of item copied. + */ +__rte_experimental +uint32_t +rte_graph_feature_arc_names_get(char *arc_names[]); + +/** + * @internal + * + * function declaration for registering arc + * + * @param reg + * Pointer to struct rte_graph_feature_arc_register + * @param caller_name + * Name of the function which is calling this API + * @param lineno + * Line number of the function which is calling this API + */ +__rte_experimental +void __rte_graph_feature_arc_register(struct rte_graph_feature_arc_register *reg, + const char *caller_name, int lineno); + +/** + * @internal + * + * function declaration for registering feature + * + * @param reg + * Pointer to struct rte_graph_feature_register + * @param caller_name + * Name of the function which is calling this API + * @param lineno + * Line number of the function which is calling this API + */ +__rte_experimental +void __rte_graph_feature_register(struct rte_graph_feature_register *reg, + const char *caller_name, int lineno); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/lib/graph/rte_graph_feature_arc_worker.h b/lib/graph/rte_graph_feature_arc_worker.h new file mode 100644 index 0000000000..4ffd51091b --- /dev/null +++ b/lib/graph/rte_graph_feature_arc_worker.h @@ -0,0 +1,608 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#ifndef _RTE_GRAPH_FEATURE_ARC_WORKER_H_ +#define _RTE_GRAPH_FEATURE_ARC_WORKER_H_ + +#include <stddef.h> +#include <rte_graph_feature_arc.h> +#include <rte_bitops.h> +#include <rte_mbuf.h> +#include <rte_mbuf_dyn.h> + +/** + * @file + * + * rte_graph_feature_arc_worker.h + * + * Defines fast path structure for feature arc + */ + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @internal + * + * Slow path feature node info list + */ +struct rte_graph_feature_node_list { + /** Next feature */ + STAILQ_ENTRY(rte_graph_feature_node_list) next_feature; + + char feature_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** node id representing feature */ + rte_node_t feature_node_id; + + /** How many indexes/interfaces using this feature */ + int32_t ref_count; + + /** + * feature arc process function overrides to feature node's original + * process function + */ + rte_node_process_t feature_node_process_fn; + + /** Callback for freeing application resources when */ + rte_graph_feature_change_notifier_cb_t notifier_cb; + + /* finfo_index in list. same as rte_graph_feature_t */ + uint32_t finfo_index; + + /** Back pointer to feature arc */ + void *feature_arc; + + /** rte_edge_t to this feature node from feature_arc->start_node */ + rte_edge_t edge_to_this_feature; + + /* rte_edge_t from this feature node to last feature node */ + rte_edge_t edge_to_last_feature; +}; + +/** + * rte_graph Feature arc object + * + * Feature arc object holds control plane and fast path information for all + * features and all interface index information for steering packets across + * feature nodes + * + * Within a feature arc, only RTE_GRAPH_FEATURE_MAX_PER_ARC features can be + * added. If more features needs to be added, another feature arc can be + * created + * + * In fast path, rte_graph_feature_arc_t can be translated to (struct + * rte_graph_feature_arc *) via rte_graph_feature_arc_get(). Later is needed to + * add as an input argument to all fast path feature arc APIs + */ +struct __rte_cache_aligned rte_graph_feature_arc { + /** Slow path variables follows*/ + RTE_MARKER slow_path_variables; + + /** All feature lists */ + STAILQ_HEAD(, rte_graph_feature_node_list) all_features; + + /** feature arc name */ + char feature_arc_name[RTE_GRAPH_FEATURE_ARC_NAMELEN]; + + /** control plane counter to track enabled features */ + uint32_t runtime_enabled_features; + + /** index in feature_arc_main */ + uint16_t feature_arc_index; + + /* process ref count to track feature_arc_destroy() */ + uint8_t process_ref_count; + + /** Back pointer to feature_arc_main */ + void *feature_arc_main; + + /** Arc's start/end node */ + struct rte_node_register *start_node; + struct rte_graph_feature_register end_feature; + + /* arc start process function */ + rte_node_process_t arc_start_process; + + /* total arc_size allocated */ + size_t arc_size; + + /** Slow path bit mask per feature per index */ + uint64_t *feature_bit_mask_by_index; + + /** Cache aligned fast path variables */ + alignas(RTE_CACHE_LINE_SIZE) RTE_MARKER fast_path_variables; + + /** + * Quick fast path bitmask indicating if any feature enabled. Each bit + * corresponds to single feature Helps in optimally process packets for + * the case when features are added but not enabled + */ + RTE_ATOMIC(uint64_t) fp_feature_enable_bitmask; + + /** maximum number of features supported by this arc + * Immutable during fast path + */ + uint16_t max_features; + + /** maximum number of index supported by this arc + * Immutable during fast path + */ + uint16_t max_indexes; + + /** arc + fp_first_feature_arr_offset + * Immutable during fast path + */ + uint16_t fp_first_feature_offset; + + /** arc + fp_feature_data_arr_offset + * Immutable during fast path + */ + uint16_t fp_feature_data_offset; + + /** + * Size of each feature in fastpath. + * ALIGN(sizeof(struct rte_graph_feature_data) * arc->max_indexes) + * Immutable during fast path + */ + uint32_t fp_feature_size; + + /** + * Arc specific fast path data + * It accommodates: + * + * 1. first enabled feature for every index + * rte_graph_feature_t (fdata as shown below) + * + * +-------------------------+ <- cache_aligned + * | 0th Index | 1st Index | + * +-------------------------+ + * | feature0 | feature1 | + * +-------------------------+ + * + * 2. struct rte_graph_feature_data per index per feature + * + * feature0-> +----------------------------------------+ ^ <- cache_aligned + * | struct rte_graph_feature_data[Index0] | | + * +----------------------------------------+ | fp_feature_size + * | struct rte_graph_feature_data[Index1] | | + * feature1-> +----------------------------------------+ v <- cache aligned + * | struct rte_graph_feature_data[Index0] | ^ + * +----------------------------------------+ | fp_feature_size + * | struct rte_graph_feature_data[Index1] | | + * +----------------------------------------+ v + * ... .... + * ... .... + */ + RTE_MARKER8 fp_arc_data; +}; + +/** + * Feature arc main object + * + * Holds all feature arcs created by application + */ +typedef struct rte_feature_arc_main { + /** number of feature arcs created by application */ + uint32_t num_feature_arcs; + + /** max features arcs allowed */ + uint32_t max_feature_arcs; + + /** Pointer to all feature arcs */ + uintptr_t feature_arcs[]; +} rte_graph_feature_arc_main_t; + +/** + * Fast path feature data object + * + * Used by fast path inline feature arc APIs + * Corresponding to rte_graph_feature_data_t + * It holds + * - edge to reach to next feature node + * - next_feature_data corresponding to next enabled feature + * - app_cookie set by application in rte_graph_feature_enable() + */ +struct rte_graph_feature_data { + /** edge from previous enabled feature to this enabled feature */ + RTE_ATOMIC(rte_edge_t) next_edge; + + /** Next feature data from this feature data */ + RTE_ATOMIC(rte_graph_feature_data_t) next_feature_data; + + /** + * app_cookie set by application in rte_graph_feature_enable() for + * - current feature + * - interface index + */ + RTE_ATOMIC(uint32_t) app_cookie; +}; + +/** feature arc specific mbuf dynfield structure. */ +struct rte_graph_feature_arc_mbuf_dynfields { + /** each mbuf carries feature data */ + rte_graph_feature_data_t feature_data; +}; + +/** Name of dynamic mbuf field offset registered in rte_graph_feature_arc_init() */ +#define RTE_GRAPH_FEATURE_ARC_DYNFIELD_NAME "__rte_graph_feature_arc_mbuf_dynfield" + +/** log2(sizeof (struct rte_graph_feature_data)) */ +#define RTE_GRAPH_FEATURE_DATA_SIZE_LOG2 3 + +/** Number of struct rte_graph_feature_data per feature*/ +#define RTE_GRAPH_FEATURE_DATA_NUM_PER_FEATURE(arc) \ + (arc->fp_feature_size >> RTE_GRAPH_FEATURE_DATA_SIZE_LOG2) + +/** Get rte_graph_feature_data_t from rte_graph_feature_t */ +#define RTE_GRAPH_FEATURE_TO_FEATURE_DATA(arc, feature, index) \ + ((rte_graph_feature_data_t) \ + ((RTE_GRAPH_FEATURE_DATA_NUM_PER_FEATURE(arc) * feature) + index)) + +/** extern variables */ +extern rte_graph_feature_arc_main_t *__rte_graph_feature_arc_main; +extern int __rte_graph_feature_arc_mbuf_dyn_offset; + +/** get feature arc dynamic offset + * + * @return + * offset to feature arc specific fields in mbuf + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_mbuf_dynfield_offset_get(void) +{ + return __rte_graph_feature_arc_mbuf_dyn_offset; +} + +/** + * Get dynfield offset to feature arc specific fields in mbuf + * + * @param mbuf + * Pointer to packet + * @param dyn_off + * offset to feature arc specific fields in mbuf + * + * @return + * NULL: On Failure + * Non-NULL pointer on Success + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature_arc_mbuf_dynfields * +rte_graph_feature_arc_mbuf_dynfields_get(struct rte_mbuf *mbuf, const int dyn_off) +{ + return RTE_MBUF_DYNFIELD(mbuf, dyn_off, + struct rte_graph_feature_arc_mbuf_dynfields *); +} + +/** + * API to know if feature is valid or not + * + * @param feature + * rte_graph_feature_t + * + * @return + * 1: If feature is valid + * 0: If feature is invalid + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_is_valid(rte_graph_feature_t feature) +{ + return (feature != RTE_GRAPH_FEATURE_INVALID); +} + +/** + * API to know if feature data is valid or not + * + * @param feature_data + * rte_graph_feature_data_t + * + * @return + * 1: If feature data is valid + * 0: If feature data is invalid + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_data_is_valid(rte_graph_feature_data_t feature_data) +{ + return (feature_data != RTE_GRAPH_FEATURE_DATA_INVALID); +} + +/** + * Get pointer to feature arc object from rte_graph_feature_arc_t + * + * @param arc + * feature arc + * + * @return + * NULL: On Failure + * Non-NULL pointer on Success + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature_arc * +rte_graph_feature_arc_get(rte_graph_feature_arc_t arc) +{ + rte_graph_feature_arc_main_t *fm = NULL; + + fm = __rte_graph_feature_arc_main; + + if (likely(fm != NULL && arc < fm->max_feature_arcs)) + return (struct rte_graph_feature_arc *)fm->feature_arcs[arc]; + + return NULL; +} + +/** + * Get rte_graph_feature_t from feature arc object without any checks + * + * @param arc + * feature arc + * @param fdata + * feature data object + * + * @return + * Pointer to feature data object + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature_data* +__rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, + rte_graph_feature_data_t fdata) +{ + return ((struct rte_graph_feature_data *) ((uint8_t *)arc + arc->fp_feature_data_offset + + (fdata << RTE_GRAPH_FEATURE_DATA_SIZE_LOG2))); +} + +/** + * Get next edge from feature data pointer, without any check + * + * @param fdata + * feature data object + * + * @return + * next edge + */ +__rte_experimental +static __rte_always_inline rte_edge_t +__rte_graph_feature_data_edge_get(struct rte_graph_feature_data *fdata) +{ + return rte_atomic_load_explicit(&fdata->next_edge, rte_memory_order_relaxed); +} + +/** + * Get app_cookie from feature data pointer, without any check + * + * @param fdata + * feature data object + * + * @return + * app_cookie set by caller in rte_graph_feature_enable() API + */ +__rte_experimental +static __rte_always_inline uint32_t +__rte_graph_feature_data_app_cookie_get(struct rte_graph_feature_data *fdata) +{ + return rte_atomic_load_explicit(&fdata->app_cookie, rte_memory_order_relaxed); +} + +/** + * Get next_enabled_feature_data from pointer to feature data, without any check + * + * @param fdata + * feature data object + * + * @return + * next enabled feature data from this feature data + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t +__rte_graph_feature_data_next_feature_get(struct rte_graph_feature_data *fdata) +{ + return rte_atomic_load_explicit(&fdata->next_feature_data, rte_memory_order_relaxed); +} + +/** + * Get next edge from feature data object with checks + * + * @param arc + * feature arc + * @param fdata + * feature data object + * + * @return + * next edge + */ +__rte_experimental +static __rte_always_inline rte_edge_t +rte_graph_feature_data_edge_get(struct rte_graph_feature_arc *arc, + rte_graph_feature_data_t fdata) +{ + struct rte_graph_feature_data *fdata_obj = __rte_graph_feature_data_get(arc, fdata); + + return __rte_graph_feature_data_edge_get(fdata_obj); +} + +/** + * Get app_cookie from feature data object with checks + * + * @param arc + * feature arc + * @param fdata + * feature data object + * + * @return + * app_cookie set by caller in rte_graph_feature_enable() API + */ +__rte_experimental +static __rte_always_inline uint32_t +rte_graph_feature_data_app_cookie_get(struct rte_graph_feature_arc *arc, + rte_graph_feature_data_t fdata) +{ + struct rte_graph_feature_data *fdata_obj = __rte_graph_feature_data_get(arc, fdata); + + return __rte_graph_feature_data_app_cookie_get(fdata_obj); +} + +/** + * Get next_enabled_feature_data from current feature data object with checks + * + * @param arc + * feature arc + * @param fdata + * feature data object + * + * @return + * next enabled feature data from this feature data + */ +__rte_experimental +static __rte_always_inline rte_graph_feature_data_t +rte_graph_feature_data_next_feature_get(struct rte_graph_feature_arc *arc, + rte_graph_feature_data_t fdata) +{ + struct rte_graph_feature_data *fdata_obj = __rte_graph_feature_data_get(arc, fdata); + + return __rte_graph_feature_data_next_feature_get(fdata_obj); +} + +/** + * Get struct rte_graph_feature_data from rte_graph_feature_dat_t + * + * @param arc + * feature arc + * @param fdata + * feature data object + * + * @return + * NULL: On Failure + * Non-NULL pointer on Success + */ +__rte_experimental +static __rte_always_inline struct rte_graph_feature_data* +rte_graph_feature_data_get(struct rte_graph_feature_arc *arc, + rte_graph_feature_data_t fdata) +{ + if (unlikely(fdata > (RTE_GRAPH_FEATURE_TO_FEATURE_DATA(arc, + arc->max_features - 1, + arc->max_indexes - 1)))) + return NULL; + + return __rte_graph_feature_data_get(arc, fdata); +} + +/** + * Get feature data corresponding to first enabled feature on index + * @param arc + * feature arc + * @param index + * Interface index + * @param[out] fdata + * feature data object + * + * @return + * 1: if any feature enabled on index, return corresponding valid feature data + * 0: if no feature is enabled on index + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_data_first_feature_get(struct rte_graph_feature_arc *arc, + uint32_t index, + rte_graph_feature_data_t *fdata) +{ + rte_graph_feature_t *feature = NULL; + + *fdata = RTE_GRAPH_FEATURE_DATA_INVALID; + + feature = (rte_graph_feature_t *)((uint8_t *)arc + arc->fp_first_feature_offset + + (sizeof(rte_graph_feature_t) * index)); + + if ((index < arc->max_indexes) && rte_graph_feature_is_valid(*feature)) { + *fdata = RTE_GRAPH_FEATURE_TO_FEATURE_DATA(arc, *feature, index); + return 1; + } + + return 0; +} + +/** + * Fast path API to check if any feature enabled on a feature arc + * Typically from arc->start_node process function + * + * @param arc + * Feature arc object + * + * @return + * 0: If no feature enabled + * Non-Zero: Bitmask of features enabled. + * + */ +__rte_experimental +static __rte_always_inline uint64_t +rte_graph_feature_arc_is_any_feature_enabled(struct rte_graph_feature_arc *arc) +{ + return (rte_atomic_load_explicit(&arc->fp_feature_enable_bitmask, + rte_memory_order_relaxed)); +} + +/** + * Fast path API to check if provided feature is enabled on any interface/index + * or not + * + * @param arc + * Feature arc object + * @param feature + * Input rte_graph_feature_t that needs to be checked. Can be retrieved in + * control path via rte_graph_feature_lookup() + * + * @return + * 1: If input [feature] is enabled in arc + * 0: If input [feature] is not enabled in arc + */ +__rte_experimental +static __rte_always_inline int +rte_graph_feature_arc_is_feature_enabled(struct rte_graph_feature_arc *arc, + rte_graph_feature_t feature) +{ + uint64_t bitmask = RTE_BIT64(feature); + + return (bitmask & rte_atomic_load_explicit(&arc->fp_feature_enable_bitmask, + rte_memory_order_relaxed)); +} + +/** + * Prefetch feature arc fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_prefetch(struct rte_graph_feature_arc *arc) +{ + rte_prefetch0((void *)arc->fast_path_variables); +} + +/** + * Prefetch feature data related fast path cache line + * + * @param arc + * RTE_GRAPH feature arc object + * @param fdata + * Pointer to feature data object + */ +__rte_experimental +static __rte_always_inline void +rte_graph_feature_arc_feature_data_prefetch(struct rte_graph_feature_arc *arc, + rte_graph_feature_data_t fdata) +{ + if (unlikely(fdata == RTE_GRAPH_FEATURE_DATA_INVALID)) + return; + + rte_prefetch0((void *)__rte_graph_feature_data_get(arc, fdata)); +} + +#ifdef __cplusplus +} +#endif +#endif diff --git a/lib/graph/version.map b/lib/graph/version.map index 44fadc00fd..4aadce446d 100644 --- a/lib/graph/version.map +++ b/lib/graph/version.map @@ -56,6 +56,26 @@ DPDK_25 { EXPERIMENTAL { global: + # added in 25.03 + __rte_graph_feature_arc_main; + __rte_graph_feature_arc_mbuf_dyn_offset; + __rte_graph_feature_arc_register; + __rte_graph_feature_register; + rte_graph_feature_arc_init; + rte_graph_feature_arc_create; + rte_graph_feature_arc_lookup_by_name; + rte_graph_feature_add; + rte_graph_feature_enable; + rte_graph_feature_disable; + rte_graph_feature_lookup; + rte_graph_feature_arc_destroy; + rte_graph_feature_arc_cleanup; + rte_graph_feature_arc_num_enabled_features; + rte_graph_feature_arc_num_features; + rte_graph_feature_arc_feature_to_name; + rte_graph_feature_arc_feature_to_node; + rte_graph_feature_arc_names_get; + # added in 24.11 rte_node_xstat_increment; }; -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v6 3/4] ip4: add ip4 output feature arc 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 1/4] graph: add API to override node process function Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 2/4] graph: add feature arc abstraction Nitin Saxena @ 2025-01-03 6:06 ` Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 4/4] app/graph: add custom feature nodes for ip4 output arc Nitin Saxena [not found] ` <SJ0PR18MB5111B56B4323FB3DFD147801B6152@SJ0PR18MB5111.namprd18.prod.outlook.com> 4 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2025-01-03 6:06 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena Added ip4 output arc to allow applications to hook feature nodes in ip4 egress direction Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- lib/node/ethdev_ctrl.c | 8 + lib/node/interface_tx_feature.c | 133 ++++++++++++ lib/node/interface_tx_feature_priv.h | 33 +++ lib/node/ip4_rewrite.c | 298 ++++++++++++++++++++++++++- lib/node/meson.build | 1 + lib/node/node_private.h | 1 + lib/node/rte_node_ip4_api.h | 4 + 7 files changed, 474 insertions(+), 4 deletions(-) create mode 100644 lib/node/interface_tx_feature.c create mode 100644 lib/node/interface_tx_feature_priv.h diff --git a/lib/node/ethdev_ctrl.c b/lib/node/ethdev_ctrl.c index cd52e8be08..93ef7fbb95 100644 --- a/lib/node/ethdev_ctrl.c +++ b/lib/node/ethdev_ctrl.c @@ -14,6 +14,7 @@ #include "ethdev_tx_priv.h" #include "ip4_rewrite_priv.h" #include "ip6_rewrite_priv.h" +#include "interface_tx_feature_priv.h" #include "node_private.h" static struct ethdev_ctrl { @@ -24,6 +25,7 @@ int rte_node_eth_config(struct rte_node_ethdev_config *conf, uint16_t nb_confs, uint16_t nb_graphs) { + struct rte_node_register *if_tx_feature_node; struct rte_node_register *ip4_rewrite_node; struct rte_node_register *ip6_rewrite_node; struct ethdev_tx_node_main *tx_node_data; @@ -35,6 +37,7 @@ rte_node_eth_config(struct rte_node_ethdev_config *conf, uint16_t nb_confs, int i, j, rc; uint32_t id; + if_tx_feature_node = if_tx_feature_node_get(); ip4_rewrite_node = ip4_rewrite_node_get(); ip6_rewrite_node = ip6_rewrite_node_get(); tx_node_data = ethdev_tx_node_data_get(); @@ -125,6 +128,11 @@ rte_node_eth_config(struct rte_node_ethdev_config *conf, uint16_t nb_confs, if (rc < 0) return rc; + /* Add this tx port node to if_tx_feature_node */ + rte_node_edge_update(if_tx_feature_node->id, RTE_EDGE_ID_INVALID, + &next_nodes, 1); + rc = if_tx_feature_node_set_next(port_id, + rte_node_edge_count(if_tx_feature_node->id) - 1); } ctrl.nb_graphs = nb_graphs; diff --git a/lib/node/interface_tx_feature.c b/lib/node/interface_tx_feature.c new file mode 100644 index 0000000000..35ac00f21e --- /dev/null +++ b/lib/node/interface_tx_feature.c @@ -0,0 +1,133 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#include <arpa/inet.h> +#include <sys/socket.h> + +#include <rte_ethdev.h> +#include <rte_ether.h> +#include <rte_graph_feature_arc_worker.h> +#include <rte_malloc.h> + +#include "rte_node_ip4_api.h" +#include "node_private.h" +#include "interface_tx_feature_priv.h" + +#define IF_TX_FEATURE_LAST_NEXT_INDEX(ctx) \ + (((struct if_tx_feature_node_ctx *)ctx)->last_index) +/* + * @internal array for mapping port to next node index + */ +struct if_tx_feature_node_main { + uint16_t next_index[RTE_MAX_ETHPORTS]; +}; + +struct if_tx_feature_node_ctx { + uint16_t last_index; +}; + +static struct if_tx_feature_node_main *if_tx_feature_nm; + +int +if_tx_feature_node_set_next(uint16_t port_id, uint16_t next_index) +{ + if (if_tx_feature_nm == NULL) { + if_tx_feature_nm = rte_zmalloc( + "if_tx_feature_nm", sizeof(struct if_tx_feature_node_main), + RTE_CACHE_LINE_SIZE); + if (if_tx_feature_nm == NULL) + return -ENOMEM; + } + if_tx_feature_nm->next_index[port_id] = next_index; + + return 0; +} + +static int +if_tx_feature_node_init(const struct rte_graph *graph, struct rte_node *node) +{ + RTE_SET_USED(graph); + + /* pkt_drop */ + IF_TX_FEATURE_LAST_NEXT_INDEX(node->ctx) = 0; + + return 0; +} + +static uint16_t +if_tx_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + uint16_t held = 0, next; + void **to_next, **from; + uint16_t last_spec = 0; + rte_edge_t next_index; + struct rte_mbuf *mbuf; + int i; + + /* Speculative next */ + next_index = IF_TX_FEATURE_LAST_NEXT_INDEX(node->ctx); + + from = objs; + to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + for (i = 0; i < nb_objs; i++) { + + mbuf = (struct rte_mbuf *)objs[i]; + + /* port-tx node starts from next edge 1*/ + next = if_tx_feature_nm->next_index[mbuf->port]; + + if (unlikely(next_index != next)) { + /* Copy things successfully speculated till now */ + rte_memcpy(to_next, from, last_spec * sizeof(from[0])); + from += last_spec; + to_next += last_spec; + held += last_spec; + last_spec = 0; + + rte_node_enqueue_x1(graph, node, next, from[0]); + from += 1; + } else { + last_spec += 1; + } + } + /* !!! Home run !!! */ + if (likely(last_spec == nb_objs)) { + rte_node_next_stream_move(graph, node, next_index); + return nb_objs; + } + held += last_spec; + rte_memcpy(to_next, from, last_spec * sizeof(from[0])); + rte_node_next_stream_put(graph, node, next_index, held); + + IF_TX_FEATURE_LAST_NEXT_INDEX(node->ctx) = next; + + return nb_objs; +} + +static struct rte_node_register if_tx_feature_node = { + .process = if_tx_feature_node_process, + .init = if_tx_feature_node_init, + .name = "interface_tx", + .nb_edges = 1, + .next_nodes = { + [0] = "pkt_drop", + }, +}; + +struct rte_node_register * +if_tx_feature_node_get(void) +{ + return &if_tx_feature_node; +} + +RTE_NODE_REGISTER(if_tx_feature_node); + +/* if_tx feature node */ +struct rte_graph_feature_register if_tx_feature = { + .feature_name = RTE_IP4_OUTPUT_END_FEATURE_NAME, + .arc_name = RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + .feature_process_fn = if_tx_feature_node_process, + .feature_node = &if_tx_feature_node, +}; diff --git a/lib/node/interface_tx_feature_priv.h b/lib/node/interface_tx_feature_priv.h new file mode 100644 index 0000000000..846191572d --- /dev/null +++ b/lib/node/interface_tx_feature_priv.h @@ -0,0 +1,33 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ +#ifndef __INCLUDE_IF_TX_FEATURE_PRIV_H__ +#define __INCLUDE_IF_TX_FEATURE_PRIV_H__ + +#include <rte_common.h> + +extern struct rte_graph_feature_register if_tx_feature; + +/** + * @internal + * + * Get the ipv4 rewrite node. + * + * @return + * Pointer to the ipv4 rewrite node. + */ +struct rte_node_register *if_tx_feature_node_get(void); + +/** + * @internal + * + * Set the Edge index of a given port_id. + * + * @param port_id + * Ethernet port identifier. + * @param next_index + * Edge index of the Given Tx node. + */ +int if_tx_feature_node_set_next(uint16_t port_id, uint16_t next_index); + +#endif /* __INCLUDE_INTERFCE_TX_FEATURE_PRIV_H */ diff --git a/lib/node/ip4_rewrite.c b/lib/node/ip4_rewrite.c index 34a920df5e..ab64aa0a3c 100644 --- a/lib/node/ip4_rewrite.c +++ b/lib/node/ip4_rewrite.c @@ -6,6 +6,7 @@ #include <rte_ether.h> #include <rte_graph.h> #include <rte_graph_worker.h> +#include <rte_graph_feature_arc_worker.h> #include <rte_ip.h> #include <rte_malloc.h> #include <rte_vect.h> @@ -14,15 +15,27 @@ #include "ip4_rewrite_priv.h" #include "node_private.h" +#include "interface_tx_feature_priv.h" + +#ifndef RTE_IP4_OUTPUT_ARC_INDEXES +#define RTE_IP4_OUTPUT_ARC_INDEXES RTE_MAX_ETHPORTS +#endif struct ip4_rewrite_node_ctx { /* Dynamic offset to mbuf priv1 */ int mbuf_priv1_off; + /* Dynamic offset to feature arc field */ + int arc_dyn_off; /* Cached next index */ uint16_t next_index; + /* tx interface of last mbuf */ + uint16_t last_tx_if; + /* Cached feature arc handle */ + rte_graph_feature_arc_t output_feature_arc; }; static struct ip4_rewrite_node_main *ip4_rewrite_nm; +static int port_to_next_index_diff; #define IP4_REWRITE_NODE_LAST_NEXT(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->next_index) @@ -30,16 +43,175 @@ static struct ip4_rewrite_node_main *ip4_rewrite_nm; #define IP4_REWRITE_NODE_PRIV1_OFF(ctx) \ (((struct ip4_rewrite_node_ctx *)ctx)->mbuf_priv1_off) -static uint16_t -ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, - void **objs, uint16_t nb_objs) +#define IP4_REWRITE_NODE_FEAT_OFF(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->arc_dyn_off) + +#define IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->output_feature_arc) + +#define IP4_REWRITE_NODE_LAST_TX_IF(ctx) \ + (((struct ip4_rewrite_node_ctx *)ctx)->last_tx_if) + +static __rte_always_inline void +check_output_feature_arc_x1(struct rte_graph_feature_arc *arc, uint16_t *tx_if, + struct rte_mbuf *mbuf0, uint16_t *next0, + uint16_t *last_next_index, + rte_graph_feature_data_t *feature_data, const int dyn) +{ + struct rte_graph_feature_arc_mbuf_dynfields *d0 = NULL; + uint16_t port0; + + /* make sure packets are not being sent to pkt_drop node */ + if (unlikely(!(*next0))) + return; + + port0 = (*next0) - port_to_next_index_diff; + + /* get pointer to feature arc mbuf */ + d0 = rte_graph_feature_arc_mbuf_dynfields_get(mbuf0, dyn); + + /* Check if last packet's tx port not same as current */ + if (unlikely(*tx_if != port0)) { + if (unlikely(rte_graph_feature_data_first_feature_get(arc, port0, + &d0->feature_data))) { + *next0 = rte_graph_feature_data_edge_get(arc, d0->feature_data); + mbuf0->port = port0; + } + + *last_next_index = *next0; + *tx_if = port0; + *feature_data = d0->feature_data; + } else { + if (unlikely(rte_graph_feature_data_is_valid(*feature_data))) { + *next0 = *last_next_index; + mbuf0->port = *tx_if; + d0->feature_data = *feature_data; + } + } +} + +static __rte_always_inline void +check_output_feature_arc_x4(struct rte_graph_feature_arc *arc, uint16_t *tx_if, + struct rte_mbuf *mbuf0, struct rte_mbuf *mbuf1, + struct rte_mbuf *mbuf2, struct rte_mbuf *mbuf3, + uint16_t *next0, uint16_t *next1, uint16_t *next2, + uint16_t *next3, uint16_t *last_next_index, + rte_graph_feature_data_t *feature_data, const int dyn) +{ + struct rte_graph_feature_arc_mbuf_dynfields *d0 = NULL, *d1 = NULL, *d2 = NULL, *d3 = NULL; + uint16_t port0, port1, port2, port3; + uint16_t xor = 0; + + /* If no ip4_rewrite_set_next() is called yet */ + if (unlikely(!port_to_next_index_diff)) + return; + + /* get pointer to feature arc dyn field */ + d0 = rte_graph_feature_arc_mbuf_dynfields_get(mbuf0, dyn); + d1 = rte_graph_feature_arc_mbuf_dynfields_get(mbuf1, dyn); + d2 = rte_graph_feature_arc_mbuf_dynfields_get(mbuf2, dyn); + d3 = rte_graph_feature_arc_mbuf_dynfields_get(mbuf3, dyn); + + /* + * Check if all four packets are going to same next_index/port + */ + xor = ((*tx_if) + port_to_next_index_diff) ^ (*next0); + xor += (*next0) ^ (*next1); + xor += (*next1) ^ (*next2); + xor += (*next2) ^ (*next3); + + if (xor) { + /* packets tx ports are not same, check first feature for each mbuf + * make sure next0 != 0 which is pkt_drop + */ + port0 = (*next0) - port_to_next_index_diff; + port1 = (*next1) - port_to_next_index_diff; + port2 = (*next2) - port_to_next_index_diff; + port3 = (*next3) - port_to_next_index_diff; + if (unlikely((*next0 >= port_to_next_index_diff) && + rte_graph_feature_data_first_feature_get(arc, port0, + &d0->feature_data))) { + /* update next0 from feature arc */ + *next0 = rte_graph_feature_data_edge_get(arc, d0->feature_data); + mbuf0->port = port0; + + *tx_if = port0; + *last_next_index = *next0; + *feature_data = d0->feature_data; + } + + if (unlikely((*next1 >= port_to_next_index_diff) && + rte_graph_feature_data_first_feature_get(arc, port1, + &d1->feature_data))) { + port1 = (*next1) - port_to_next_index_diff; + *next1 = rte_graph_feature_data_edge_get(arc, d1->feature_data); + mbuf1->port = port1; + *tx_if = port1; + *last_next_index = *next1; + *feature_data = d1->feature_data; + } + + if (unlikely((*next2 >= port_to_next_index_diff) && + rte_graph_feature_data_first_feature_get(arc, port2, + &d2->feature_data))) { + port2 = (*next2) - port_to_next_index_diff; + *next2 = rte_graph_feature_data_edge_get(arc, d2->feature_data); + mbuf2->port = port2; + *tx_if = port2; + *last_next_index = *next2; + *feature_data = d2->feature_data; + } + + if (unlikely((*next3 >= port_to_next_index_diff) && + rte_graph_feature_data_first_feature_get(arc, port3, + &d3->feature_data))) { + port3 = (*next3) - port_to_next_index_diff; + *next3 = rte_graph_feature_data_edge_get(arc, d3->feature_data); + mbuf3->port = port3; + *tx_if = port3; + *last_next_index = *next3; + *feature_data = d3->feature_data; + } + } else { + /* All packets are same as last tx port. Check if feature enabled + * on last packet is valid or not. If invalid no need to + * change any next[0-3] + * Also check packet is not being sent to pkt_drop node + */ + if (unlikely(rte_graph_feature_data_is_valid(*feature_data) && + (*next0 != 0))) { + *next0 = *last_next_index; + *next1 = *last_next_index; + *next2 = *last_next_index; + *next3 = *last_next_index; + + d0->feature_data = *feature_data; + d1->feature_data = *feature_data; + d2->feature_data = *feature_data; + d3->feature_data = *feature_data; + + mbuf0->port = *tx_if; + mbuf1->port = *tx_if; + mbuf2->port = *tx_if; + mbuf3->port = *tx_if; + } + } +} + +static __rte_always_inline uint16_t +__ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs, + const int dyn, const int check_enabled_features, + const int feat_dyn, + struct rte_graph_feature_arc *out_feature_arc) { struct rte_mbuf *mbuf0, *mbuf1, *mbuf2, *mbuf3, **pkts; struct ip4_rewrite_nh_header *nh = ip4_rewrite_nm->nh; - const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); uint16_t next0, next1, next2, next3, next_index; struct rte_ipv4_hdr *ip0, *ip1, *ip2, *ip3; uint16_t n_left_from, held = 0, last_spec = 0; + rte_graph_feature_data_t feature_data; + uint16_t last_tx_if, last_next_index; void *d0, *d1, *d2, *d3; void **to_next, **from; rte_xmm_t priv01; @@ -57,6 +229,27 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, for (i = 0; i < 4 && i < n_left_from; i++) rte_prefetch0(pkts[i]); + if (check_enabled_features) { + rte_graph_feature_arc_prefetch(out_feature_arc); + + last_tx_if = IP4_REWRITE_NODE_LAST_TX_IF(node->ctx); + + /* If feature is enabled on last_tx_if, prefetch data + * corresponding to first feature + */ + if (unlikely(rte_graph_feature_data_first_feature_get(out_feature_arc, + last_tx_if, + &feature_data))) + rte_graph_feature_arc_feature_data_prefetch(out_feature_arc, + feature_data); + + /* Reset last_tx_if and last_next_index to call feature arc APIs + * for initial packets in every node loop + */ + last_tx_if = UINT16_MAX; + last_next_index = UINT16_MAX; + } + /* Get stream for the speculated next node */ to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); /* Update Ethernet header of pkts */ @@ -78,6 +271,7 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, pkts += 4; n_left_from -= 4; + priv01.u64[0] = node_mbuf_priv1(mbuf0, dyn)->u; priv01.u64[1] = node_mbuf_priv1(mbuf1, dyn)->u; priv23.u64[0] = node_mbuf_priv1(mbuf2, dyn)->u; @@ -132,6 +326,16 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, ip3->time_to_live = priv23.u16[5] - 1; ip3->hdr_checksum = priv23.u16[6] + priv23.u16[7]; + /* Once all mbufs are updated with next hop data. + * check if any feature is enabled to override + * next edges + */ + if (check_enabled_features) + check_output_feature_arc_x4(out_feature_arc, &last_tx_if, + mbuf0, mbuf1, mbuf2, mbuf3, + &next0, &next1, &next2, &next3, + &last_next_index, &feature_data, feat_dyn); + /* Enqueue four to next node */ rte_edge_t fix_spec = ((next_index == next0) && (next0 == next1) && @@ -225,6 +429,11 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, ip0->hdr_checksum = chksum; ip0->time_to_live = node_mbuf_priv1(mbuf0, dyn)->ttl - 1; + if (check_enabled_features) + check_output_feature_arc_x1(out_feature_arc, &last_tx_if, + mbuf0, &next0, &last_next_index, + &feature_data, feat_dyn); + if (unlikely(next_index ^ next0)) { /* Copy things successfully speculated till now */ rte_memcpy(to_next, from, last_spec * sizeof(from[0])); @@ -252,12 +461,49 @@ ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, /* Save the last next used */ IP4_REWRITE_NODE_LAST_NEXT(node->ctx) = next_index; + if (check_enabled_features) + IP4_REWRITE_NODE_LAST_TX_IF(node->ctx) = last_tx_if; + return nb_objs; } +static uint16_t +ip4_rewrite_feature_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = NULL; + rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + const int feat_dyn = IP4_REWRITE_NODE_FEAT_OFF(node->ctx); + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + arc = rte_graph_feature_arc_get(IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx)); + if (unlikely(arc && rte_graph_feature_arc_is_any_feature_enabled(arc))) + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 1 /* check features */, feat_dyn, arc); + + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, + 0 /* don't care */, + NULL /* don't care*/); +} + +static uint16_t +ip4_rewrite_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + const int dyn = IP4_REWRITE_NODE_PRIV1_OFF(node->ctx); + + return __ip4_rewrite_node_process(graph, node, objs, nb_objs, dyn, + 0/* don't check features*/, + 0 /* don't care */, + NULL/* don't care */); +} + + static int ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) { + rte_graph_feature_arc_t feature_arc; static bool init_once; RTE_SET_USED(graph); @@ -268,9 +514,27 @@ ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) &node_mbuf_priv1_dynfield_desc); if (node_mbuf_priv1_dynfield_offset < 0) return -rte_errno; + + /* Create ipv4-output feature arc, if not created + */ + if (rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + &feature_arc) < 0) { + node_err("ip4_rewrite", "Feature arc \"%s\" not found", + RTE_IP4_OUTPUT_FEATURE_ARC_NAME); + } else { + node_err("ip4_rewrite", "Feature arc \"%s\" found", + RTE_IP4_OUTPUT_FEATURE_ARC_NAME); + } + init_once = true; } IP4_REWRITE_NODE_PRIV1_OFF(node->ctx) = node_mbuf_priv1_dynfield_offset; + IP4_REWRITE_NODE_FEAT_OFF(node->ctx) = rte_graph_feature_arc_mbuf_dynfield_offset_get(); + IP4_REWRITE_NODE_OUTPUT_FEATURE_ARC(node->ctx) = feature_arc; + + /* By default, set cached next node to pkt_drop */ + IP4_REWRITE_NODE_LAST_NEXT(node->ctx) = 0; + IP4_REWRITE_NODE_LAST_TX_IF(node->ctx) = 0; node_dbg("ip4_rewrite", "Initialized ip4_rewrite node initialized"); @@ -280,6 +544,8 @@ ip4_rewrite_node_init(const struct rte_graph *graph, struct rte_node *node) int ip4_rewrite_set_next(uint16_t port_id, uint16_t next_index) { + static int once; + if (ip4_rewrite_nm == NULL) { ip4_rewrite_nm = rte_zmalloc( "ip4_rewrite", sizeof(struct ip4_rewrite_node_main), @@ -287,6 +553,10 @@ ip4_rewrite_set_next(uint16_t port_id, uint16_t next_index) if (ip4_rewrite_nm == NULL) return -ENOMEM; } + if (!once) { + port_to_next_index_diff = next_index - port_id; + once = 1; + } ip4_rewrite_nm->next_index[port_id] = next_index; return 0; @@ -345,3 +615,23 @@ ip4_rewrite_node_get(void) } RTE_NODE_REGISTER(ip4_rewrite_node); + +/* IP4 output arc */ +static struct rte_graph_feature_arc_register ip4_output_arc = { + .arc_name = RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + + /* This arc works on all ethdevs */ + .max_indexes = RTE_IP4_OUTPUT_ARC_INDEXES, + + .start_node = &ip4_rewrite_node, + + /* overwrites start_node->process() function with following only if + * application calls rte_graph_feature_arc_init() + */ + .start_node_feature_process_fn = ip4_rewrite_feature_node_process, + + /* end feature node of an arc*/ + .end_feature = &if_tx_feature, +}; + +RTE_GRAPH_FEATURE_ARC_REGISTER(ip4_output_arc); diff --git a/lib/node/meson.build b/lib/node/meson.build index 0bed97a96c..11e03d9ef6 100644 --- a/lib/node/meson.build +++ b/lib/node/meson.build @@ -24,6 +24,7 @@ sources = files( 'pkt_cls.c', 'pkt_drop.c', 'udp4_input.c', + 'interface_tx_feature.c' ) headers = files( 'rte_node_eth_api.h', diff --git a/lib/node/node_private.h b/lib/node/node_private.h index 4fafab19be..9488df225c 100644 --- a/lib/node/node_private.h +++ b/lib/node/node_private.h @@ -13,6 +13,7 @@ #include <rte_mbuf_dyn.h> #include <rte_graph_worker_common.h> +#include <rte_graph_feature_arc_worker.h> extern int rte_node_logtype; #define RTE_LOGTYPE_NODE rte_node_logtype diff --git a/lib/node/rte_node_ip4_api.h b/lib/node/rte_node_ip4_api.h index 950751a525..8cfae14b46 100644 --- a/lib/node/rte_node_ip4_api.h +++ b/lib/node/rte_node_ip4_api.h @@ -24,6 +24,10 @@ extern "C" { #endif +/** IP4 output arc */ +#define RTE_IP4_OUTPUT_FEATURE_ARC_NAME "rte_ip4_output_arc" +#define RTE_IP4_OUTPUT_END_FEATURE_NAME "rte_if_tx_feature" + /** * IP4 lookup next nodes. */ -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
* [PATCH v6 4/4] app/graph: add custom feature nodes for ip4 output arc 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena ` (2 preceding siblings ...) 2025-01-03 6:06 ` [PATCH v6 3/4] ip4: add ip4 output feature arc Nitin Saxena @ 2025-01-03 6:06 ` Nitin Saxena [not found] ` <SJ0PR18MB5111B56B4323FB3DFD147801B6152@SJ0PR18MB5111.namprd18.prod.outlook.com> 4 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2025-01-03 6:06 UTC (permalink / raw) To: Jerin Jacob, Kiran Kumar K, Nithin Dabilpuram, Zhirun Yan, Robin Jarry, Christophe Fontaine Cc: dev, Nitin Saxena - Added cmdline argument "--enable-graph-feature-arc" to call rte_graph_feature_arc_init() before rte_graph_create() which creates in-built arcs and feature nodes - Added custom feature nodes in app/graph which are added to ip4 output arc. - Custom features can be enabled/disabled at runtime on any ethdev via CLI. graph> help feature graph> feature enable <arc name> <feature name> <port-id> graph> feature disable <arc name> <feature name> <port-id> graph> graph stats show Signed-off-by: Nitin Saxena <nsaxena@marvell.com> --- app/graph/commands.list | 6 ++ app/graph/feature.c | 141 ++++++++++++++++++++++++++++++ app/graph/feature.h | 13 +++ app/graph/graph.c | 4 + app/graph/ip4_output_hook.c | 169 ++++++++++++++++++++++++++++++++++++ app/graph/main.c | 15 +++- app/graph/meson.build | 2 + app/graph/module_api.h | 2 + 8 files changed, 351 insertions(+), 1 deletion(-) create mode 100644 app/graph/feature.c create mode 100644 app/graph/feature.h create mode 100644 app/graph/ip4_output_hook.c diff --git a/app/graph/commands.list b/app/graph/commands.list index c027f73b0e..49d81f50ae 100644 --- a/app/graph/commands.list +++ b/app/graph/commands.list @@ -31,3 +31,9 @@ help ipv6_lookup # Print help on ipv6_lo neigh add ipv4 <IPv4>ip <STRING>mac # Add static neighbour for IPv4 neigh add ipv6 <IPv6>ip <STRING>mac # Add static neighbour for IPv6 help neigh # Print help on neigh commands + +feature arcs # show all feature arcs +feature <STRING>name show # Show feature arc details +feature enable <STRING>arc_name <STRING>feature_name <UINT16>interface # Enable feature on interface +feature disable <STRING>arc_name <STRING>feature_name <UINT16>interface # Disable feature on interface +help feature # Print help on feature command diff --git a/app/graph/feature.c b/app/graph/feature.c new file mode 100644 index 0000000000..2cf21b11ce --- /dev/null +++ b/app/graph/feature.c @@ -0,0 +1,141 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +#include <cmdline_parse.h> +#include <cmdline_parse_num.h> +#include <cmdline_parse_string.h> +#include <cmdline_socket.h> +#include <rte_ethdev.h> + +#include "module_api.h" + +static const char +cmd_feature_arcs_help[] = "feature arcs # Display all feature arcs"; + +static const char +cmd_feature_show_help[] = "feature <arc_name> show # Display features within an arc"; + +static const char +cmd_feature_enable_help[] = "feature enable <arc name> <feature name> <port-id>"; + +static const char +cmd_feature_disable_help[] = "feature disable <arc name> <feature name> <port-id>"; + +static void +feature_show(const char *arc_name) +{ + rte_graph_feature_arc_t _arc; + uint32_t length, count, i; + + length = strlen(conn->msg_out); + conn->msg_out += length; + + if (rte_graph_feature_arc_lookup_by_name(arc_name, &_arc) < 0) + return; + + count = rte_graph_feature_arc_num_features(_arc); + + if (count) { + snprintf(conn->msg_out, conn->msg_out_len_max, "\n%s%s%s\n", + "----------------------------- feature arc: ", + rte_graph_feature_arc_get(_arc)->feature_arc_name, + " -----------------------------"); + for (i = 0; i < count; i++) + snprintf(conn->msg_out + strlen(conn->msg_out), + conn->msg_out_len_max, "%s\n", + rte_graph_feature_arc_feature_to_name(_arc, i)); + } + length = strlen(conn->msg_out); + conn->msg_out_len_max -= length; +} + +static void +feature_arcs_show(void) +{ + uint32_t length, count, i; + char **names; + + length = strlen(conn->msg_out); + conn->msg_out += length; + + count = rte_graph_feature_arc_names_get(NULL); + + if (count) { + names = malloc(count); + if (!names) { + snprintf(conn->msg_out, conn->msg_out_len_max, "Failed to allocate memory\n"); + return; + } + count = rte_graph_feature_arc_names_get(names); + snprintf(conn->msg_out, conn->msg_out_len_max, "\n%s\n", + "----------------------------- feature arcs -----------------------------"); + for (i = 0; i < count; i++) + feature_show(names[i]); + free(names); + } + length = strlen(conn->msg_out); + conn->msg_out_len_max -= length; +} + +void +cmd_feature_parsed(void *parsed_result, __rte_unused struct cmdline *cl, + __rte_unused void *data) +{ + struct cmd_feature_result *res = parsed_result; + + feature_show(res->name); +} + + +void +cmd_feature_arcs_parsed(__rte_unused void *parsed_result, __rte_unused struct cmdline *cl, + __rte_unused void *data) +{ + feature_arcs_show(); +} + +void +cmd_feature_enable_parsed(void *parsed_result, __rte_unused struct cmdline *cl, + __rte_unused void *data) +{ + struct cmd_feature_enable_result *res = parsed_result; + rte_graph_feature_arc_t arc; + + if (!rte_graph_feature_arc_lookup_by_name(res->arc_name, &arc)) + rte_graph_feature_enable(arc, res->interface, res->feature_name, + res->interface, NULL); +} + +void +cmd_feature_disable_parsed(void *parsed_result, __rte_unused struct cmdline *cl, + __rte_unused void *data) +{ + struct cmd_feature_disable_result *res = parsed_result; + rte_graph_feature_arc_t arc; + + if (!rte_graph_feature_arc_lookup_by_name(res->arc_name, &arc)) + rte_graph_feature_disable(arc, res->interface, res->feature_name, NULL); + +} + +void +cmd_help_feature_parsed(__rte_unused void *parsed_result, __rte_unused struct cmdline *cl, + __rte_unused void *data) +{ + size_t len; + + len = strlen(conn->msg_out); + conn->msg_out += len; + snprintf(conn->msg_out, conn->msg_out_len_max, "\n%s\n%s\n%s\n%s\n%s\n", + "----------------------------- feature command help -----------------------------", + cmd_feature_arcs_help, cmd_feature_show_help, cmd_feature_enable_help, + cmd_feature_disable_help); + + len = strlen(conn->msg_out); + conn->msg_out_len_max -= len; +} diff --git a/app/graph/feature.h b/app/graph/feature.h new file mode 100644 index 0000000000..76393dabc6 --- /dev/null +++ b/app/graph/feature.h @@ -0,0 +1,13 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#ifndef APP_GRAPH_FEATURE_H +#define APP_GRAPH_FEATURE_H + +#include <cmdline_parse.h> +#include <rte_graph_feature_arc_worker.h> + +int feature_enable(const char *arc_name, const char *feature_name, uint16_t portid); +int feature_disable(const char *arc_name, const char *feature_name, uint16_t portid); +#endif diff --git a/app/graph/graph.c b/app/graph/graph.c index 3af031cbaf..3ccd702ed9 100644 --- a/app/graph/graph.c +++ b/app/graph/graph.c @@ -12,6 +12,7 @@ #include <cmdline_socket.h> #include <rte_ethdev.h> #include <rte_graph_worker.h> +#include <rte_graph_feature_arc_worker.h> #include <rte_log.h> #include "graph_priv.h" @@ -265,6 +266,9 @@ cmd_graph_start_parsed(__rte_unused void *parsed_result, __rte_unused struct cmd uint32_t nb_graphs = 0, nb_conf, i; int rc = -EINVAL; + if (app_graph_feature_arc_enabled()) + rte_graph_feature_arc_init(); + conf = graph_rxtx_node_config_get(&nb_conf, &nb_graphs); for (i = 0; i < MAX_GRAPH_USECASES; i++) { if (!strcmp(graph_config.usecases[i].name, "l3fwd")) { diff --git a/app/graph/ip4_output_hook.c b/app/graph/ip4_output_hook.c new file mode 100644 index 0000000000..1a389f1113 --- /dev/null +++ b/app/graph/ip4_output_hook.c @@ -0,0 +1,169 @@ +/* SPDX-License-Identifier: BSD-3-Clause + * Copyright(C) 2025 Marvell International Ltd. + */ + +#include <rte_graph.h> +#include <rte_graph_worker.h> +#include <rte_graph_feature_arc_worker.h> + +#include "rte_node_ip4_api.h" + +#define IP4_OUTPUT_HOOK_FEATURE1_NAME "app_graph_ip4_output_hook_f1" +#define IP4_OUTPUT_HOOK_FEATURE2_NAME "app_graph_ip4_output_hook_f2" + +struct output_hook_node_ctx { + rte_graph_feature_arc_t out_arc; + uint16_t last_index; +}; + +#define OUTPUT_HOOK_FEATURE_ARC(ctx) \ + (((struct output_hook_node_ctx *)ctx)->out_arc) + +#define OUTPUT_HOOK_LAST_NEXT_INDEX(ctx) \ + (((struct output_hook_node_ctx *)ctx)->last_index) + +static int +__app_graph_ip4_output_hook_node_init(const struct rte_graph *graph, struct rte_node *node) +{ + rte_graph_feature_arc_t feature; + + RTE_SET_USED(graph); + + rte_graph_feature_arc_lookup_by_name(RTE_IP4_OUTPUT_FEATURE_ARC_NAME, &feature); + + OUTPUT_HOOK_FEATURE_ARC(node->ctx) = feature; + /* pkt_drop */ + OUTPUT_HOOK_LAST_NEXT_INDEX(node->ctx) = 0; + + return 0; +} + +static __rte_always_inline uint16_t +__app_graph_ip4_output_hook_node_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + struct rte_graph_feature_arc *arc = + rte_graph_feature_arc_get(OUTPUT_HOOK_FEATURE_ARC(node->ctx)); + int feat_dyn_off = rte_graph_feature_arc_mbuf_dynfield_offset_get(); + struct rte_graph_feature_arc_mbuf_dynfields *mbfields = NULL; + rte_graph_feature_data_t fdata; + void **to_next, **from; + uint16_t last_spec = 0; + rte_edge_t next_index; + struct rte_mbuf *mbuf; + uint16_t held = 0; + uint16_t next; + int i; + + /* Speculative next */ + next_index = OUTPUT_HOOK_LAST_NEXT_INDEX(node->ctx); + + from = objs; + to_next = rte_node_next_stream_get(graph, node, next_index, nb_objs); + for (i = 0; i < nb_objs; i++) { + + mbuf = (struct rte_mbuf *)objs[i]; + + /* Send mbuf to next enabled feature */ + mbfields = rte_graph_feature_arc_mbuf_dynfields_get(mbuf, feat_dyn_off); + fdata = rte_graph_feature_data_next_feature_get(arc, mbfields->feature_data); + next = rte_graph_feature_data_edge_get(arc, fdata); + + if (unlikely(next_index != next)) { + /* Copy things successfully speculated till now */ + rte_memcpy(to_next, from, last_spec * sizeof(from[0])); + from += last_spec; + to_next += last_spec; + held += last_spec; + last_spec = 0; + + rte_node_enqueue_x1(graph, node, next, from[0]); + from += 1; + } else { + last_spec += 1; + } + } + /* !!! Home run !!! */ + if (likely(last_spec == nb_objs)) { + rte_node_next_stream_move(graph, node, next_index); + return nb_objs; + } + held += last_spec; + rte_memcpy(to_next, from, last_spec * sizeof(from[0])); + rte_node_next_stream_put(graph, node, next_index, held); + OUTPUT_HOOK_LAST_NEXT_INDEX(node->ctx) = next; + + return nb_objs; +} + +static int +app_graph_ip4_output_hook_node1_init(const struct rte_graph *graph, struct rte_node *node) +{ + return __app_graph_ip4_output_hook_node_init(graph, node); +} + +static __rte_always_inline uint16_t +app_graph_ip4_output_hook_node1_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + return __app_graph_ip4_output_hook_node_process(graph, node, objs, nb_objs); +} + +static struct rte_node_register app_graph_ip4_output_hook_node1 = { + .process = app_graph_ip4_output_hook_node1_process, + .init = app_graph_ip4_output_hook_node1_init, + .name = "app_graph_ip4_output_hook_node1", + .nb_edges = 1, + .next_nodes = { + [0] = "pkt_drop", + }, +}; + +RTE_NODE_REGISTER(app_graph_ip4_output_hook_node1); + +static int +app_graph_ip4_output_hook_node2_init(const struct rte_graph *graph, struct rte_node *node) +{ + return __app_graph_ip4_output_hook_node_init(graph, node); +} + +static __rte_always_inline uint16_t +app_graph_ip4_output_hook_node2_process(struct rte_graph *graph, struct rte_node *node, + void **objs, uint16_t nb_objs) +{ + return __app_graph_ip4_output_hook_node_process(graph, node, objs, nb_objs); +} + +static struct rte_node_register app_graph_ip4_output_hook_node2 = { + .process = app_graph_ip4_output_hook_node2_process, + .init = app_graph_ip4_output_hook_node2_init, + .name = "app_graph_ip4_output_hook_node2", + .nb_edges = 1, + .next_nodes = { + [0] = "pkt_drop", + }, +}; + +RTE_NODE_REGISTER(app_graph_ip4_output_hook_node2); + +/* if feature1 */ +struct rte_graph_feature_register app_graph_ip4_output_hook_feature1 = { + .feature_name = IP4_OUTPUT_HOOK_FEATURE1_NAME, + .arc_name = RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + /* Same as regular function */ + .feature_process_fn = app_graph_ip4_output_hook_node1_process, + .feature_node = &app_graph_ip4_output_hook_node1, + .runs_before = IP4_OUTPUT_HOOK_FEATURE2_NAME, +}; + +/* if feature2 (same as f1) */ +struct rte_graph_feature_register app_graph_ip4_output_hook_feature2 = { + .feature_name = IP4_OUTPUT_HOOK_FEATURE2_NAME, + .arc_name = RTE_IP4_OUTPUT_FEATURE_ARC_NAME, + /* Same as regular function */ + .feature_node = &app_graph_ip4_output_hook_node2, + .feature_process_fn = app_graph_ip4_output_hook_node2_process, +}; + +RTE_GRAPH_FEATURE_REGISTER(app_graph_ip4_output_hook_feature1); +RTE_GRAPH_FEATURE_REGISTER(app_graph_ip4_output_hook_feature2); diff --git a/app/graph/main.c b/app/graph/main.c index 465376425c..56294f2693 100644 --- a/app/graph/main.c +++ b/app/graph/main.c @@ -28,6 +28,7 @@ static struct app_params { struct conn_params conn; char *script_name; bool enable_graph_stats; + bool enable_feature_arc; } app = { .conn = { .welcome = "\nWelcome!\n\n", @@ -42,6 +43,7 @@ static struct app_params { }, .script_name = NULL, .enable_graph_stats = false, + .enable_feature_arc = false, }; static void @@ -59,6 +61,7 @@ app_args_parse(int argc, char **argv) struct option lgopts[] = { {"help", 0, 0, 'H'}, {"enable-graph-stats", 0, 0, 'g'}, + {"enable-graph-feature-arc", 0, 0, 'f'}, }; int h_present, p_present, s_present, n_args, i; char *app_name = argv[0]; @@ -81,7 +84,7 @@ app_args_parse(int argc, char **argv) p_present = 0; s_present = 0; - while ((opt = getopt_long(argc, argv, "h:p:s:", lgopts, &option_index)) != EOF) { + while ((opt = getopt_long(argc, argv, "h:p:s:f", lgopts, &option_index)) != EOF) { switch (opt) { case 'h': if (h_present) { @@ -142,6 +145,10 @@ app_args_parse(int argc, char **argv) "--enable-graph-stats"); break; + case 'f': + app.enable_feature_arc = true; + break; + case 'H': default: printf(usage, app_name); @@ -159,6 +166,12 @@ app_graph_stats_enabled(void) return app.enable_graph_stats; } +bool +app_graph_feature_arc_enabled(void) +{ + return app.enable_feature_arc; +} + bool app_graph_exit(void) { diff --git a/app/graph/meson.build b/app/graph/meson.build index 344e4a418f..d7e8c431ea 100644 --- a/app/graph/meson.build +++ b/app/graph/meson.build @@ -24,6 +24,8 @@ sources = files( 'mempool.c', 'neigh.c', 'utils.c', + 'feature.c', + 'ip4_output_hook.c' ) cmd_h = custom_target('commands_hdr', diff --git a/app/graph/module_api.h b/app/graph/module_api.h index b872872dc1..25f88e28de 100644 --- a/app/graph/module_api.h +++ b/app/graph/module_api.h @@ -20,6 +20,7 @@ #include "neigh.h" #include "route.h" #include "utils.h" +#include "feature.h" /* * Externs @@ -28,6 +29,7 @@ extern volatile bool force_quit; extern struct conn *conn; bool app_graph_stats_enabled(void); +bool app_graph_feature_arc_enabled(void); bool app_graph_exit(void); #endif -- 2.43.0 ^ permalink raw reply [flat|nested] 55+ messages in thread
[parent not found: <SJ0PR18MB5111B56B4323FB3DFD147801B6152@SJ0PR18MB5111.namprd18.prod.outlook.com>]
* Feature arc slides [not found] ` <SJ0PR18MB5111B56B4323FB3DFD147801B6152@SJ0PR18MB5111.namprd18.prod.outlook.com> @ 2025-01-03 14:59 ` Nitin Saxena 2025-01-06 0:15 ` Stephen Hemminger 2025-01-10 13:59 ` [EXTERNAL] [PATCH v6 0/4] add feature arc in rte_graph Robin Jarry 1 sibling, 1 reply; 55+ messages in thread From: Nitin Saxena @ 2025-01-03 14:59 UTC (permalink / raw) To: dev [-- Attachment #1: Type: text/plain, Size: 6082 bytes --] Sending to DPDK community again Thanks, Nitin ---------- Forwarded message --------- From: Nitin Saxena <nsaxena@marvell.com> Date: Fri, Jan 3, 2025 at 8:11 PM Subject: RE: [EXTERNAL] [PATCH v6 0/4] add feature arc in rte_graph To: Jerin Jacob <jerinj@marvell.com>, Kiran Kumar Kokkilagadda <kirankumark@marvell.com>, Nithin Kumar Dabilpuram <ndabilpuram@marvell.com>, Zhirun Yan <yanzhirun_163@163.com>, Robin Jarry <rjarry@redhat.com>, Christophe Fontaine <cfontain@redhat.com> Cc: dev@dpdk.org <dev@dpdk.org>, Nitin Saxena <nsaxena16@gmail.com> Hi, Please find attached slides explaining feature arc concept and its usage in graph library I can also present these slides to community to facilitate this patch series review process Thanks, Nitin From: Nitin Saxena <nsaxena@marvell.com> Sent: Friday, January 3, 2025 11:36 AM To: Jerin Jacob <jerinj@marvell.com>; Kiran Kumar Kokkilagadda <kirankumark@marvell.com>; Nithin Kumar Dabilpuram <ndabilpuram@marvell.com>; Zhirun Yan <yanzhirun_163@163.com>; Robin Jarry <rjarry@redhat.com>; Christophe Fontaine <cfontain@redhat.com> Cc: dev@dpdk.org; Nitin Saxena <nsaxena16@gmail.com> Subject: [EXTERNAL] [PATCH v6 0/4] add feature arc in rte_graph Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Feature arc represents an ordered list of features/protocols at a given networking layer. It is a high level abstraction to connect various rte_graph nodes, as feature nodes, and allow packets steering across these nodes in a generic manner. Features (or feature nodes) are nodes which handles partial or complete handling of a protocol in fast path. Like ipv4-rewrite node, which adds rewrite data to an outgoing IPv4 packet. However in above example, outgoing interface(say "eth0") may have outbound IPsec policy enabled, hence packets must be steered from ipv4-rewrite node to ipsec-outbound-policy node for outbound IPsec policy lookup. On the other hand, packets routed to another interface (eth1) will not be sent to ipsec-outbound-policy node as IPsec feature is disabled on eth1. Feature-arc allows rte_graph applications to manage such constraints easily Feature arc abstraction allows rte_graph based application to 1. Seamlessly steer packets across feature nodes based on whether feature is enabled or disabled on an interface. Features enabled on one interface may not be enabled on another interface with in a same feature arc. 2. Allow enabling/disabling of features on an interface at runtime, so that if a feature is disabled, packets associated with that interface won't be steered to corresponding feature node. 3. Provides mechanism to hook custom/user-defined nodes to a feature node and allow packet steering from feature node to custom node without changing former's fast path function 4. Allow expressing features in a particular sequential order so that packets are steered in an ordered way across nodes in fast path. For eg: if IPsec and IPv4 features are enabled on an ingress interface, packets must be sent to IPsec inbound policy node first and then to ipv4 lookup node. This patch series adds feature arc library in rte_graph and also adds "ipv4-output" feature arc handling in "ipv4-rewrite" node. Changes in v6: - Rebased to latest main for DPDK-25.03 - Added constructor based feature arc/feature registration - Changed design to handle fast path synchronization via RCU mechanism when any feature is enabled or disabled - Added feature arc specific mbuf dynamic field to carry feature data across nodes - Added feature arc example in app/graph - Programming guide and functional test cases in future versions Nitin Saxena (4): graph: add API to override node process function graph: add feature arc abstraction ip4: add ip4 output feature arc app/graph: add custom feature nodes for ip4 output arc app/graph/commands.list | 6 + app/graph/feature.c | 141 ++ app/graph/feature.h | 13 + app/graph/graph.c | 4 + app/graph/ip4_output_hook.c | 169 ++ app/graph/main.c | 15 +- app/graph/meson.build | 2 + app/graph/module_api.h | 2 + doc/api/doxy-api-index.md | 2 + doc/guides/rel_notes/release_25_03.rst | 10 + lib/graph/graph_feature_arc.c | 1780 ++++++++++++++++++++++ lib/graph/graph_private.h | 15 + lib/graph/meson.build | 4 +- lib/graph/node.c | 23 + lib/graph/rte_graph_feature_arc.h | 552 +++++++ lib/graph/rte_graph_feature_arc_worker.h | 608 ++++++++ lib/graph/version.map | 20 + lib/node/ethdev_ctrl.c | 8 + lib/node/interface_tx_feature.c | 133 ++ lib/node/interface_tx_feature_priv.h | 33 + lib/node/ip4_rewrite.c | 298 +++- lib/node/meson.build | 1 + lib/node/node_private.h | 1 + lib/node/rte_node_ip4_api.h | 4 + 24 files changed, 3838 insertions(+), 6 deletions(-) create mode 100644 app/graph/feature.c create mode 100644 app/graph/feature.h create mode 100644 app/graph/ip4_output_hook.c create mode 100644 lib/graph/graph_feature_arc.c create mode 100644 lib/graph/rte_graph_feature_arc.h create mode 100644 lib/graph/rte_graph_feature_arc_worker.h create mode 100644 lib/node/interface_tx_feature.c create mode 100644 lib/node/interface_tx_feature_priv.h -- 2.43.0 [-- Attachment #2: dpdk_graph_feature_arc_v1.pdf --] [-- Type: application/pdf, Size: 757136 bytes --] ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: Feature arc slides 2025-01-03 14:59 ` Feature arc slides Nitin Saxena @ 2025-01-06 0:15 ` Stephen Hemminger 2025-01-07 12:37 ` Nitin Saxena 0 siblings, 1 reply; 55+ messages in thread From: Stephen Hemminger @ 2025-01-06 0:15 UTC (permalink / raw) To: Nitin Saxena; +Cc: dev On Fri, 3 Jan 2025 20:29:15 +0530 Nitin Saxena <nsaxena16@gmail.com> wrote: > Sending to DPDK community again > > Thanks, > Nitin Why not convert the slides into useful long term documentation in the doc/guides directory. ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: Feature arc slides 2025-01-06 0:15 ` Stephen Hemminger @ 2025-01-07 12:37 ` Nitin Saxena 0 siblings, 0 replies; 55+ messages in thread From: Nitin Saxena @ 2025-01-07 12:37 UTC (permalink / raw) To: stephen; +Cc: dev Hi Stephen, On Mon, Jan 6, 2025 at 5:45 AM Stephen Hemminger <stephen@networkplumber.org> wrote: > > On Fri, 3 Jan 2025 20:29:15 +0530 > Nitin Saxena <nsaxena16@gmail.com> wrote: > > > Sending to DPDK community again > > > > Thanks, > > Nitin > > Why not convert the slides into useful long term documentation in > the doc/guides directory. I will also add a programming guide to this patch. PPT purpose was to describe feature arc concept and, if needed, present to the community Thanks, Nitin ^ permalink raw reply [flat|nested] 55+ messages in thread
* Re: [EXTERNAL] [PATCH v6 0/4] add feature arc in rte_graph [not found] ` <SJ0PR18MB5111B56B4323FB3DFD147801B6152@SJ0PR18MB5111.namprd18.prod.outlook.com> 2025-01-03 14:59 ` Feature arc slides Nitin Saxena @ 2025-01-10 13:59 ` Robin Jarry 1 sibling, 0 replies; 55+ messages in thread From: Robin Jarry @ 2025-01-10 13:59 UTC (permalink / raw) To: Nitin Saxena, Jerin Jacob, Kiran Kumar Kokkilagadda, Nithin Kumar Dabilpuram, Zhirun Yan, Christophe Fontaine Cc: dev, Nitin Saxena Nitin Saxena, Jan 03, 2025 at 15:41: > Hi, > > Please find attached slides explaining feature arc concept and its > usage in graph library > > I can also present these slides to community to facilitate this patch > series review process Hi Nitin, thanks for taking the time to explain your design better. I would really like to have a video meeting to really understand what problem you are trying to solve. And how it could help the grout code base. Let me know how what you think. Cheers ^ permalink raw reply [flat|nested] 55+ messages in thread
end of thread, other threads:[~2025-01-10 13:59 UTC | newest] Thread overview: 55+ messages (download: mbox.gz / follow: Atom feed) -- links below jump to the message on this page -- 2024-09-07 7:31 [RFC PATCH 0/3] add feature arc in rte_graph Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 1/3] graph: add feature arc support Nitin Saxena 2024-09-11 4:41 ` Kiran Kumar Kokkilagadda 2024-10-10 4:42 ` Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 2/3] graph: add feature arc option in graph create Nitin Saxena 2024-09-07 7:31 ` [RFC PATCH 3/3] graph: add IPv4 output feature arc Nitin Saxena 2024-10-08 8:04 ` [RFC PATCH 0/3] add feature arc in rte_graph David Marchand 2024-10-08 14:26 ` [EXTERNAL] " Nitin Saxena 2024-10-14 11:11 ` Nitin Saxena 2024-10-16 9:24 ` David Marchand 2024-10-16 9:38 ` Robin Jarry 2024-10-16 13:50 ` Nitin Saxena 2024-10-17 7:03 ` Nitin Saxena 2024-10-17 7:50 ` Robin Jarry 2024-10-17 8:32 ` [EXTERNAL] " Christophe Fontaine 2024-10-17 10:56 ` Nitin Saxena 2024-10-17 8:48 ` [EXTERNAL] " Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 0/5] " Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 1/5] graph: add feature arc support Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 2/5] graph: add feature arc option in graph create Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 3/5] graph: add IPv4 output feature arc Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 4/5] test/graph_feature_arc: add functional tests Nitin Saxena 2024-10-08 13:30 ` [RFC PATCH v2 5/5] docs: add programming guide for feature arc Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 0/5] add feature arc in rte_graph Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 1/5] graph: add feature arc support Nitin Saxena 2024-10-09 13:29 ` [PATCH v3 2/5] graph: add feature arc option in graph create Nitin Saxena 2024-10-09 13:30 ` [PATCH v3 3/5] graph: add IPv4 output feature arc Nitin Saxena 2024-10-09 13:30 ` [PATCH v3 4/5] test/graph_feature_arc: add functional tests Nitin Saxena 2024-10-09 13:30 ` [PATCH v3 5/5] docs: add programming guide for feature arc Nitin Saxena 2024-10-09 14:21 ` [PATCH v3 0/5] add feature arc in rte_graph Christophe Fontaine 2024-10-10 4:13 ` [EXTERNAL] " Nitin Saxena 2024-10-09 17:37 ` Stephen Hemminger 2024-10-10 4:24 ` [EXTERNAL] " Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 " Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 1/5] graph: add feature arc support Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 2/5] graph: add feature arc option in graph create Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 3/5] graph: add IPv4 output feature arc Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 4/5] test/graph_feature_arc: add functional tests Nitin Saxena 2024-10-10 13:31 ` [PATCH v4 5/5] docs: add programming guide for feature arc Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 0/5] add feature arc in rte_graph Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 1/5] graph: add feature arc support Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 2/5] graph: add feature arc option in graph create Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 3/5] graph: add IPv4 output feature arc Nitin Saxena 2024-10-14 14:33 ` [PATCH v5 4/5] test/graph_feature_arc: add functional tests Nitin Saxena 2024-10-14 19:54 ` Stephen Hemminger 2024-10-14 14:33 ` [PATCH v5 5/5] docs: add programming guide for feature arc Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 0/4] add feature arc in rte_graph Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 1/4] graph: add API to override node process function Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 2/4] graph: add feature arc abstraction Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 3/4] ip4: add ip4 output feature arc Nitin Saxena 2025-01-03 6:06 ` [PATCH v6 4/4] app/graph: add custom feature nodes for ip4 output arc Nitin Saxena [not found] ` <SJ0PR18MB5111B56B4323FB3DFD147801B6152@SJ0PR18MB5111.namprd18.prod.outlook.com> 2025-01-03 14:59 ` Feature arc slides Nitin Saxena 2025-01-06 0:15 ` Stephen Hemminger 2025-01-07 12:37 ` Nitin Saxena 2025-01-10 13:59 ` [EXTERNAL] [PATCH v6 0/4] add feature arc in rte_graph Robin Jarry
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).