- 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 <nsax...@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