Add testpmd CLI interface for the flow update API. flow queue 1 update 2 rule 3 actions_template 4 postpone yes pattern end actions queue index 3 / end
Signed-off-by: Alexander Kozyrev <akozy...@nvidia.com> --- app/test-pmd/cmdline_flow.c | 29 ++++++++++++ app/test-pmd/config.c | 92 ++++++++++++++++++++++++++++++++++++- app/test-pmd/testpmd.h | 5 ++ 3 files changed, 125 insertions(+), 1 deletion(-) diff --git a/app/test-pmd/cmdline_flow.c b/app/test-pmd/cmdline_flow.c index 00bac0a60d..5771281125 100644 --- a/app/test-pmd/cmdline_flow.c +++ b/app/test-pmd/cmdline_flow.c @@ -132,6 +132,7 @@ enum index { /* Queue arguments. */ QUEUE_CREATE, QUEUE_DESTROY, + QUEUE_UPDATE, QUEUE_AGED, QUEUE_INDIRECT_ACTION, @@ -146,6 +147,9 @@ enum index { QUEUE_DESTROY_ID, QUEUE_DESTROY_POSTPONE, + /* Queue update arguments. */ + QUEUE_UPDATE_ID, + /* Queue indirect action arguments */ QUEUE_INDIRECT_ACTION_CREATE, QUEUE_INDIRECT_ACTION_LIST_CREATE, @@ -1312,6 +1316,7 @@ static const enum index next_table_destroy_attr[] = { static const enum index next_queue_subcmd[] = { QUEUE_CREATE, QUEUE_DESTROY, + QUEUE_UPDATE, QUEUE_AGED, QUEUE_INDIRECT_ACTION, ZERO, @@ -3408,6 +3413,14 @@ static const struct token token_list[] = { .args = ARGS(ARGS_ENTRY(struct buffer, queue)), .call = parse_qo_destroy, }, + [QUEUE_UPDATE] = { + .name = "update", + .help = "update a flow rule", + .next = NEXT(NEXT_ENTRY(QUEUE_UPDATE_ID), + NEXT_ENTRY(COMMON_QUEUE_ID)), + .args = ARGS(ARGS_ENTRY(struct buffer, queue)), + .call = parse_qo, + }, [QUEUE_AGED] = { .name = "aged", .help = "list and destroy aged flows", @@ -3484,6 +3497,15 @@ static const struct token token_list[] = { args.destroy.rule)), .call = parse_qo_destroy, }, + [QUEUE_UPDATE_ID] = { + .name = "rule", + .help = "specify rule id to update", + .next = NEXT(NEXT_ENTRY(QUEUE_ACTIONS_TEMPLATE), + NEXT_ENTRY(COMMON_UNSIGNED)), + .args = ARGS(ARGS_ENTRY(struct buffer, + args.vc.rule_id)), + .call = parse_qo, + }, /* Queue indirect action arguments */ [QUEUE_INDIRECT_ACTION_CREATE] = { .name = "create", @@ -10207,6 +10229,7 @@ parse_qo(struct context *ctx, const struct token *token, } switch (ctx->curr) { case QUEUE_CREATE: + case QUEUE_UPDATE: out->command = ctx->curr; ctx->objdata = 0; ctx->object = out; @@ -10218,6 +10241,7 @@ parse_qo(struct context *ctx, const struct token *token, case QUEUE_ACTIONS_TEMPLATE: case QUEUE_CREATE_POSTPONE: case QUEUE_RULE_ID: + case QUEUE_UPDATE_ID: return len; case ITEM_PATTERN: out->args.vc.pattern = @@ -12233,6 +12257,11 @@ cmd_flow_parsed(const struct buffer *in) in->args.destroy.rule_n, in->args.destroy.rule); break; + case QUEUE_UPDATE: + port_queue_flow_update(in->port, in->queue, in->postpone, + in->args.vc.rule_id, in->args.vc.act_templ_id, + in->args.vc.actions); + break; case PUSH: port_queue_flow_push(in->port, in->queue); break; diff --git a/app/test-pmd/config.c b/app/test-pmd/config.c index 2cf01d09b3..d9437bd164 100644 --- a/app/test-pmd/config.c +++ b/app/test-pmd/config.c @@ -2799,6 +2799,7 @@ port_queue_flow_create(portid_t port_id, queueid_t queue_id, pf->next = port->flow_list; pf->id = id; + pf->table = pt; pf->flow = flow; job->pf = pf; port->flow_list = pf; @@ -2905,6 +2906,94 @@ queue_action_list_handle_create(portid_t port_id, uint32_t queue_id, job, error); } +/** Enqueue update flow rule operation. */ +int +port_queue_flow_update(portid_t port_id, queueid_t queue_id, + bool postpone, uint32_t rule_idx, uint32_t actions_idx, + const struct rte_flow_action *actions) +{ + struct rte_flow_op_attr op_attr = { .postpone = postpone }; + struct rte_port *port; + struct port_flow *pf, *uf; + struct port_flow **tmp; + struct port_table *pt; + bool found; + struct rte_flow_error error = { RTE_FLOW_ERROR_TYPE_NONE, NULL, NULL }; + struct rte_flow_action_age *age = age_action_get(actions); + struct queue_job *job; + + if (port_id_is_invalid(port_id, ENABLED_WARN) || + port_id == (portid_t)RTE_PORT_ALL) + return -EINVAL; + port = &ports[port_id]; + + if (queue_id >= port->queue_nb) { + printf("Queue #%u is invalid\n", queue_id); + return -EINVAL; + } + + found = false; + tmp = &port->flow_list; + while (*tmp) { + pf = *tmp; + if (rule_idx == pf->id) { + found = true; + break; + } + tmp = &(*tmp)->next; + } + if (!found) { + printf("Flow rule #%u is invalid\n", rule_idx); + return -EINVAL; + } + + pt = pf->table; + if (actions_idx >= pt->nb_actions_templates) { + printf("Actions template index #%u is invalid," + " %u templates present in the table\n", + actions_idx, pt->nb_actions_templates); + return -EINVAL; + } + + job = calloc(1, sizeof(*job)); + if (!job) { + printf("Queue flow create job allocate failed\n"); + return -ENOMEM; + } + job->type = QUEUE_JOB_TYPE_FLOW_UPDATE; + + uf = port_flow_new(&pt->flow_attr, pf->rule.pattern_ro, actions, &error); + if (!uf) { + free(job); + return port_flow_complain(&error); + } + + if (age) { + uf->age_type = ACTION_AGE_CONTEXT_TYPE_FLOW; + age->context = &uf->age_type; + } + + /* + * Poisoning to make sure PMD update it in case of error. + */ + memset(&error, 0x44, sizeof(error)); + if (rte_flow_async_actions_update(port_id, queue_id, &op_attr, pf->flow, + actions, actions_idx, job, &error)) { + free(uf); + free(job); + return port_flow_complain(&error); + } + uf->next = pf->next; + uf->id = pf->id; + uf->table = pt; + uf->flow = pf->flow; + *tmp = uf; + job->pf = pf; + + printf("Flow rule #%u update enqueued\n", pf->id); + return 0; +} + /** Enqueue indirect action create operation. */ int port_queue_action_handle_create(portid_t port_id, uint32_t queue_id, @@ -3394,7 +3483,8 @@ port_queue_flow_pull(portid_t port_id, queueid_t queue_id) if (res[i].status == RTE_FLOW_OP_SUCCESS) success++; job = (struct queue_job *)res[i].user_data; - if (job->type == QUEUE_JOB_TYPE_FLOW_DESTROY) + if (job->type == QUEUE_JOB_TYPE_FLOW_DESTROY || + job->type == QUEUE_JOB_TYPE_FLOW_UPDATE) free(job->pf); else if (job->type == QUEUE_JOB_TYPE_ACTION_DESTROY) free(job->pia); diff --git a/app/test-pmd/testpmd.h b/app/test-pmd/testpmd.h index 5eeb50f67c..1761768add 100644 --- a/app/test-pmd/testpmd.h +++ b/app/test-pmd/testpmd.h @@ -110,6 +110,7 @@ enum { enum { QUEUE_JOB_TYPE_FLOW_CREATE, QUEUE_JOB_TYPE_FLOW_DESTROY, + QUEUE_JOB_TYPE_FLOW_UPDATE, QUEUE_JOB_TYPE_ACTION_CREATE, QUEUE_JOB_TYPE_ACTION_DESTROY, QUEUE_JOB_TYPE_ACTION_UPDATE, @@ -225,6 +226,7 @@ struct port_flow { struct port_flow *next; /**< Next flow in list. */ struct port_flow *tmp; /**< Temporary linking. */ uint32_t id; /**< Flow rule ID. */ + struct port_table *table; /**< Flow table. */ struct rte_flow *flow; /**< Opaque flow object returned by PMD. */ struct rte_flow_conv_rule rule; /**< Saved flow rule description. */ enum age_action_context_type age_type; /**< Age action context type. */ @@ -983,6 +985,9 @@ int port_queue_flow_create(portid_t port_id, queueid_t queue_id, const struct rte_flow_action *actions); int port_queue_flow_destroy(portid_t port_id, queueid_t queue_id, bool postpone, uint32_t n, const uint32_t *rule); +int port_queue_flow_update(portid_t port_id, queueid_t queue_id, + bool postpone, uint32_t rule_idx, uint32_t actions_idx, + const struct rte_flow_action *actions); int port_queue_action_handle_create(portid_t port_id, uint32_t queue_id, bool postpone, uint32_t id, const struct rte_flow_indir_action_conf *conf, -- 2.18.2