Since [1] flow API forbids usage of direction attributes in transfer
flow rules. This patch adapts mlx5 PMD to this requirement.

>From this patch, flow rule validation in mxl5 PMD will reject transfer
flow rules with any of the direction attributes set
(i.e. 'ingress' or 'egress').
As a result flow rule can only have one of 'ingress', 'egress' or
'transfer' attributes set.

This patch also changes the following:

- Control flow rules used in FDB are 'transfer' only.
- Checks which assumed that 'transfer' can be used
  with 'ingress' and 'egress' are reduced to just checking
  for direction attributes, since all attributes are exclusive.
- Flow rules for updating flow_tag are created for both ingress
  and transfer flow rules which have MARK action.
- Moves mlx5_flow_validate_attributes() function from generic flow
  implementation to legacy Verbs flow engine implementation,
  since it was used only there. Function is renamed accordingly.
  Also removes checking if E-Switch uses DV in that
  function, since if legacy Verbs flow engine is used,
  then that is always not the case.

[1] commit bd2a4d4b2e3a ("ethdev: forbid direction attribute in
                         transfer flow rules")

Signed-off-by: Dariusz Sosnowski <dsosnow...@nvidia.com>
Acked-by: Matan Azrad <ma...@nvidia.com>
---
 drivers/net/mlx5/mlx5_flow.c       | 59 +++---------------------------
 drivers/net/mlx5/mlx5_flow.h       |  3 --
 drivers/net/mlx5/mlx5_flow_dv.c    | 42 ++++++++-------------
 drivers/net/mlx5/mlx5_flow_verbs.c | 50 ++++++++++++++++++++++++-
 4 files changed, 71 insertions(+), 83 deletions(-)

diff --git a/drivers/net/mlx5/mlx5_flow.c b/drivers/net/mlx5/mlx5_flow.c
index 8e7d649d15..921e419d05 100644
--- a/drivers/net/mlx5/mlx5_flow.c
+++ b/drivers/net/mlx5/mlx5_flow.c
@@ -2299,53 +2299,6 @@ mlx5_validate_action_ct(struct rte_eth_dev *dev,
        return 0;
 }
 
-/**
- * Verify the @p attributes will be correctly understood by the NIC and store
- * them in the @p flow if everything is correct.
- *
- * @param[in] dev
- *   Pointer to the Ethernet device structure.
- * @param[in] attributes
- *   Pointer to flow attributes
- * @param[out] error
- *   Pointer to error structure.
- *
- * @return
- *   0 on success, a negative errno value otherwise and rte_errno is set.
- */
-int
-mlx5_flow_validate_attributes(struct rte_eth_dev *dev,
-                             const struct rte_flow_attr *attributes,
-                             struct rte_flow_error *error)
-{
-       struct mlx5_priv *priv = dev->data->dev_private;
-       uint32_t priority_max = priv->sh->flow_max_priority - 1;
-
-       if (attributes->group)
-               return rte_flow_error_set(error, ENOTSUP,
-                                         RTE_FLOW_ERROR_TYPE_ATTR_GROUP,
-                                         NULL, "groups is not supported");
-       if (attributes->priority != MLX5_FLOW_LOWEST_PRIO_INDICATOR &&
-           attributes->priority >= priority_max)
-               return rte_flow_error_set(error, ENOTSUP,
-                                         RTE_FLOW_ERROR_TYPE_ATTR_PRIORITY,
-                                         NULL, "priority out of range");
-       if (attributes->egress)
-               return rte_flow_error_set(error, ENOTSUP,
-                                         RTE_FLOW_ERROR_TYPE_ATTR_EGRESS, NULL,
-                                         "egress is not supported");
-       if (attributes->transfer && !priv->sh->config.dv_esw_en)
-               return rte_flow_error_set(error, ENOTSUP,
-                                         RTE_FLOW_ERROR_TYPE_ATTR_TRANSFER,
-                                         NULL, "transfer is not supported");
-       if (!attributes->ingress)
-               return rte_flow_error_set(error, EINVAL,
-                                         RTE_FLOW_ERROR_TYPE_ATTR_INGRESS,
-                                         NULL,
-                                         "ingress attribute is mandatory");
-       return 0;
-}
-
 /**
  * Validate ICMP6 item.
  *
@@ -6342,7 +6295,7 @@ flow_create_split_metadata(struct rte_eth_dev *dev,
                        ret = -rte_errno;
                        goto exit;
                }
-       } else if (attr->egress && !attr->transfer) {
+       } else if (attr->egress) {
                /*
                 * All the actions on NIC Tx should have a metadata register
                 * copy action to copy reg_a from WQE to reg_c[meta]
@@ -7061,7 +7014,7 @@ flow_list_create(struct rte_eth_dev *dev, enum 
mlx5_flow_type type,
                    flow->drv_type < MLX5_FLOW_TYPE_MAX);
        memset(rss_desc, 0, offsetof(struct mlx5_flow_rss_desc, queue));
        /* RSS Action only works on NIC RX domain */
-       if (attr->ingress && !attr->transfer)
+       if (attr->ingress)
                rss = flow_get_rss_action(dev, p_actions_rx);
        if (rss) {
                if (flow_rss_workspace_adjust(wks, rss_desc, rss->queue_num))
@@ -7159,11 +7112,11 @@ flow_list_create(struct rte_eth_dev *dev, enum 
mlx5_flow_type type,
         * we might create the extra rte_flow for each unique
         * MARK/FLAG action ID.
         *
-        * The table is updated for ingress Flows only, because
+        * The table is updated for ingress and transfer flows only, because
         * the egress Flows belong to the different device and
         * copy table should be updated in peer NIC Rx domain.
         */
-       if (attr->ingress &&
+       if ((attr->ingress || attr->transfer) &&
            (external || attr->group != MLX5_FLOW_MREG_CP_TABLE_GROUP)) {
                ret = flow_mreg_update_copy_table(dev, flow, actions, error);
                if (ret)
@@ -7235,7 +7188,7 @@ mlx5_flow_create_esw_table_zero_flow(struct rte_eth_dev 
*dev)
        const struct rte_flow_attr attr = {
                .group = 0,
                .priority = 0,
-               .ingress = 1,
+               .ingress = 0,
                .egress = 0,
                .transfer = 1,
        };
@@ -7279,7 +7232,7 @@ mlx5_flow_create_devx_sq_miss_flow(struct rte_eth_dev 
*dev, uint32_t sq_num)
        struct rte_flow_attr attr = {
                .group = 0,
                .priority = MLX5_FLOW_LOWEST_PRIO_INDICATOR,
-               .ingress = 1,
+               .ingress = 0,
                .egress = 0,
                .transfer = 1,
        };
diff --git a/drivers/net/mlx5/mlx5_flow.h b/drivers/net/mlx5/mlx5_flow.h
index da9b65d8fe..2398e44ea0 100644
--- a/drivers/net/mlx5/mlx5_flow.h
+++ b/drivers/net/mlx5/mlx5_flow.h
@@ -2228,9 +2228,6 @@ int mlx5_flow_validate_action_rss(const struct 
rte_flow_action *action,
 int mlx5_flow_validate_action_default_miss(uint64_t action_flags,
                                const struct rte_flow_attr *attr,
                                struct rte_flow_error *error);
-int mlx5_flow_validate_attributes(struct rte_eth_dev *dev,
-                                 const struct rte_flow_attr *attributes,
-                                 struct rte_flow_error *error);
 int mlx5_flow_item_acceptable(const struct rte_flow_item *item,
                              const uint8_t *mask,
                              const uint8_t *nic_mask,
diff --git a/drivers/net/mlx5/mlx5_flow_dv.c b/drivers/net/mlx5/mlx5_flow_dv.c
index 1e52278191..2be62cbdcb 100644
--- a/drivers/net/mlx5/mlx5_flow_dv.c
+++ b/drivers/net/mlx5/mlx5_flow_dv.c
@@ -3384,8 +3384,7 @@ flow_dv_validate_action_set_tag(struct rte_eth_dev *dev,
        ret = mlx5_flow_get_reg_id(dev, MLX5_APP_TAG, conf->index, error);
        if (ret < 0)
                return ret;
-       if (!attr->transfer && attr->ingress &&
-           (action_flags & terminal_action_flags))
+       if (attr->ingress && (action_flags & terminal_action_flags))
                return rte_flow_error_set(error, EINVAL,
                                          RTE_FLOW_ERROR_TYPE_ACTION, action,
                                          "set_tag has no effect"
@@ -5942,7 +5941,7 @@ flow_dv_validate_action_sample(uint64_t *action_flags,
                                                  "action");
                }
        }
-       if (attr->ingress && !attr->transfer) {
+       if (attr->ingress) {
                if (!(sub_action_flags & (MLX5_FLOW_ACTION_QUEUE |
                                          MLX5_FLOW_ACTION_RSS)))
                        return rte_flow_error_set(error, EINVAL,
@@ -5950,7 +5949,7 @@ flow_dv_validate_action_sample(uint64_t *action_flags,
                                                  NULL,
                                                  "Ingress must has a dest "
                                                  "QUEUE for Sample");
-       } else if (attr->egress && !attr->transfer) {
+       } else if (attr->egress) {
                return rte_flow_error_set(error, ENOTSUP,
                                          RTE_FLOW_ERROR_TYPE_ACTION,
                                          NULL,
@@ -5995,8 +5994,7 @@ flow_dv_validate_action_sample(uint64_t *action_flags,
                                                  NULL, "encap and decap "
                                                  "combination aren't "
                                                  "supported");
-               if (!attr->transfer && attr->ingress && (sub_action_flags &
-                                                       MLX5_FLOW_ACTION_ENCAP))
+               if (attr->ingress && (sub_action_flags & 
MLX5_FLOW_ACTION_ENCAP))
                        return rte_flow_error_set(error, ENOTSUP,
                                                  RTE_FLOW_ERROR_TYPE_ACTION,
                                                  NULL, "encap is not supported"
@@ -6820,23 +6818,16 @@ flow_dv_validate_attributes(struct rte_eth_dev *dev,
                                          RTE_FLOW_ERROR_TYPE_ATTR_PRIORITY,
                                          NULL,
                                          "priority out of range");
-       if (attributes->transfer) {
-               if (!priv->sh->config.dv_esw_en)
-                       return rte_flow_error_set
-                               (error, ENOTSUP,
-                                RTE_FLOW_ERROR_TYPE_UNSPECIFIED, NULL,
-                                "E-Switch dr is not supported");
-               if (attributes->egress)
-                       return rte_flow_error_set
-                               (error, ENOTSUP,
-                                RTE_FLOW_ERROR_TYPE_ATTR_EGRESS, attributes,
-                                "egress is not supported");
-       }
-       if (!(attributes->egress ^ attributes->ingress))
+       if (attributes->transfer && !priv->sh->config.dv_esw_en)
                return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_UNSPECIFIED, NULL,
+                                         "E-Switch dr is not supported");
+       if (attributes->ingress + attributes->egress + attributes->transfer != 
1) {
+               return rte_flow_error_set(error, EINVAL,
                                          RTE_FLOW_ERROR_TYPE_ATTR, NULL,
                                          "must specify exactly one of "
-                                         "ingress or egress");
+                                         "ingress, egress or transfer");
+       }
        return ret;
 }
 
@@ -8153,11 +8144,11 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct 
rte_flow_attr *attr,
                                        RTE_FLOW_ERROR_TYPE_ACTION, NULL,
                                        "tunnel set decap rule must terminate "
                                        "with JUMP");
-               if (!attr->ingress)
+               if (attr->egress)
                        return rte_flow_error_set
                                        (error, EINVAL,
                                        RTE_FLOW_ERROR_TYPE_ACTION, NULL,
-                                       "tunnel flows for ingress traffic 
only");
+                                       "tunnel flows for ingress and transfer 
traffic only");
        }
        if (action_flags & MLX5_FLOW_ACTION_TUNNEL_MATCH) {
                uint64_t bad_actions_mask = MLX5_FLOW_ACTION_JUMP    |
@@ -8262,7 +8253,7 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct 
rte_flow_attr *attr,
                                                          "push VLAN action not 
supported "
                                                          "for ingress");
                }
-               if (!attr->transfer && attr->ingress) {
+               if (attr->ingress) {
                        if (action_flags & MLX5_FLOW_ACTION_ENCAP)
                                return rte_flow_error_set
                                                (error, ENOTSUP,
@@ -8350,8 +8341,7 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct 
rte_flow_attr *attr,
         * hairpin default egress flow with TX_QUEUE item, other flows not
         * work due to metadata regC0 mismatch.
         */
-       if ((!attr->transfer && attr->egress) && priv->representor &&
-           !(item_flags & MLX5_FLOW_ITEM_SQ))
+       if (attr->egress && priv->representor && !(item_flags & 
MLX5_FLOW_ITEM_SQ))
                return rte_flow_error_set(error, EINVAL,
                                          RTE_FLOW_ERROR_TYPE_ITEM,
                                          NULL,
@@ -13673,7 +13663,7 @@ flow_dv_translate_items_sws(struct rte_eth_dev *dev,
            !(wks.item_flags & MLX5_FLOW_ITEM_REPRESENTED_PORT) &&
            !(wks.item_flags & MLX5_FLOW_ITEM_PORT_REPRESENTOR) &&
            priv->sh->esw_mode &&
-           !(attr->egress && !attr->transfer) &&
+           !attr->egress &&
            attr->group != MLX5_FLOW_MREG_CP_TABLE_GROUP) {
                if (flow_dv_translate_item_port_id_all(dev, match_mask,
                                                   match_value, NULL, attr))
diff --git a/drivers/net/mlx5/mlx5_flow_verbs.c 
b/drivers/net/mlx5/mlx5_flow_verbs.c
index 81a33ddf09..8da2bd1f78 100644
--- a/drivers/net/mlx5/mlx5_flow_verbs.c
+++ b/drivers/net/mlx5/mlx5_flow_verbs.c
@@ -1207,6 +1207,54 @@ flow_verbs_translate_action_count(struct mlx5_flow 
*dev_flow,
        return 0;
 }
 
+/**
+ * Validates @p attributes of the flow rule.
+ *
+ * This function is used if and only if legacy Verbs flow engine is used.
+ *
+ * @param[in] dev
+ *   Pointer to the Ethernet device structure.
+ * @param[in] attributes
+ *   Pointer to flow attributes
+ * @param[out] error
+ *   Pointer to error structure.
+ *
+ * @return
+ *   0 on success, a negative errno value otherwise and rte_errno is set.
+ */
+static int
+flow_verbs_validate_attributes(struct rte_eth_dev *dev,
+                              const struct rte_flow_attr *attributes,
+                              struct rte_flow_error *error)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       uint32_t priority_max = priv->sh->flow_max_priority - 1;
+
+       if (attributes->group)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ATTR_GROUP,
+                                         NULL, "groups is not supported");
+       if (attributes->priority != MLX5_FLOW_LOWEST_PRIO_INDICATOR &&
+           attributes->priority >= priority_max)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ATTR_PRIORITY,
+                                         NULL, "priority out of range");
+       if (attributes->egress)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ATTR_EGRESS, NULL,
+                                         "egress is not supported");
+       if (attributes->transfer)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ATTR_TRANSFER,
+                                         NULL, "transfer is not supported");
+       if (!attributes->ingress)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ATTR_INGRESS,
+                                         NULL,
+                                         "ingress attribute is mandatory");
+       return 0;
+}
+
 /**
  * Internal validation function. For validating both actions and items.
  *
@@ -1249,7 +1297,7 @@ flow_verbs_validate(struct rte_eth_dev *dev,
 
        if (items == NULL)
                return -1;
-       ret = mlx5_flow_validate_attributes(dev, attr, error);
+       ret = flow_verbs_validate_attributes(dev, attr, error);
        if (ret < 0)
                return ret;
        is_root = ret;
-- 
2.25.1

Reply via email to