Add the framework of modify field flow action and the logic of modify
IPv4 source address action.

Signed-off-by: Chaoyong He <chaoyong...@corigine.com>
Reviewed-by: Long Wu <long...@corigine.com>
Reviewed-by: Peng Zhang <peng.zh...@corigine.com>
---
 doc/guides/nics/features/nfp.ini         |   1 +
 drivers/net/nfp/flower/nfp_flower_flow.c | 172 +++++++++++++++++++++++
 2 files changed, 173 insertions(+)

diff --git a/doc/guides/nics/features/nfp.ini b/doc/guides/nics/features/nfp.ini
index b20049b4f5..5b507cfe94 100644
--- a/doc/guides/nics/features/nfp.ini
+++ b/doc/guides/nics/features/nfp.ini
@@ -51,6 +51,7 @@ drop                 = Y
 jump                 = Y
 mark                 = Y
 meter                = Y
+modify_field         = Y
 of_pop_vlan          = Y
 of_push_vlan         = Y
 of_set_vlan_pcp      = Y
diff --git a/drivers/net/nfp/flower/nfp_flower_flow.c 
b/drivers/net/nfp/flower/nfp_flower_flow.c
index 95f9e8c2d6..8997d67627 100644
--- a/drivers/net/nfp/flower/nfp_flower_flow.c
+++ b/drivers/net/nfp/flower/nfp_flower_flow.c
@@ -1205,6 +1205,129 @@ nfp_flow_action_calculate_mark(struct 
nfp_action_calculate_param *param)
        return 0;
 }
 
