net/mlx5: extend flow metadata support
[dpdk.git] / drivers / net / mlx5 / mlx5_flow_dv.c
index 5f3ea1f..204e15e 100644 (file)
@@ -872,10 +872,12 @@ flow_dv_convert_action_set_reg
                return rte_flow_error_set(error, EINVAL,
                                          RTE_FLOW_ERROR_TYPE_ACTION, NULL,
                                          "too many items to modify");
+       assert(conf->id != REG_NONE);
+       assert(conf->id < RTE_DIM(reg_to_field));
        actions[i].action_type = MLX5_MODIFICATION_TYPE_SET;
        actions[i].field = reg_to_field[conf->id];
        actions[i].data0 = rte_cpu_to_be_32(actions[i].data0);
-       actions[i].data1 = conf->data;
+       actions[i].data1 = rte_cpu_to_be_32(conf->data);
        ++i;
        resource->actions_num = i;
        if (!resource->actions_num)
@@ -885,6 +887,52 @@ flow_dv_convert_action_set_reg
        return 0;
 }
 
+/**
+ * Convert SET_TAG action to DV specification.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in,out] resource
+ *   Pointer to the modify-header resource.
+ * @param[in] conf
+ *   Pointer to action specification.
+ * @param[out] error
+ *   Pointer to the error structure.
+ *
+ * @return
+ *   0 on success, a negative errno value otherwise and rte_errno is set.
+ */
+static int
+flow_dv_convert_action_set_tag
+                       (struct rte_eth_dev *dev,
+                        struct mlx5_flow_dv_modify_hdr_resource *resource,
+                        const struct rte_flow_action_set_tag *conf,
+                        struct rte_flow_error *error)
+{
+       rte_be32_t data = rte_cpu_to_be_32(conf->data);
+       rte_be32_t mask = rte_cpu_to_be_32(conf->mask);
+       struct rte_flow_item item = {
+               .spec = &data,
+               .mask = &mask,
+       };
+       struct field_modify_info reg_c_x[] = {
+               [1] = {0, 0, 0},
+       };
+       enum mlx5_modification_field reg_type;
+       int ret;
+
+       ret = mlx5_flow_get_reg_id(dev, MLX5_APP_TAG, conf->index, error);
+       if (ret < 0)
+               return ret;
+       assert(ret != REG_NONE);
+       assert((unsigned int)ret < RTE_DIM(reg_to_field));
+       reg_type = reg_to_field[ret];
+       assert(reg_type > 0);
+       reg_c_x[0] = (struct field_modify_info){4, 0, reg_type};
+       return flow_dv_convert_modify_action(&item, reg_c_x, NULL, resource,
+                                            MLX5_MODIFICATION_TYPE_SET, error);
+}
+
 /**
  * Convert internal COPY_REG action to DV specification.
  *
@@ -961,6 +1009,222 @@ flow_dv_convert_action_copy_mreg(struct rte_eth_dev *dev,
                                             error);
 }
 
+/**
+ * Convert MARK action to DV specification. This routine is used
+ * in extensive metadata only and requires metadata register to be
+ * handled. In legacy mode hardware tag resource is engaged.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] conf
+ *   Pointer to MARK action specification.
+ * @param[in,out] resource
+ *   Pointer to the modify-header resource.
+ * @param[out] error
+ *   Pointer to the error structure.
+ *
+ * @return
+ *   0 on success, a negative errno value otherwise and rte_errno is set.
+ */
+static int
+flow_dv_convert_action_mark(struct rte_eth_dev *dev,
+                           const struct rte_flow_action_mark *conf,
+                           struct mlx5_flow_dv_modify_hdr_resource *resource,
+                           struct rte_flow_error *error)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       rte_be32_t mask = rte_cpu_to_be_32(MLX5_FLOW_MARK_MASK &
+                                          priv->sh->dv_mark_mask);
+       rte_be32_t data = rte_cpu_to_be_32(conf->id) & mask;
+       struct rte_flow_item item = {
+               .spec = &data,
+               .mask = &mask,
+       };
+       struct field_modify_info reg_c_x[] = {
+               {4, 0, 0}, /* dynamic instead of MLX5_MODI_META_REG_C_1. */
+               {0, 0, 0},
+       };
+       enum modify_reg reg;
+
+       if (!mask)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION_CONF,
+                                         NULL, "zero mark action mask");
+       reg = mlx5_flow_get_reg_id(dev, MLX5_FLOW_MARK, 0, error);
+       if (reg < 0)
+               return reg;
+       assert(reg > 0);
+       reg_c_x[0].id = reg_to_field[reg];
+       return flow_dv_convert_modify_action(&item, reg_c_x, NULL, resource,
+                                            MLX5_MODIFICATION_TYPE_SET, error);
+}
+
+/**
+ * Get metadata register index for specified steering domain.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] attr
+ *   Attributes of flow to determine steering domain.
+ * @param[out] error
+ *   Pointer to the error structure.
+ *
+ * @return
+ *   positive index on success, a negative errno value otherwise
+ *   and rte_errno is set.
+ */
+static enum modify_reg
+flow_dv_get_metadata_reg(struct rte_eth_dev *dev,
+                        const struct rte_flow_attr *attr,
+                        struct rte_flow_error *error)
+{
+       enum modify_reg reg =
+               mlx5_flow_get_reg_id(dev, attr->transfer ?
+                                         MLX5_METADATA_FDB :
+                                           attr->egress ?
+                                           MLX5_METADATA_TX :
+                                           MLX5_METADATA_RX, 0, error);
+       if (reg < 0)
+               return rte_flow_error_set(error,
+                                         ENOTSUP, RTE_FLOW_ERROR_TYPE_ITEM,
+                                         NULL, "unavailable "
+                                         "metadata register");
+       return reg;
+}
+
+/**
+ * Convert SET_META action to DV specification.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in,out] resource
+ *   Pointer to the modify-header resource.
+ * @param[in] attr
+ *   Attributes of flow that includes this item.
+ * @param[in] conf
+ *   Pointer to action specification.
+ * @param[out] error
+ *   Pointer to the error structure.
+ *
+ * @return
+ *   0 on success, a negative errno value otherwise and rte_errno is set.
+ */
+static int
+flow_dv_convert_action_set_meta
+                       (struct rte_eth_dev *dev,
+                        struct mlx5_flow_dv_modify_hdr_resource *resource,
+                        const struct rte_flow_attr *attr,
+                        const struct rte_flow_action_set_meta *conf,
+                        struct rte_flow_error *error)
+{
+       uint32_t data = conf->data;
+       uint32_t mask = conf->mask;
+       struct rte_flow_item item = {
+               .spec = &data,
+               .mask = &mask,
+       };
+       struct field_modify_info reg_c_x[] = {
+               [1] = {0, 0, 0},
+       };
+       enum modify_reg reg = flow_dv_get_metadata_reg(dev, attr, error);
+
+       if (reg < 0)
+               return reg;
+       /*
+        * In datapath code there is no endianness
+        * coversions for perfromance reasons, all
+        * pattern conversions are done in rte_flow.
+        */
+       if (reg == REG_C_0) {
+               struct mlx5_priv *priv = dev->data->dev_private;
+               uint32_t msk_c0 = priv->sh->dv_regc0_mask;
+               uint32_t shl_c0;
+
+               assert(msk_c0);
+#if RTE_BYTE_ORDER == RTE_BIG_ENDIAN
+               shl_c0 = rte_bsf32(msk_c0);
+#else
+               shl_c0 = sizeof(msk_c0) * CHAR_BIT - rte_fls_u32(msk_c0);
+#endif
+               mask <<= shl_c0;
+               data <<= shl_c0;
+               assert(!(~msk_c0 & rte_cpu_to_be_32(mask)));
+       }
+       reg_c_x[0] = (struct field_modify_info){4, 0, reg_to_field[reg]};
+       /* The routine expects parameters in memory as big-endian ones. */
+       return flow_dv_convert_modify_action(&item, reg_c_x, NULL, resource,
+                                            MLX5_MODIFICATION_TYPE_SET, error);
+}
+
+/**
+ * Validate MARK item.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] item
+ *   Item specification.
+ * @param[in] attr
+ *   Attributes of flow that includes this item.
+ * @param[out] error
+ *   Pointer to error structure.
+ *
+ * @return
+ *   0 on success, a negative errno value otherwise and rte_errno is set.
+ */
+static int
+flow_dv_validate_item_mark(struct rte_eth_dev *dev,
+                          const struct rte_flow_item *item,
+                          const struct rte_flow_attr *attr __rte_unused,
+                          struct rte_flow_error *error)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_dev_config *config = &priv->config;
+       const struct rte_flow_item_mark *spec = item->spec;
+       const struct rte_flow_item_mark *mask = item->mask;
+       const struct rte_flow_item_mark nic_mask = {
+               .id = priv->sh->dv_mark_mask,
+       };
+       int ret;
+
+       if (config->dv_xmeta_en == MLX5_XMETA_MODE_LEGACY)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                         "extended metadata feature"
+                                         " isn't enabled");
+       if (!mlx5_flow_ext_mreg_supported(dev))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                         "extended metadata register"
+                                         " isn't supported");
+       if (!nic_mask.id)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                         "extended metadata register"
+                                         " isn't available");
+       ret = mlx5_flow_get_reg_id(dev, MLX5_FLOW_MARK, 0, error);
+       if (ret < 0)
+               return ret;
+       if (!spec)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ITEM_SPEC,
+                                         item->spec,
+                                         "data cannot be empty");
+       if (spec->id >= (MLX5_FLOW_MARK_MAX & nic_mask.id))
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION_CONF,
+                                         &spec->id,
+                                         "mark id exceeds the limit");
+       if (!mask)
+               mask = &nic_mask;
+       ret = mlx5_flow_item_acceptable(item, (const uint8_t *)mask,
+                                       (const uint8_t *)&nic_mask,
+                                       sizeof(struct rte_flow_item_mark),
+                                       error);
+       if (ret < 0)
+               return ret;
+       return 0;
+}
+
 /**
  * Validate META item.
  *
@@ -982,11 +1246,14 @@ flow_dv_validate_item_meta(struct rte_eth_dev *dev __rte_unused,
                           const struct rte_flow_attr *attr,
                           struct rte_flow_error *error)
 {
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_dev_config *config = &priv->config;
        const struct rte_flow_item_meta *spec = item->spec;
        const struct rte_flow_item_meta *mask = item->mask;
-       const struct rte_flow_item_meta nic_mask = {
+       struct rte_flow_item_meta nic_mask = {
                .data = UINT32_MAX
        };
+       enum modify_reg reg;
        int ret;
 
        if (!spec)
@@ -996,22 +1263,90 @@ flow_dv_validate_item_meta(struct rte_eth_dev *dev __rte_unused,
                                          "data cannot be empty");
        if (!spec->data)
                return rte_flow_error_set(error, EINVAL,
-                                         RTE_FLOW_ERROR_TYPE_ITEM_SPEC,
-                                         NULL,
+                                         RTE_FLOW_ERROR_TYPE_ITEM_SPEC, NULL,
                                          "data cannot be zero");
+       if (config->dv_xmeta_en != MLX5_XMETA_MODE_LEGACY) {
+               if (!mlx5_flow_ext_mreg_supported(dev))
+                       return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                         "extended metadata register"
+                                         " isn't supported");
+               reg = flow_dv_get_metadata_reg(dev, attr, error);
+               if (reg < 0)
+                       return reg;
+               if (reg == REG_B)
+                       return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                         "match on reg_b "
+                                         "isn't supported");
+               if (reg != REG_A)
+                       nic_mask.data = priv->sh->dv_meta_mask;
+       }
        if (!mask)
                mask = &rte_flow_item_meta_mask;
        ret = mlx5_flow_item_acceptable(item, (const uint8_t *)mask,
                                        (const uint8_t *)&nic_mask,
                                        sizeof(struct rte_flow_item_meta),
                                        error);
+       return ret;
+}
+
+/**
+ * Validate TAG item.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] item
+ *   Item specification.
+ * @param[in] attr
+ *   Attributes of flow that includes this item.
+ * @param[out] error
+ *   Pointer to error structure.
+ *
+ * @return
+ *   0 on success, a negative errno value otherwise and rte_errno is set.
+ */
+static int
+flow_dv_validate_item_tag(struct rte_eth_dev *dev,
+                         const struct rte_flow_item *item,
+                         const struct rte_flow_attr *attr __rte_unused,
+                         struct rte_flow_error *error)
+{
+       const struct rte_flow_item_tag *spec = item->spec;
+       const struct rte_flow_item_tag *mask = item->mask;
+       const struct rte_flow_item_tag nic_mask = {
+               .data = RTE_BE32(UINT32_MAX),
+               .index = 0xff,
+       };
+       int ret;
+
+       if (!mlx5_flow_ext_mreg_supported(dev))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                         "extensive metadata register"
+                                         " isn't supported");
+       if (!spec)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ITEM_SPEC,
+                                         item->spec,
+                                         "data cannot be empty");
+       if (!mask)
+               mask = &rte_flow_item_tag_mask;
+       ret = mlx5_flow_item_acceptable(item, (const uint8_t *)mask,
+                                       (const uint8_t *)&nic_mask,
+                                       sizeof(struct rte_flow_item_tag),
+                                       error);
        if (ret < 0)
                return ret;
