On Wed, Apr 04, 2018 at 05:56:49PM +0200, Adrien Mazarguil wrote:
> TPID handling in rte_flow VLAN and E_TAG pattern item definitions is not
> consistent with the normal stacking order of pattern items, which is
> confusing to applications.
> 
> Problem is that when followed by one of these layers, the EtherType field
> of the preceding layer keeps its "inner" definition, and the "outer" TPID
> is provided by the subsequent layer, the reverse of how a packet looks like
> on the wire:
> 
>  Wire:     [ ETH TPID = A | VLAN EtherType = B | B DATA ]
>  rte_flow: [ ETH EtherType = B | VLAN TPID = A | B DATA ]
> 
> Worse, when QinQ is involved, the stacking order of VLAN layers is
> unspecified. It is unclear whether it should be reversed (innermost to
> outermost) as well given TPID applies to the previous layer:
> 
>  Wire:       [ ETH TPID = A | VLAN TPID = B | VLAN EtherType = C | C DATA ]
>  rte_flow 1: [ ETH EtherType = C | VLAN TPID = B | VLAN TPID = A | C DATA ]
>  rte_flow 2: [ ETH EtherType = C | VLAN TPID = A | VLAN TPID = B | C DATA ]
> 
> While specifying EtherType/TPID is hopefully rarely necessary, the stacking
> order in case of QinQ and the lack of documentation remain an issue.
> 
> This patch replaces TPID in the VLAN pattern item with an inner
> EtherType/TPID as is usually done everywhere else (e.g. struct vlan_hdr),
> clarifies documentation and updates all relevant code.
> 
> Summary of changes for PMDs that implement ETH, VLAN or E_TAG pattern
> items:
> 
> - bnxt: EtherType matching is supported, and vlan->inner_type overrides
>   eth->type if the latter has standard TPID value 0x8100, otherwise an
>   error is triggered.
> 
> - e1000: EtherType matching is only supported with the ETHERTYPE filter,
>   which does not support VLAN matching, therefore no impact.
> 
> - enic: same as bnxt.
> 
> - i40e: same as bnxt with a configurable TPID value for the FDIR filter,
>   with existing limitations on allowed EtherType values. The remaining
>   filter types (VXLAN, NVGRE, QINQ) do not support EtherType matching.
> 
> - ixgbe: same as e1000.
> 
> - mlx4: EtherType/TPID matching is not supported, no impact.
> 
> - mlx5: same as bnxt.
> 
> - mrvl: EtherType matching is supported but eth->type cannot be specified
>   when a VLAN item is present. However vlan->inner_type is used if
>   specified.
> 
> - sfc: same as bnxt with QinQ TPID value 0x88a8 additionally supported.
> 
> - tap: same as bnxt.
> 
> Signed-off-by: Adrien Mazarguil <adrien.mazarg...@6wind.com>
> Cc: Ferruh Yigit <ferruh.yi...@intel.com>
> Cc: Thomas Monjalon <tho...@monjalon.net>
> Cc: Wenzhuo Lu <wenzhuo...@intel.com>
> Cc: Jingjing Wu <jingjing...@intel.com>
> Cc: Ajit Khaparde <ajit.khapa...@broadcom.com>
> Cc: Somnath Kotur <somnath.ko...@broadcom.com>
> Cc: John Daley <johnd...@cisco.com>
> Cc: Hyong Youb Kim <hyon...@cisco.com>
> Cc: Beilei Xing <beilei.x...@intel.com>
> Cc: Qi Zhang <qi.z.zh...@intel.com>
> Cc: Konstantin Ananyev <konstantin.anan...@intel.com>
> Cc: Nelio Laranjeiro <nelio.laranje...@6wind.com>
> Cc: Yongseok Koh <ys...@mellanox.com>
> Cc: Jacek Siuda <j...@semihalf.com>
> Cc: Tomasz Duszynski <t...@semihalf.com>
> Cc: Dmitri Epshtein <d...@marvell.com>
> Cc: Natalie Samsonov <nsams...@marvell.com>
> Cc: Jianbo Liu <jianbo....@arm.com>
> Cc: Andrew Rybchenko <arybche...@solarflare.com>
> Cc: Pascal Mazon <pascal.ma...@6wind.com>
> 
> ---
> 
> Hi PMD maintainers, while I'm pretty confident in these changes, I could
> not validate them with all devices.
> 
> It would be great if you could apply this patch, run testpmd, create VLAN
> flow rules with/without inner EtherType as described and send matching
> traffic while making sure nothing was broken in the process.
> 
> Thanks!
> ---
>  app/test-pmd/cmdline_flow.c                 | 17 +++---
>  doc/guides/nics/tap.rst                     |  2 +-
>  doc/guides/prog_guide/rte_flow.rst          | 21 +++++--
>  doc/guides/testpmd_app_ug/testpmd_funcs.rst |  4 +-
>  drivers/net/bnxt/bnxt_filter.c              | 38 ++++++++++--
>  drivers/net/enic/enic_flow.c                | 21 ++++---
>  drivers/net/i40e/i40e_flow.c                | 74 ++++++++++++++++++++----
>  drivers/net/mlx5/mlx5_flow.c                | 14 ++++-
>  drivers/net/mvpp2/mrvl_flow.c               | 27 +++++++--
>  drivers/net/sfc/sfc_flow.c                  | 27 +++++++++
>  drivers/net/tap/tap_flow.c                  | 16 +++--
>  lib/librte_ether/rte_flow.h                 | 24 +++++---
>  12 files changed, 227 insertions(+), 58 deletions(-)
> 
> diff --git a/app/test-pmd/cmdline_flow.c b/app/test-pmd/cmdline_flow.c
> index 2fbd3d8ef..3a486032d 100644
> --- a/app/test-pmd/cmdline_flow.c
> +++ b/app/test-pmd/cmdline_flow.c
> @@ -99,11 +99,11 @@ enum index {
>       ITEM_ETH_SRC,
>       ITEM_ETH_TYPE,
>       ITEM_VLAN,
> -     ITEM_VLAN_TPID,
>       ITEM_VLAN_TCI,
>       ITEM_VLAN_PCP,
>       ITEM_VLAN_DEI,
>       ITEM_VLAN_VID,
> +     ITEM_VLAN_INNER_TYPE,
>       ITEM_IPV4,
>       ITEM_IPV4_TOS,
>       ITEM_IPV4_TTL,
> @@ -505,11 +505,11 @@ static const enum index item_eth[] = {
>  };
>  
>  static const enum index item_vlan[] = {
> -     ITEM_VLAN_TPID,
>       ITEM_VLAN_TCI,
>       ITEM_VLAN_PCP,
>       ITEM_VLAN_DEI,
>       ITEM_VLAN_VID,
> +     ITEM_VLAN_INNER_TYPE,
>       ITEM_NEXT,
>       ZERO,
>  };
> @@ -1142,12 +1142,6 @@ static const struct token token_list[] = {
>               .next = NEXT(item_vlan),
>               .call = parse_vc,
>       },
> -     [ITEM_VLAN_TPID] = {
> -             .name = "tpid",
> -             .help = "tag protocol identifier",
> -             .next = NEXT(item_vlan, NEXT_ENTRY(UNSIGNED), item_param),
> -             .args = ARGS(ARGS_ENTRY_HTON(struct rte_flow_item_vlan, tpid)),
> -     },
>       [ITEM_VLAN_TCI] = {
>               .name = "tci",
>               .help = "tag control information",
> @@ -1175,6 +1169,13 @@ static const struct token token_list[] = {
>               .args = ARGS(ARGS_ENTRY_MASK_HTON(struct rte_flow_item_vlan,
>                                                 tci, "\x0f\xff")),
>       },
> +     [ITEM_VLAN_INNER_TYPE] = {
> +             .name = "inner_type",
> +             .help = "inner EtherType",
> +             .next = NEXT(item_vlan, NEXT_ENTRY(UNSIGNED), item_param),
> +             .args = ARGS(ARGS_ENTRY_HTON(struct rte_flow_item_vlan,
> +                                          inner_type)),
> +     },
>       [ITEM_IPV4] = {
>               .name = "ipv4",
>               .help = "match IPv4 header",
> diff --git a/doc/guides/nics/tap.rst b/doc/guides/nics/tap.rst
> index 76eb0bde4..bcf3efe9e 100644
> --- a/doc/guides/nics/tap.rst
> +++ b/doc/guides/nics/tap.rst
> @@ -97,7 +97,7 @@ The kernel support can be checked with this command::
>  Supported items:
>  
>  - eth: src and dst (with variable masks), and eth_type (0xffff mask).
> -- vlan: vid, pcp, tpid, but not eid. (requires kernel 4.9)
> +- vlan: vid, pcp, but not eid. (requires kernel 4.9)
>  - ipv4/6: src and dst (with variable masks), and ip_proto (0xffff mask).
>  - udp/tcp: src and dst port (0xffff) mask.
>  
> diff --git a/doc/guides/prog_guide/rte_flow.rst 
> b/doc/guides/prog_guide/rte_flow.rst
> index c893d737a..f1b9d8d76 100644
> --- a/doc/guides/prog_guide/rte_flow.rst
> +++ b/doc/guides/prog_guide/rte_flow.rst
> @@ -784,9 +784,15 @@ Item: ``ETH``
>  
>  Matches an Ethernet header.
>  
> +The ``type`` field either stands for "EtherType" or "TPID" when followed by
> +so-called layer 2.5 pattern items such as ``RTE_FLOW_ITEM_TYPE_VLAN``. In
> +the latter case, ``type`` refers to that of the outer header, with the inner
> +EtherType/TPID provided by the subsequent pattern item. This is the same
> +order as on the wire.
> +
>  - ``dst``: destination MAC.
>  - ``src``: source MAC.
> -- ``type``: EtherType.
> +- ``type``: EtherType or TPID.
>  - Default ``mask`` matches destination and source addresses only.
>  
>  Item: ``VLAN``
> @@ -794,9 +800,13 @@ Item: ``VLAN``
>  
>  Matches an 802.1Q/ad VLAN tag.
>  
> -- ``tpid``: tag protocol identifier.
> +The corresponding standard outer EtherType (TPID) values are ``0x8100`` or
> +``0x88a8`` (in case of outer QinQ). It can be overridden by the preceding
> +pattern item.
> +
>  - ``tci``: tag control information.
> -- Default ``mask`` matches TCI only.
> +- ``inner_type``: inner EtherType or TPID.
> +- Default ``mask`` matches the VID part of TCI only (lower 12 bits).
>  
>  Item: ``IPV4``
>  ^^^^^^^^^^^^^^
> @@ -866,12 +876,15 @@ Item: ``E_TAG``
>  
>  Matches an IEEE 802.1BR E-Tag header.
>  
> -- ``tpid``: tag protocol identifier (0x893F)
> +The corresponding standard outer EtherType (TPID) value is 0x893f.
> +It can be overridden by the preceding pattern item.
> +
>  - ``epcp_edei_in_ecid_b``: E-Tag control information (E-TCI), E-PCP (3b),
>    E-DEI (1b), ingress E-CID base (12b).
>  - ``rsvd_grp_ecid_b``: reserved (2b), GRP (2b), E-CID base (12b).
>  - ``in_ecid_e``: ingress E-CID ext.
>  - ``ecid_e``: E-CID ext.
> +- ``inner_type``: inner EtherType or TPID.
>  - Default ``mask`` simultaneously matches GRP and E-CID base.
>  
>  Item: ``NVGRE``
> diff --git a/doc/guides/testpmd_app_ug/testpmd_funcs.rst 
> b/doc/guides/testpmd_app_ug/testpmd_funcs.rst
> index 738461f44..25fac8430 100644
> --- a/doc/guides/testpmd_app_ug/testpmd_funcs.rst
> +++ b/doc/guides/testpmd_app_ug/testpmd_funcs.rst
> @@ -3223,15 +3223,15 @@ This section lists supported pattern items and their 
> attributes, if any.
>  
>    - ``dst {MAC-48}``: destination MAC.
>    - ``src {MAC-48}``: source MAC.
> -  - ``type {unsigned}``: EtherType.
> +  - ``type {unsigned}``: EtherType or TPID.
>  
>  - ``vlan``: match 802.1Q/ad VLAN tag.
>  
> -  - ``tpid {unsigned}``: tag protocol identifier.
>    - ``tci {unsigned}``: tag control information.
>    - ``pcp {unsigned}``: priority code point.
>    - ``dei {unsigned}``: drop eligible indicator.
>    - ``vid {unsigned}``: VLAN identifier.
> +  - ``inner_type {unsigned}``: inner EtherType or TPID.
>  
>  - ``ipv4``: match IPv4 header.
>  
> diff --git a/drivers/net/bnxt/bnxt_filter.c b/drivers/net/bnxt/bnxt_filter.c
> index 0d735edf6..a97a8dd46 100644
> --- a/drivers/net/bnxt/bnxt_filter.c
> +++ b/drivers/net/bnxt/bnxt_filter.c
> @@ -327,6 +327,7 @@ bnxt_validate_and_parse_flow_type(struct bnxt *bp,
>       uint32_t vf = 0;
>       int use_ntuple;
>       uint32_t en = 0;
> +     uint32_t en_ethertype;
>       int dflt_vnic;
>  
>       use_ntuple = bnxt_filter_type_check(pattern, error);
> @@ -336,6 +337,9 @@ bnxt_validate_and_parse_flow_type(struct bnxt *bp,
>  
>       filter->filter_type = use_ntuple ?
>               HWRM_CFA_NTUPLE_FILTER : HWRM_CFA_EM_FILTER;
> +     en_ethertype = use_ntuple ?
> +             NTUPLE_FLTR_ALLOC_INPUT_EN_ETHERTYPE :
> +             EM_FLOW_ALLOC_INPUT_EN_ETHERTYPE;
>  
>       while (item->type != RTE_FLOW_ITEM_TYPE_END) {
>               if (item->last) {
> @@ -405,30 +409,52 @@ bnxt_validate_and_parse_flow_type(struct bnxt *bp,
>                       if (eth_mask->type) {
>                               filter->ethertype =
>                                       rte_be_to_cpu_16(eth_spec->type);
> -                             en |= use_ntuple ?
> -                                     NTUPLE_FLTR_ALLOC_INPUT_EN_ETHERTYPE :
> -                                     EM_FLOW_ALLOC_INPUT_EN_ETHERTYPE;
> +                             en |= en_ethertype;
>                       }
>  
>                       break;
>               case RTE_FLOW_ITEM_TYPE_VLAN:
>                       vlan_spec = item->spec;
>                       vlan_mask = item->mask;
> +                     if (en & en_ethertype &&
> +                         filter->ethertype != RTE_BE16(0x8100)) {
> +                             rte_flow_error_set(error, EINVAL,
> +                                                RTE_FLOW_ERROR_TYPE_ITEM,
> +                                                item,
> +                                                "unsupported outer TPID");
> +                             return -rte_errno;
> +                     }
>                       if (vlan_mask->tci &&
> -                         vlan_mask->tci == RTE_BE16(0x0fff) &&
> -                         !vlan_mask->tpid) {
> +                         vlan_mask->tci == RTE_BE16(0x0fff)) {
>                               /* Only the VLAN ID can be matched. */
>                               filter->l2_ovlan =
>                                       rte_be_to_cpu_16(vlan_spec->tci &
>                                                        RTE_BE16(0x0fff));
>                               en |= EM_FLOW_ALLOC_INPUT_EN_OVLAN_VID;
> -                     } else if (vlan_mask->tci || vlan_mask->tpid) {
> +                     } else if (vlan_mask->tci) {
>                               rte_flow_error_set(error, EINVAL,
>                                                  RTE_FLOW_ERROR_TYPE_ITEM,
>                                                  item,
>                                                  "VLAN mask is invalid");
>                               return -rte_errno;
>                       }
> +                     if (vlan_mask->inner_type &&
> +                         vlan_mask->inner_type != RTE_BE16(0xffff)) {
> +                             rte_flow_error_set(error, EINVAL,
> +                                                RTE_FLOW_ERROR_TYPE_ITEM,
> +                                                item,
> +                                                "inner ethertype mask not"
> +                                                " valid");
> +                             return -rte_errno;
> +                     }
> +                     if (vlan_mask->inner_type) {
> +                             filter->ethertype =
> +                                     rte_be_to_cpu_16(vlan_spec->inner_type);
> +                             en |= en_ethertype;
> +                     } else {
> +                             filter->ethertype = RTE_BE16(0x0000);
> +                             en &= ~en_ethertype;
> +                     }
>  
>                       break;
>               case RTE_FLOW_ITEM_TYPE_IPV4:
> diff --git a/drivers/net/enic/enic_flow.c b/drivers/net/enic/enic_flow.c
> index c5c98b870..a413740ab 100644
> --- a/drivers/net/enic/enic_flow.c
> +++ b/drivers/net/enic/enic_flow.c
> @@ -4,6 +4,7 @@
>  
>  #include <errno.h>
>  #include <stdint.h>
> +#include <rte_byteorder.h>
>  #include <rte_log.h>
>  #include <rte_ethdev_driver.h>
>  #include <rte_flow_driver.h>
> @@ -545,16 +546,22 @@ enic_copy_item_vlan_v2(const struct rte_flow_item *item,
>       if (!spec)
>               return 0;
>  
> -     /* Don't support filtering in tpid */
> -     if (mask) {
> -             if (mask->tpid != 0)
> -                     return ENOTSUP;
> -     } else {
> +     if (!mask)
>               mask = &rte_flow_item_vlan_mask;
> -             RTE_ASSERT(mask->tpid == 0);
> -     }
>  
>       if (*inner_ofst == 0) {
> +             struct ether_hdr *eth_mask =
> +                     (void *)gp->layer[FILTER_GENERIC_1_L2].mask;
> +             struct ether_hdr *eth_val =
> +                     (void *)gp->layer[FILTER_GENERIC_1_L2].val;
> +
> +             /* Exactly one TPID value is allowed if specified */
> +             if ((eth_val->ether_type & eth_mask->ether_type) !=
> +                 (RTE_BE16(0x8100) & eth_mask->ether_type))
> +                     return ENOTSUP;
> +             eth_mask->ether_type = mask->inner_type;
> +             eth_val->ether_type = spec->inner_type;
> +
>               /* Outer header. Use the vlan mask/val fields */
>               gp->mask_vlan = mask->tci;
>               gp->val_vlan = spec->tci;
> diff --git a/drivers/net/i40e/i40e_flow.c b/drivers/net/i40e/i40e_flow.c
> index 1b336df74..c6dd889ad 100644
> --- a/drivers/net/i40e/i40e_flow.c
> +++ b/drivers/net/i40e/i40e_flow.c
> @@ -2490,24 +2490,36 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
>                                                     "Invalid MAC_addr mask.");
>                                       return -rte_errno;
>                               }
> +                     }
> +                     if (eth_spec && eth_mask && eth_mask->type) {
> +                             enum rte_flow_item_type next = (item + 1)->type;
>  
> -                             if ((eth_mask->type & UINT16_MAX) ==
> -                                 UINT16_MAX) {
> -                                     input_set |= I40E_INSET_LAST_ETHER_TYPE;
> -                                     filter->input.flow.l2_flow.ether_type =
> -                                             eth_spec->type;
> +                             if (eth_mask->type != RTE_BE16(0xffff)) {
> +                                     rte_flow_error_set(error, EINVAL,
> +                                                   RTE_FLOW_ERROR_TYPE_ITEM,
> +                                                   item,
> +                                                   "Invalid type mask.");
> +                                     return -rte_errno;
>                               }
>  
>                               ether_type = rte_be_to_cpu_16(eth_spec->type);
> -                             if (ether_type == ETHER_TYPE_IPv4 ||
> -                                 ether_type == ETHER_TYPE_IPv6 ||
> -                                 ether_type == ETHER_TYPE_ARP ||
> -                                 ether_type == outer_tpid) {
> +
> +                             if ((next == RTE_FLOW_ITEM_TYPE_VLAN &&
> +                                  ether_type != outer_tpid) ||
> +                                 (next != RTE_FLOW_ITEM_TYPE_VLAN &&
> +                                  (ether_type == ETHER_TYPE_IPv4 ||
> +                                   ether_type == ETHER_TYPE_IPv6 ||
> +                                   ether_type == ETHER_TYPE_ARP ||
> +                                   ether_type == outer_tpid))) {
>                                       rte_flow_error_set(error, EINVAL,
>                                                    RTE_FLOW_ERROR_TYPE_ITEM,
>                                                    item,
>                                                    "Unsupported ether_type.");
>                                       return -rte_errno;
> +                             } else if (next != RTE_FLOW_ITEM_TYPE_VLAN) {
> +                                     input_set |= I40E_INSET_LAST_ETHER_TYPE;
> +                                     filter->input.flow.l2_flow.ether_type =
> +                                             eth_spec->type;
>                               }
>                       }
>  
> @@ -2518,6 +2530,14 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
>               case RTE_FLOW_ITEM_TYPE_VLAN:
>                       vlan_spec = item->spec;
>                       vlan_mask = item->mask;
> +
> +                     if (input_set & I40E_INSET_LAST_ETHER_TYPE) {
> +                             rte_flow_error_set(error, EINVAL,
> +                                                RTE_FLOW_ERROR_TYPE_ITEM,
> +                                                item,
> +                                                "Unsupported outer TPID.");
> +                             return -rte_errno;
> +                     }
>                       if (vlan_spec && vlan_mask) {
>                               if (vlan_mask->tci ==
>                                   rte_cpu_to_be_16(I40E_TCI_MASK)) {
> @@ -2526,6 +2546,33 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
>                                               vlan_spec->tci;
>                               }
>                       }
> +                     if (vlan_spec && vlan_mask && vlan_mask->inner_type) {
> +                             if (vlan_mask->inner_type != RTE_BE16(0xffff)) {
> +                                     rte_flow_error_set(error, EINVAL,
> +                                                   RTE_FLOW_ERROR_TYPE_ITEM,
> +                                                   item,
> +                                                   "Invalid inner_type"
> +                                                   " mask.");
> +                                     return -rte_errno;
> +                             }
> +
> +                             ether_type =
> +                                     rte_be_to_cpu_16(vlan_spec->inner_type);
> +
> +                             if (ether_type == ETHER_TYPE_IPv4 ||
> +                                 ether_type == ETHER_TYPE_IPv6 ||
> +                                 ether_type == ETHER_TYPE_ARP ||
> +                                 ether_type == outer_tpid) {
> +                                     rte_flow_error_set(error, EINVAL,
> +                                                  RTE_FLOW_ERROR_TYPE_ITEM,
> +                                                  item,
> +                                                  "Unsupported inner_type.");
> +                                     return -rte_errno;
> +                             }
> +                             input_set |= I40E_INSET_LAST_ETHER_TYPE;
> +                             filter->input.flow.l2_flow.ether_type =
> +                                     vlan_spec->inner_type;
> +                     }
>  
>                       pctype = I40E_FILTER_PCTYPE_L2_PAYLOAD;
>                       layer_idx = I40E_FLXPLD_L2_IDX;
> @@ -3284,7 +3331,8 @@ i40e_flow_parse_vxlan_pattern(__rte_unused struct 
> rte_eth_dev *dev,
>               case RTE_FLOW_ITEM_TYPE_VLAN:
>                       vlan_spec = item->spec;
>                       vlan_mask = item->mask;
> -                     if (!(vlan_spec && vlan_mask)) {
> +                     if (!(vlan_spec && vlan_mask) ||
> +                         vlan_mask->inner_type) {
>                               rte_flow_error_set(error, EINVAL,
>                                                  RTE_FLOW_ERROR_TYPE_ITEM,
>                                                  item,
> @@ -3514,7 +3562,8 @@ i40e_flow_parse_nvgre_pattern(__rte_unused struct 
> rte_eth_dev *dev,
>               case RTE_FLOW_ITEM_TYPE_VLAN:
>                       vlan_spec = item->spec;
>                       vlan_mask = item->mask;
> -                     if (!(vlan_spec && vlan_mask)) {
> +                     if (!(vlan_spec && vlan_mask) ||
> +                         vlan_mask->inner_type) {
>                               rte_flow_error_set(error, EINVAL,
>                                                  RTE_FLOW_ERROR_TYPE_ITEM,
>                                                  item,
> @@ -4022,7 +4071,8 @@ i40e_flow_parse_qinq_pattern(__rte_unused struct 
> rte_eth_dev *dev,
>                       vlan_spec = item->spec;
>                       vlan_mask = item->mask;
>  
> -                     if (!(vlan_spec && vlan_mask)) {
> +                     if (!(vlan_spec && vlan_mask) ||
> +                         vlan_mask->inner_type) {
>                               rte_flow_error_set(error, EINVAL,
>                                          RTE_FLOW_ERROR_TYPE_ITEM,
>                                          item,
> diff --git a/drivers/net/mlx5/mlx5_flow.c b/drivers/net/mlx5/mlx5_flow.c
> index bc1176819..f03413d32 100644
> --- a/drivers/net/mlx5/mlx5_flow.c
> +++ b/drivers/net/mlx5/mlx5_flow.c
> @@ -17,6 +17,7 @@
>  #pragma GCC diagnostic error "-Wpedantic"
>  #endif
>  
> +#include <rte_byteorder.h>
>  #include <rte_common.h>
>  #include <rte_eth_ctrl.h>
>  #include <rte_ethdev_driver.h>
> @@ -306,6 +307,7 @@ static const struct mlx5_flow_items mlx5_flow_items[] = {
>               .actions = valid_actions,
>               .mask = &(const struct rte_flow_item_vlan){
>                       .tci = -1,
> +                     .inner_type = -1,
>               },
>               .default_mask = &rte_flow_item_vlan_mask,
>               .mask_sz = sizeof(struct rte_flow_item_vlan),
> @@ -1285,6 +1287,7 @@ mlx5_flow_create_vlan(const struct rte_flow_item *item,
>       struct mlx5_flow_parse *parser = data->parser;
>       struct ibv_flow_spec_eth *eth;
>       const unsigned int eth_size = sizeof(struct ibv_flow_spec_eth);
> +     const char *msg = "VLAN cannot be empty";
>  
>       if (spec) {
>               unsigned int i;
> @@ -1306,12 +1309,21 @@ mlx5_flow_create_vlan(const struct rte_flow_item 
> *item,
>                        */
>                       if (!eth->mask.vlan_tag)
>                               goto error;
> +                     /* Exactly one TPID value is allowed if specified. */
> +                     if ((eth->val.ether_type & eth->mask.ether_type) !=
> +                         (RTE_BE16(0x8100) & eth->mask.ether_type)) {

You can use ETHER_TYPE_VLAN present in rte_ether.h instead of hard coded
values.

> +                             msg = "unsupported outer TPID";
> +                             goto error;
> +                     }
> +                     eth->val.ether_type = spec->inner_type;
> +                     eth->mask.ether_type = mask->inner_type;
> +                     eth->val.ether_type &= eth->mask.ether_type;
>               }
>               return 0;
>       }
>  error:
>       return rte_flow_error_set(data->error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM,
> -                               item, "VLAN cannot be empty");
> +                               item, msg);
>  }
>  
>  /**
> diff --git a/drivers/net/mvpp2/mrvl_flow.c b/drivers/net/mvpp2/mrvl_flow.c
> index 8fd4dbfb1..6604a411f 100644
> --- a/drivers/net/mvpp2/mrvl_flow.c
> +++ b/drivers/net/mvpp2/mrvl_flow.c
> @@ -1091,12 +1091,6 @@ mrvl_parse_vlan(const struct rte_flow_item *item,
>       if (ret)
>               return ret;
>  
> -     if (mask->tpid) {
> -             rte_flow_error_set(error, EINVAL, RTE_FLOW_ERROR_TYPE_ITEM,
> -                                NULL, "Not supported by classifier\n");
> -             return -rte_errno;
> -     }
> -
>       m = rte_be_to_cpu_16(mask->tci);
>       if (m & MRVL_VLAN_ID_MASK) {
>               RTE_LOG(WARNING, PMD, "vlan id mask is ignored\n");
> @@ -1112,6 +1106,27 @@ mrvl_parse_vlan(const struct rte_flow_item *item,
>                       goto out;
>       }
>  
> +     if (flow->pattern & F_TYPE) {
> +             rte_flow_error_set(error, ENOTSUP,
> +                                RTE_FLOW_ERROR_TYPE_ITEM, item,
> +                                "outer TPID cannot be explicitly matched"
> +                                " when VLAN item is also specified\n");
> +             return -rte_errno;
> +     }
> +     if (mask->inner_type) {
> +             struct rte_flow_item_eth spec_eth = {
> +                     .type = spec->inner_type,
> +             };
> +             struct rte_flow_item_eth mask_eth = {
> +                     .type = mask->inner_type,
> +             };
> +
> +             RTE_LOG(WARNING, PMD, "inner eth type mask is ignored\n");
> +             ret = mrvl_parse_type(spec_eth, mask_eth, flow);
> +             if (ret)
> +                     goto out;
> +     }
> +
>       return 0;
>  out:
>       rte_flow_error_set(error, EINVAL, RTE_FLOW_ERROR_TYPE_UNSPECIFIED, NULL,
> diff --git a/drivers/net/sfc/sfc_flow.c b/drivers/net/sfc/sfc_flow.c
> index bf9609735..515a61325 100644
> --- a/drivers/net/sfc/sfc_flow.c
> +++ b/drivers/net/sfc/sfc_flow.c
> @@ -352,6 +352,7 @@ sfc_flow_parse_vlan(const struct rte_flow_item *item,
>       const struct rte_flow_item_vlan *mask = NULL;
>       const struct rte_flow_item_vlan supp_mask = {
>               .tci = rte_cpu_to_be_16(ETH_VLAN_ID_MAX),
> +             .inner_type = RTE_BE16(0xffff),
>       };
>  
>       rc = sfc_flow_parse_init(item,
> @@ -394,6 +395,32 @@ sfc_flow_parse_vlan(const struct rte_flow_item *item,
>               return -rte_errno;
>       }
>  
> +     /*
> +      * If an EtherType was already specified, make sure it is a valid
> +      * TPID for the current VLAN layer before overwriting it with the
> +      * specified inner type.
> +      */
> +     if (efx_spec->efs_match_flags & EFX_FILTER_MATCH_ETHER_TYPE &&
> +         efx_spec->efs_ether_type != RTE_BE16(0x8100) &&
> +         efx_spec->efs_ether_type != RTE_BE16(0x88a8)) {
> +             rte_flow_error_set(error, EINVAL,
> +                                RTE_FLOW_ERROR_TYPE_ITEM, item,
> +                                "Unsupported outer TPID");
> +             return -rte_errno;
> +     }
> +     if (!mask->inner_type) {
> +             efx_spec->efs_match_flags &= ~EFX_FILTER_MATCH_ETHER_TYPE;
> +             efx_spec->efs_ether_type = RTE_BE16(0x0000);
> +     } else if (mask->inner_type == supp_mask.inner_type) {
> +             efx_spec->efs_match_flags |= EFX_FILTER_MATCH_ETHER_TYPE;
> +             efx_spec->efs_ether_type = rte_be_to_cpu_16(spec->inner_type);
> +     } else {
> +             rte_flow_error_set(error, EINVAL,
> +                                RTE_FLOW_ERROR_TYPE_ITEM, item,
> +                                "Bad mask for VLAN inner_type");
> +             return -rte_errno;
> +     }
> +
>       return 0;
>  }
>  
> diff --git a/drivers/net/tap/tap_flow.c b/drivers/net/tap/tap_flow.c
> index e5eb50fc5..e53eff6ce 100644
> --- a/drivers/net/tap/tap_flow.c
> +++ b/drivers/net/tap/tap_flow.c
> @@ -270,13 +270,13 @@ static const struct tap_flow_items tap_flow_items[] = {
>               .items = ITEMS(RTE_FLOW_ITEM_TYPE_IPV4,
>                              RTE_FLOW_ITEM_TYPE_IPV6),
>               .mask = &(const struct rte_flow_item_vlan){
> -                     .tpid = -1,
>                       /* DEI matching is not supported */
>  #if RTE_BYTE_ORDER == RTE_LITTLE_ENDIAN
>                       .tci = 0xffef,
>  #else
>                       .tci = 0xefff,
>  #endif
> +                     .inner_type = -1,
>               },
>               .mask_sz = sizeof(struct rte_flow_item_vlan),
>               .default_mask = &rte_flow_item_vlan_mask,
> @@ -578,13 +578,21 @@ tap_flow_create_vlan(const struct rte_flow_item *item, 
> void *data)
>       /* use default mask if none provided */
>       if (!mask)
>               mask = tap_flow_items[RTE_FLOW_ITEM_TYPE_VLAN].default_mask;
> -     /* TC does not support tpid masking. Only accept if exact match. */
> -     if (mask->tpid && mask->tpid != 0xffff)
> +     /* check that previous eth type is compatible with VLAN */
> +     if (info->eth_type && info->eth_type != RTE_BE16(ETH_P_8021Q))
>               return -1;
>       /* Double-tagging not supported. */
> -     if (spec && mask->tpid && spec->tpid != htons(ETH_P_8021Q))
> +     if (info->vlan)
>               return -1;
>       info->vlan = 1;
> +     if (mask->inner_type) {
> +             /* TC does not support partial eth_type masking */
> +             if (mask->inner_type != RTE_BE16(0xffff))
> +                     return -1;
> +             info->eth_type = spec->inner_type;
> +     } else {
> +             info->eth_type = 0;
> +     }
>       if (!flow)
>               return 0;
>       msg = &flow->msg;
> diff --git a/lib/librte_ether/rte_flow.h b/lib/librte_ether/rte_flow.h
> index 1b222ba60..15e383f95 100644
> --- a/lib/librte_ether/rte_flow.h
> +++ b/lib/librte_ether/rte_flow.h
> @@ -454,11 +454,17 @@ static const struct rte_flow_item_raw 
> rte_flow_item_raw_mask = {
>   * RTE_FLOW_ITEM_TYPE_ETH
>   *
>   * Matches an Ethernet header.
> + *
> + * The @p type field either stands for "EtherType" or "TPID" when followed
> + * by so-called layer 2.5 pattern items such as RTE_FLOW_ITEM_TYPE_VLAN. In
> + * the latter case, @p type refers to that of the outer header, with the
> + * inner EtherType/TPID provided by the subsequent pattern item. This is the
> + * same order as on the wire.
>   */
>  struct rte_flow_item_eth {
>       struct ether_addr dst; /**< Destination MAC. */
>       struct ether_addr src; /**< Source MAC. */
> -     rte_be16_t type; /**< EtherType. */
> +     rte_be16_t type; /**< EtherType or TPID. */
>  };
>  
>  /** Default mask for RTE_FLOW_ITEM_TYPE_ETH. */
> @@ -475,19 +481,20 @@ static const struct rte_flow_item_eth 
> rte_flow_item_eth_mask = {
>   *
>   * Matches an 802.1Q/ad VLAN tag.
>   *
> - * This type normally follows either RTE_FLOW_ITEM_TYPE_ETH or
> - * RTE_FLOW_ITEM_TYPE_VLAN.
> + * The corresponding standard outer EtherType (TPID) values are 0x8100 or
> + * 0x88a8 (in case of outer QinQ). It can be overridden by the preceding
> + * pattern item.
>   */
>  struct rte_flow_item_vlan {
> -     rte_be16_t tpid; /**< Tag protocol identifier. */
>       rte_be16_t tci; /**< Tag control information. */
> +     rte_be16_t inner_type; /**< Inner EtherType or TPID. */
>  };
>  
>  /** Default mask for RTE_FLOW_ITEM_TYPE_VLAN. */
>  #ifndef __cplusplus
>  static const struct rte_flow_item_vlan rte_flow_item_vlan_mask = {
> -     .tpid = RTE_BE16(0x0000),
> -     .tci = RTE_BE16(0xffff),
> +     .tci = RTE_BE16(0x0fff),
> +     .inner_type = RTE_BE16(0x0000),
>  };
>  #endif
>  
> @@ -636,9 +643,11 @@ static const struct rte_flow_item_vxlan 
> rte_flow_item_vxlan_mask = {
>   * RTE_FLOW_ITEM_TYPE_E_TAG.
>   *
>   * Matches a E-tag header.
> + *
> + * The corresponding standard outer EtherType (TPID) value is 0x893f.
> + * It can be overridden by the preceding pattern item.
>   */
>  struct rte_flow_item_e_tag {
> -     rte_be16_t tpid; /**< Tag protocol identifier (0x893F). */
>       /**
>        * E-Tag control information (E-TCI).
>        * E-PCP (3b), E-DEI (1b), ingress E-CID base (12b).
> @@ -648,6 +657,7 @@ struct rte_flow_item_e_tag {
>       rte_be16_t rsvd_grp_ecid_b;
>       uint8_t in_ecid_e; /**< Ingress E-CID ext. */
>       uint8_t ecid_e; /**< E-CID ext. */
> +     rte_be16_t inner_type; /**< Inner EtherType or TPID. */
>  };
>  
>  /** Default mask for RTE_FLOW_ITEM_TYPE_E_TAG. */
> -- 
> 2.11.0

-- 
Nélio Laranjeiro
6WIND

Reply via email to