+static bool
+nfp_flow_field_id_dst_support(enum rte_flow_field_id field)
+{
+       switch (field) {
+       case RTE_FLOW_FIELD_IPV4_SRC:
+               return true;
+       default:
+               break;
+       }
+
+       return false;
+}
+
+static bool
+nfp_flow_field_id_src_support(enum rte_flow_field_id field)
+{
+       return field == RTE_FLOW_FIELD_POINTER ||
+                       field == RTE_FLOW_FIELD_VALUE;
+}
+
+static uint32_t
+nfp_flow_field_width(enum rte_flow_field_id field,
+               uint32_t inherit)
+{
+       switch (field) {
+       case RTE_FLOW_FIELD_IPV4_SRC:
+               return 32;
+       case RTE_FLOW_FIELD_POINTER:
+               /* FALLTHROUGH */
+       case RTE_FLOW_FIELD_VALUE:
+               return inherit;
+       default:
+               break;
+       }
+
+       return 0;
+}
+
+static bool
+nfp_flow_is_validate_field_data(const struct rte_flow_field_data *data,
+               uint32_t conf_width,
+               uint32_t data_width)
+{
+       if (data->level != 0) {
+               PMD_DRV_LOG(ERR, "The 'level' is not support");
+               return false;
+       }
+
+       if (data->tag_index != 0) {
+               PMD_DRV_LOG(ERR, "The 'tag_index' is not support");
+               return false;
+       }
+
+       if (data->class_id != 0) {
+               PMD_DRV_LOG(ERR, "The 'class_id' is not support");
+               return false;
+       }
+
+       if (data->offset + conf_width > data_width) {
+               PMD_DRV_LOG(ERR, "The 'offset' value is too big");
+               return false;
+       }
+
+       return true;
+}
+
+static int
+nfp_flow_action_calculate_modify_dispatch(struct nfp_action_calculate_param 
*param,
+               enum rte_flow_field_id field)
+{
+       switch (field) {
+       case RTE_FLOW_FIELD_IPV4_SRC:
+               return nfp_flow_action_calculate_ipv4_addr(param);
+       default:
+               break;    /* NOTREACHED */
+       }
+
+       return -ENOTSUP;
+}
+
+static int
+nfp_flow_action_calculate_modify(struct nfp_action_calculate_param *param)
+{
+       uint32_t width;
+       uint32_t dst_width;
+       uint32_t src_width;
+       const struct rte_flow_field_data *dst_data;
+       const struct rte_flow_field_data *src_data;
+       const struct rte_flow_action_modify_field *conf;
+
+       conf = param->action->conf;
+       if (conf == NULL)
+               return -EINVAL;
+
+       dst_data = &conf->dst;
+       src_data = &conf->src;
+       if (!nfp_flow_field_id_dst_support(dst_data->field) ||
+                       !nfp_flow_field_id_src_support(src_data->field)) {
+               PMD_DRV_LOG(ERR, "Not supported field id");
+               return -EINVAL;
+       }
+
+       width = conf->width;
+       if (width == 0) {
+               PMD_DRV_LOG(ERR, "No bits are required to modify");
+               return -EINVAL;
+       }
+
+       dst_width = nfp_flow_field_width(dst_data->field, 0);
+       src_width = nfp_flow_field_width(src_data->field, dst_width);
+       if (width > dst_width || width > src_width) {
+               PMD_DRV_LOG(ERR, "Cannot modify more bits than the width of a 
field");
+               return -EINVAL;
+       }
+
+       if (!nfp_flow_is_validate_field_data(dst_data, width, dst_width)) {
+               PMD_DRV_LOG(ERR, "The dest field data has problem");
+               return -EINVAL;
+       }
+
+       return nfp_flow_action_calculate_modify_dispatch(param, 
dst_data->field);
+}
+
 static nfp_flow_key_calculate_action_fn action_fns[] = {
        [RTE_FLOW_ACTION_TYPE_VOID]             = 
nfp_flow_action_calculate_stub,
        [RTE_FLOW_ACTION_TYPE_DROP]             = 
nfp_flow_action_calculate_stub,
@@ -1235,6 +1358,7 @@ static nfp_flow_key_calculate_action_fn action_fns[] = {
        [RTE_FLOW_ACTION_TYPE_CONNTRACK]        = 
nfp_flow_action_calculate_stub,
        [RTE_FLOW_ACTION_TYPE_MARK]             = 
nfp_flow_action_calculate_mark,
        [RTE_FLOW_ACTION_TYPE_RSS]              = 
nfp_flow_action_calculate_stub,
+       [RTE_FLOW_ACTION_TYPE_MODIFY_FIELD]     = 
nfp_flow_action_calculate_modify,
 };
 
 static int
@@ -4104,6 +4228,53 @@ nfp_flow_action_compile_rss(struct 
nfp_action_compile_param *param)
        return 0;
 }
 
+static int
+nfp_flow_action_compile_modify_dispatch(struct nfp_action_compile_param *param,
+               enum rte_flow_field_id field)
+{
+       switch (field) {
+       case RTE_FLOW_FIELD_IPV4_SRC:
+               return nfp_flow_action_compile_ipv4_src(param);
+       default:
+               break;    /* NOTREACHED */
+       }
+
+       return -ENOTSUP;
+}
+
+static int
+nfp_flow_action_compile_modify(struct nfp_action_compile_param *param)
+{
+       int ret;
+       struct rte_flow_action action = {};
+       const struct rte_flow_action *action_old;
+       const struct rte_flow_action_modify_field *conf;
+
+       conf = param->action->conf;
+
+       if (conf->src.field == RTE_FLOW_FIELD_POINTER) {
+               action.conf = conf->src.pvalue;
+       } else if (conf->src.field == RTE_FLOW_FIELD_VALUE) {
+               action.conf = (void *)(uintptr_t)&conf->src.value;
+       } else {
+               PMD_DRV_LOG(ERR, "The SRC field of flow modify is not right");
+               return -EINVAL;
+       }
+
+       /* Store the old action pointer */
+       action_old = param->action;
+
+       param->action = &action;
+       ret = nfp_flow_action_compile_modify_dispatch(param, conf->dst.field);
+       if (ret != 0)
+               PMD_DRV_LOG(ERR, "Something wrong when modify dispatch");
+
+       /* Reload the old action pointer */
+       param->action = action_old;
+
+       return ret;
+}
+
 static nfp_flow_action_compile_fn action_compile_fns[] = {
        [RTE_FLOW_ACTION_TYPE_VOID]             = nfp_flow_action_compile_stub,
        [RTE_FLOW_ACTION_TYPE_DROP]             = nfp_flow_action_compile_drop,
@@ -4134,6 +4305,7 @@ static nfp_flow_action_compile_fn action_compile_fns[] = {
        [RTE_FLOW_ACTION_TYPE_CONNTRACK]        = nfp_flow_action_compile_stub,
        [RTE_FLOW_ACTION_TYPE_MARK]             = nfp_flow_action_compile_mark,
        [RTE_FLOW_ACTION_TYPE_RSS]              = nfp_flow_action_compile_rss,
+       [RTE_FLOW_ACTION_TYPE_MODIFY_FIELD]     = 
nfp_flow_action_compile_modify,
 };
 
 static int
-- 
2.39.1

Reply via email to