-       if (attr->ingress)
-               return rte_flow_error_set(error, ENOTSUP,
-                                         RTE_FLOW_ERROR_TYPE_ATTR_INGRESS,
-                                         NULL,
-                                         "pattern not supported for ingress");
+       if (mask->index != 0xff)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ITEM_SPEC, NULL,
+                                         "partial mask for tag index"
+                                         " is not supported");
+       ret = mlx5_flow_get_reg_id(dev, MLX5_APP_TAG, spec->index, error);
+       if (ret < 0)
+               return ret;
+       assert(ret != REG_NONE);
        return 0;
 }
 
@@ -1375,6 +1710,256 @@ flow_dv_validate_action_set_vlan_vid(uint64_t item_flags,
        return 0;
 }
 
+/*
+ * Validate the FLAG action.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] action_flags
+ *   Holds the actions detected until now.
+ * @param[in] attr
+ *   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_dv_validate_action_flag(struct rte_eth_dev *dev,
+                            uint64_t action_flags,
+                            const struct rte_flow_attr *attr,
+                            struct rte_flow_error *error)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_dev_config *config = &priv->config;
+       int ret;
+
+       /* Fall back if no extended metadata register support. */
+       if (config->dv_xmeta_en == MLX5_XMETA_MODE_LEGACY)
+               return mlx5_flow_validate_action_flag(action_flags, attr,
+                                                     error);
+       /* Extensive metadata mode requires registers. */
+       if (!mlx5_flow_ext_mreg_supported(dev))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "no metadata registers "
+                                         "to support flag action");
+       if (!(priv->sh->dv_mark_mask & MLX5_FLOW_MARK_DEFAULT))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "extended metadata register"
+                                         " isn't available");
+       ret = mlx5_flow_get_reg_id(dev, MLX5_FLOW_MARK, 0, error);
+       if (ret < 0)
+               return ret;
+       assert(ret > 0);
+       if (action_flags & MLX5_FLOW_ACTION_DROP)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "can't drop and flag in same flow");
+       if (action_flags & MLX5_FLOW_ACTION_MARK)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "can't mark and flag in same flow");
+       if (action_flags & MLX5_FLOW_ACTION_FLAG)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "can't have 2 flag"
+                                         " actions in same flow");
+       return 0;
+}
+
+/**
+ * Validate MARK action.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] action
+ *   Pointer to action.
+ * @param[in] action_flags
+ *   Holds the actions detected until now.
+ * @param[in] attr
+ *   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_dv_validate_action_mark(struct rte_eth_dev *dev,
+                            const struct rte_flow_action *action,
+                            uint64_t action_flags,
+                            const struct rte_flow_attr *attr,
+                            struct rte_flow_error *error)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_dev_config *config = &priv->config;
+       const struct rte_flow_action_mark *mark = action->conf;
+       int ret;
+
+       /* Fall back if no extended metadata register support. */
+       if (config->dv_xmeta_en == MLX5_XMETA_MODE_LEGACY)
+               return mlx5_flow_validate_action_mark(action, action_flags,
+                                                     attr, error);
+       /* Extensive metadata mode requires registers. */
+       if (!mlx5_flow_ext_mreg_supported(dev))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "no metadata registers "
+                                         "to support mark action");
+       if (!priv->sh->dv_mark_mask)
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "extended metadata register"
+                                         " isn't available");
+       ret = mlx5_flow_get_reg_id(dev, MLX5_FLOW_MARK, 0, error);
+       if (ret < 0)
+               return ret;
+       assert(ret > 0);
+       if (!mark)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "configuration cannot be null");
+       if (mark->id >= (MLX5_FLOW_MARK_MAX & priv->sh->dv_mark_mask))
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION_CONF,
+                                         &mark->id,
+                                         "mark id exceeds the limit");
+       if (action_flags & MLX5_FLOW_ACTION_DROP)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "can't drop and mark in same flow");
+       if (action_flags & MLX5_FLOW_ACTION_FLAG)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "can't flag and mark in same flow");
+       if (action_flags & MLX5_FLOW_ACTION_MARK)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, NULL,
+                                         "can't have 2 mark actions in same"
+                                         " flow");
+       return 0;
+}
+
+/**
+ * Validate SET_META action.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] action
+ *   Pointer to the encap action.
+ * @param[in] action_flags
+ *   Holds the actions detected until now.
+ * @param[in] attr
+ *   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_dv_validate_action_set_meta(struct rte_eth_dev *dev,
+                                const struct rte_flow_action *action,
+                                uint64_t action_flags __rte_unused,
+                                const struct rte_flow_attr *attr,
+                                struct rte_flow_error *error)
+{
+       const struct rte_flow_action_set_meta *conf;
+       uint32_t nic_mask = UINT32_MAX;
+       enum modify_reg reg;
+
+       if (!mlx5_flow_ext_mreg_supported(dev))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "extended metadata register"
+                                         " isn't supported");
+       reg = flow_dv_get_metadata_reg(dev, attr, error);
+       if (reg < 0)
+               return reg;
+       if (reg != REG_A && reg != REG_B) {
+               struct mlx5_priv *priv = dev->data->dev_private;
+
+               nic_mask = priv->sh->dv_meta_mask;
+       }
+       if (!(action->conf))
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "configuration cannot be null");
+       conf = (const struct rte_flow_action_set_meta *)action->conf;
+       if (!conf->mask)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "zero mask doesn't have any effect");
+       if (conf->mask & ~nic_mask)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "meta data must be within reg C0");
+       if (!(conf->data & conf->mask))
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "zero value has no effect");
+       return 0;
+}
+
+/**
+ * Validate SET_TAG action.
+ *
+ * @param[in] dev
+ *   Pointer to the rte_eth_dev structure.
+ * @param[in] action
+ *   Pointer to the encap action.
+ * @param[in] action_flags
+ *   Holds the actions detected until now.
+ * @param[in] attr
+ *   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_dv_validate_action_set_tag(struct rte_eth_dev *dev,
+                               const struct rte_flow_action *action,
+                               uint64_t action_flags,
+                               const struct rte_flow_attr *attr,
+                               struct rte_flow_error *error)
+{
+       const struct rte_flow_action_set_tag *conf;
+       const uint64_t terminal_action_flags =
+               MLX5_FLOW_ACTION_DROP | MLX5_FLOW_ACTION_QUEUE |
+               MLX5_FLOW_ACTION_RSS;
+       int ret;
+
+       if (!mlx5_flow_ext_mreg_supported(dev))
+               return rte_flow_error_set(error, ENOTSUP,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "extensive metadata register"
+                                         " isn't supported");
+       if (!(action->conf))
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "configuration cannot be null");
+       conf = (const struct rte_flow_action_set_tag *)action->conf;
+       if (!conf->mask)
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "zero mask doesn't have any effect");
+       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))
+               return rte_flow_error_set(error, EINVAL,
+                                         RTE_FLOW_ERROR_TYPE_ACTION, action,
+                                         "set_tag has no effect"
+                                         " with terminal actions");
+       return 0;
+}
+
 /**
  * Validate count action.
  *
@@ -3587,6 +4172,8 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct rte_flow_attr *attr,
                        .dst_port = RTE_BE16(UINT16_MAX),
                }
        };
+       struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_dev_config *dev_conf = &priv->config;
 
        if (items == NULL)
                return -1;
@@ -3769,6 +4356,14 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct rte_flow_attr *attr,
                                return ret;
                        last_item = MLX5_FLOW_LAYER_MPLS;
                        break;
+
+               case RTE_FLOW_ITEM_TYPE_MARK:
+                       ret = flow_dv_validate_item_mark(dev, items, attr,
+                                                        error);
+                       if (ret < 0)
+                               return ret;
+                       last_item = MLX5_FLOW_ITEM_MARK;
+                       break;
                case RTE_FLOW_ITEM_TYPE_META:
                        ret = flow_dv_validate_item_meta(dev, items, attr,
                                                         error);
@@ -3792,6 +4387,13 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct rte_flow_attr *attr,
                                return ret;
                        last_item = MLX5_FLOW_LAYER_ICMP6;
                        break;
+               case RTE_FLOW_ITEM_TYPE_TAG:
+                       ret = flow_dv_validate_item_tag(dev, items,
+                                                       attr, error);
+                       if (ret < 0)
+                               return ret;
+                       last_item = MLX5_FLOW_ITEM_TAG;
+                       break;
                case MLX5_RTE_FLOW_ITEM_TYPE_TAG:
                case MLX5_RTE_FLOW_ITEM_TYPE_TX_QUEUE:
                        break;
@@ -3823,21 +4425,61 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct rte_flow_attr *attr,
                        ++actions_n;
                        break;
                case RTE_FLOW_ACTION_TYPE_FLAG:
-                       ret = mlx5_flow_validate_action_flag(action_flags,
-                                                            attr, error);
+                       ret = flow_dv_validate_action_flag(dev, action_flags,
+                                                          attr, error);
                        if (ret < 0)
                                return ret;
-                       action_flags |= MLX5_FLOW_ACTION_FLAG;
-                       ++actions_n;
+                       if (dev_conf->dv_xmeta_en != MLX5_XMETA_MODE_LEGACY) {
+                               /* Count all modify-header actions as one. */
+                               if (!(action_flags &
+                                     MLX5_FLOW_MODIFY_HDR_ACTIONS))
+                                       ++actions_n;
+                               action_flags |= MLX5_FLOW_ACTION_FLAG |
+                                               MLX5_FLOW_ACTION_MARK_EXT;
+                       } else {
+                               action_flags |= MLX5_FLOW_ACTION_FLAG;
+                               ++actions_n;
+                       }
                        break;
                case RTE_FLOW_ACTION_TYPE_MARK:
-                       ret = mlx5_flow_validate_action_mark(actions,
-                                                            action_flags,
-                                                            attr, error);
+                       ret = flow_dv_validate_action_mark(dev, actions,
+                                                          action_flags,
+                                                          attr, error);
                        if (ret < 0)
                                return ret;
-                       action_flags |= MLX5_FLOW_ACTION_MARK;
-                       ++actions_n;
+                       if (dev_conf->dv_xmeta_en != MLX5_XMETA_MODE_LEGACY) {
+                               /* Count all modify-header actions as one. */
+                               if (!(action_flags &
+                                     MLX5_FLOW_MODIFY_HDR_ACTIONS))
+                                       ++actions_n;
+                               action_flags |= MLX5_FLOW_ACTION_MARK |
+                                               MLX5_FLOW_ACTION_MARK_EXT;
+                       } else {
+                               action_flags |= MLX5_FLOW_ACTION_MARK;
+                               ++actions_n;
+                       }
+                       break;
+               case RTE_FLOW_ACTION_TYPE_SET_META:
+                       ret = flow_dv_validate_action_set_meta(dev, actions,
+                                                              action_flags,
+                                                              attr, error);
+                       if (ret < 0)
+                               return ret;
+                       /* Count all modify-header actions as one action. */
+                       if (!(action_flags & MLX5_FLOW_MODIFY_HDR_ACTIONS))
+                               ++actions_n;
+                       action_flags |= MLX5_FLOW_ACTION_SET_META;
+                       break;
+               case RTE_FLOW_ACTION_TYPE_SET_TAG:
+                       ret = flow_dv_validate_action_set_tag(dev, actions,
+                                                             action_flags,
+                                                             attr, error);
+                       if (ret < 0)
+                               return ret;
+                       /* Count all modify-header actions as one action. */
+                       if (!(action_flags & MLX5_FLOW_MODIFY_HDR_ACTIONS))
+                               ++actions_n;
+                       action_flags |= MLX5_FLOW_ACTION_SET_TAG;
                        break;
                case RTE_FLOW_ACTION_TYPE_DROP:
                        ret = mlx5_flow_validate_action_drop(action_flags,
@@ -4097,12 +4739,14 @@ flow_dv_validate(struct rte_eth_dev *dev, const struct rte_flow_attr *attr,
                                          " actions in the same rule");
        /* Eswitch has few restrictions on using items and actions */
        if (attr->transfer) {
-               if (action_flags & MLX5_FLOW_ACTION_FLAG)
+               if (!mlx5_flow_ext_mreg_supported(dev) &&
+                   action_flags & MLX5_FLOW_ACTION_FLAG)
                        return rte_flow_error_set(error, ENOTSUP,
                                                  RTE_FLOW_ERROR_TYPE_ACTION,
                                                  NULL,
                                                  "unsupported action FLAG");
-               if (action_flags & MLX5_FLOW_ACTION_MARK)
+               if (!mlx5_flow_ext_mreg_supported(dev) &&
+                   action_flags & MLX5_FLOW_ACTION_MARK)
                        return rte_flow_error_set(error, ENOTSUP,
                                                  RTE_FLOW_ERROR_TYPE_ACTION,
                                                  NULL,
@@ -5064,20 +5708,62 @@ flow_dv_match_meta_reg(void *matcher, void *key,
        }
 }
 
+/**
+ * Add MARK item to matcher
+ *
+ * @param[in] dev
+ *   The device to configure through.
+ * @param[in, out] matcher
+ *   Flow matcher.
+ * @param[in, out] key
+ *   Flow matcher value.
+ * @param[in] item
+ *   Flow pattern to translate.
+ */
+static void
+flow_dv_translate_item_mark(struct rte_eth_dev *dev,
+                           void *matcher, void *key,
+                           const struct rte_flow_item *item)
+{
+       struct mlx5_priv *priv = dev->data->dev_private;
+       const struct rte_flow_item_mark *mark;
+       uint32_t value;
+       uint32_t mask;
+
+       mark = item->mask ? (const void *)item->mask :
+                           &rte_flow_item_mark_mask;
+       mask = mark->id & priv->sh->dv_mark_mask;
+       mark = (const void *)item->spec;
+       assert(mark);
+       value = mark->id & priv->sh->dv_mark_mask & mask;
+       if (mask) {
+               enum modify_reg reg;
+
+               /* Get the metadata register index for the mark. */
+               reg = mlx5_flow_get_reg_id(dev, MLX5_FLOW_MARK, 0, NULL);
+               assert(reg > 0);
+               flow_dv_match_meta_reg(matcher, key, reg, value, mask);
+       }
+}
+
 /**
  * Add META item to matcher
  *
+ * @param[in] dev
+ *   The devich to configure through.
  * @param[in, out] matcher
  *   Flow matcher.
  * @param[in, out] key
  *   Flow matcher value.
+ * @param[in] attr
+ *   Attributes of flow that includes this item.
  * @param[in] item
  *   Flow pattern to translate.
- * @param[in] inner
- *   Item is inner pattern.
  */
 static void
-flow_dv_translate_item_meta(void *matcher, void *key,
+flow_dv_translate_item_meta(struct rte_eth_dev *dev,
+                           void *matcher, void *key,
+                           const struct rte_flow_attr *attr,
                            const struct rte_flow_item *item)
 {
        const struct rte_flow_item_meta *meta_m;
@@ -5087,10 +5773,34 @@ flow_dv_translate_item_meta(void *matcher, void *key,
        if (!meta_m)
                meta_m = &rte_flow_item_meta_mask;
        meta_v = (const void *)item->spec;
-       if (meta_v)
-               flow_dv_match_meta_reg(matcher, key, REG_A,
-                                      rte_cpu_to_be_32(meta_v->data),
-                                      rte_cpu_to_be_32(meta_m->data));
+       if (meta_v) {
+               enum modify_reg reg;
+               uint32_t value = meta_v->data;
+               uint32_t mask = meta_m->data;
+
+               reg = flow_dv_get_metadata_reg(dev, attr, NULL);
+               if (reg < 0)
+                       return;
+               /*
+                * In datapath code there is no endianness
+                * coversions for perfromance reasons, all
+                * pattern conversions are done in rte_flow.
+                */
+               value = rte_cpu_to_be_32(value);
+               mask = rte_cpu_to_be_32(mask);
+               if (reg == REG_C_0) {
+                       struct mlx5_priv *priv = dev->data->dev_private;
+                       uint32_t msk_c0 = priv->sh->dv_regc0_mask;
+                       uint32_t shl_c0 = rte_bsf32(msk_c0);
+
+                       msk_c0 = rte_cpu_to_be_32(msk_c0);
+                       value <<= shl_c0;
+                       mask <<= shl_c0;
+                       assert(msk_c0);
+                       assert(!(~msk_c0 & mask));
+               }
+               flow_dv_match_meta_reg(matcher, key, reg, value, mask);
+       }
 }
 
 /**
@@ -5126,8 +5836,38 @@ flow_dv_translate_mlx5_item_tag(void *matcher, void *key,
 {
        const struct mlx5_rte_flow_item_tag *tag_v = item->spec;
        const struct mlx5_rte_flow_item_tag *tag_m = item->mask;
-       enum modify_reg reg = tag_v->id;
 
+       assert(tag_v);
+       flow_dv_match_meta_reg(matcher, key, tag_v->id, tag_v->data,
+                              tag_m ? tag_m->data : UINT32_MAX);
+}
+
+/**
+ * Add TAG item to matcher
+ *
+ * @param[in] dev
+ *   The devich to configure through.
+ * @param[in, out] matcher
+ *   Flow matcher.
+ * @param[in, out] key
+ *   Flow matcher value.
+ * @param[in] item
+ *   Flow pattern to translate.
+ */
+static void
+flow_dv_translate_item_tag(struct rte_eth_dev *dev,
+                          void *matcher, void *key,
+                          const struct rte_flow_item *item)
+{
+       const struct rte_flow_item_tag *tag_v = item->spec;
+       const struct rte_flow_item_tag *tag_m = item->mask;
+       enum modify_reg reg;
+
+       assert(tag_v);
+       tag_m = tag_m ? tag_m : &rte_flow_item_tag_mask;
+       /* Get the metadata register index for the tag. */
+       reg = mlx5_flow_get_reg_id(dev, MLX5_APP_TAG, tag_v->index, NULL);
+       assert(reg > 0);
        flow_dv_match_meta_reg(matcher, key, reg, tag_v->data, tag_m->data);
 }
 
@@ -5715,6 +6455,7 @@ __flow_dv_translate(struct rte_eth_dev *dev,
                    struct rte_flow_error *error)
 {
        struct mlx5_priv *priv = dev->data->dev_private;
+       struct mlx5_dev_config *dev_conf = &priv->config;
        struct rte_flow *flow = dev_flow->flow;
        uint64_t item_flags = 0;
        uint64_t last_item = 0;
@@ -5749,7 +6490,7 @@ __flow_dv_translate(struct rte_eth_dev *dev,
        if (attr->transfer)
                mhdr_res.ft_type = MLX5DV_FLOW_TABLE_TYPE_FDB;
        if (priority == MLX5_FLOW_PRIO_RSVD)
-               priority = priv->config.flow_prio - 1;
+               priority = dev_conf->flow_prio - 1;
        for (; !actions_end ; actions++) {
                const struct rte_flow_action_queue *queue;
                const struct rte_flow_action_rss *rss;
@@ -5780,6 +6521,19 @@ __flow_dv_translate(struct rte_eth_dev *dev,
                        action_flags |= MLX5_FLOW_ACTION_PORT_ID;
                        break;
                case RTE_FLOW_ACTION_TYPE_FLAG:
+                       action_flags |= MLX5_FLOW_ACTION_FLAG;
+                       if (dev_conf->dv_xmeta_en != MLX5_XMETA_MODE_LEGACY) {
+                               struct rte_flow_action_mark mark = {
+                                       .id = MLX5_FLOW_MARK_DEFAULT,
+                               };
+
+                               if (flow_dv_convert_action_mark(dev, &mark,
+                                                               &mhdr_res,
+                                                               error))
+                                       return -rte_errno;
+                               action_flags |= MLX5_FLOW_ACTION_MARK_EXT;
+                               break;
+                       }
                        tag_resource.tag =
                                mlx5_flow_mark_set(MLX5_FLOW_MARK_DEFAULT);
                        if (!dev_flow->dv.tag_resource)
@@ -5788,9 +6542,22 @@ __flow_dv_translate(struct rte_eth_dev *dev,
                                        return errno;
                        dev_flow->dv.actions[actions_n++] =
                                dev_flow->dv.tag_resource->action;
-                       action_flags |= MLX5_FLOW_ACTION_FLAG;
                        break;
                case RTE_FLOW_ACTION_TYPE_MARK:
+                       action_flags |= MLX5_FLOW_ACTION_MARK;
+                       if (dev_conf->dv_xmeta_en != MLX5_XMETA_MODE_LEGACY) {
+                               const struct rte_flow_action_mark *mark =
+                                       (const struct rte_flow_action_mark *)
+                                               actions->conf;
+
+                               if (flow_dv_convert_action_mark(dev, mark,
+                                                               &mhdr_res,
+                                                               error))
+                                       return -rte_errno;
+                               action_flags |= MLX5_FLOW_ACTION_MARK_EXT;
+                               break;
+                       }
+                       /* Legacy (non-extensive) MARK action. */
                        tag_resource.tag = mlx5_flow_mark_set
                              (((const struct rte_flow_action_mark *)
                               (actions->conf))->id);
@@ -5800,7 +6567,22 @@ __flow_dv_translate(struct rte_eth_dev *dev,
                                        return errno;
                        dev_flow->dv.actions[actions_n++] =
                                dev_flow->dv.tag_resource->action;
-                       action_flags |= MLX5_FLOW_ACTION_MARK;
+                       break;
+               case RTE_FLOW_ACTION_TYPE_SET_META:
+                       if (flow_dv_convert_action_set_meta
+                               (dev, &mhdr_res, attr,
+                                (const struct rte_flow_action_set_meta *)
+                                 actions->conf, error))
+                               return -rte_errno;
+                       action_flags |= MLX5_FLOW_ACTION_SET_META;
+                       break;
+               case RTE_FLOW_ACTION_TYPE_SET_TAG:
+                       if (flow_dv_convert_action_set_tag
+                               (dev, &mhdr_res,
+                                (const struct rte_flow_action_set_tag *)
+                                 actions->conf, error))
+                               return -rte_errno;
+                       action_flags |= MLX5_FLOW_ACTION_SET_TAG;
                        break;
                case RTE_FLOW_ACTION_TYPE_DROP:
                        action_flags |= MLX5_FLOW_ACTION_DROP;
@@ -5829,7 +6611,7 @@ __flow_dv_translate(struct rte_eth_dev *dev,
                        action_flags |= MLX5_FLOW_ACTION_RSS;
                        break;
                case RTE_FLOW_ACTION_TYPE_COUNT:
-                       if (!priv->config.devx) {
+                       if (!dev_conf->devx) {
                                rte_errno = ENOTSUP;
                                goto cnt_err;
                        }
@@ -6082,7 +6864,7 @@ cnt_err:
                        break;
                case RTE_FLOW_ACTION_TYPE_END:
                        actions_end = true;
-                       if (action_flags & MLX5_FLOW_MODIFY_HDR_ACTIONS) {
+                       if (mhdr_res.actions_num) {
                                /* create modify action if needed. */
                                if (flow_dv_modify_hdr_resource_register
                                        (dev, &mhdr_res, dev_flow, error))
@@ -6094,7 +6876,7 @@ cnt_err:
                default:
                        break;
                }
-               if ((action_flags & MLX5_FLOW_MODIFY_HDR_ACTIONS) &&
+               if (mhdr_res.actions_num &&
                    modify_action_position == UINT32_MAX)
                        modify_action_position = actions_n++;
        }
@@ -6242,9 +7024,14 @@ cnt_err:
                                                    items, last_item, tunnel);
                        last_item = MLX5_FLOW_LAYER_MPLS;
                        break;
+               case RTE_FLOW_ITEM_TYPE_MARK:
+                       flow_dv_translate_item_mark(dev, match_mask,
+                                                   match_value, items);
+                       last_item = MLX5_FLOW_ITEM_MARK;
+                       break;
                case RTE_FLOW_ITEM_TYPE_META:
-                       flow_dv_translate_item_meta(match_mask, match_value,
-                                                   items);
+                       flow_dv_translate_item_meta(dev, match_mask,
+                                                   match_value, attr, items);
                        last_item = MLX5_FLOW_ITEM_METADATA;
                        break;
                case RTE_FLOW_ITEM_TYPE_ICMP:
@@ -6257,6 +7044,11 @@ cnt_err:
                                                      items, tunnel);
                        last_item = MLX5_FLOW_LAYER_ICMP6;
                        break;
+               case RTE_FLOW_ITEM_TYPE_TAG:
+                       flow_dv_translate_item_tag(dev, match_mask,
+                                                  match_value, items);
+                       last_item = MLX5_FLOW_ITEM_TAG;
+                       break;
                case MLX5_RTE_FLOW_ITEM_TYPE_TAG:
                        flow_dv_translate_mlx5_item_tag(match_mask,
                                                        match_value, items);