When OVS-DPDK is working as a pure L2 switch, Destination MAC match
offload with Mark+RSS action would help the performance speed up.
And FVL FDIR should support to change input set to be destination MAC.

When create a FDIR rule which pattern only has ether dst_mac like below:
flow create 0 ingress pattern eth dst is <mac_addr> / end
actions mark id <id> / rss / end
The following 11 pctypes of matched packets can Mark+RSS.
PCTYPE_NONF_IPV4_UDP
PCTYPE_NONF_IPV4_TCP
PCTYPE_NONF_IPV4_SCTP
PCTYPE_NONF_IPV4_OTHER
PCTYPE_FRAG_IPV4
PCTYPE_NONF_IPV6_UDP
PCTYPE_NONF_IPV6_TCP
PCTYPE_NONF_IPV6_SCTP
PCTYPE_NONF_IPV6_OTHER
PCTYPE_FRAG_IPV6
PCTYPE_L2_PAYLOAD

Signed-off-by: Lunyuan Cui <lunyuanx....@intel.com>
---
 drivers/net/i40e/i40e_ethdev.c |  92 +++++++-------
 drivers/net/i40e/i40e_ethdev.h |  10 +-
 drivers/net/i40e/i40e_fdir.c   |   8 +-
 drivers/net/i40e/i40e_flow.c   | 213 +++++++++++++++++++++++++--------
 4 files changed, 226 insertions(+), 97 deletions(-)

diff --git a/drivers/net/i40e/i40e_ethdev.c b/drivers/net/i40e/i40e_ethdev.c
index 9fbda1c34..bc5522d53 100644
--- a/drivers/net/i40e/i40e_ethdev.c
+++ b/drivers/net/i40e/i40e_ethdev.c
@@ -9337,15 +9337,16 @@ i40e_get_valid_input_set(enum i40e_filter_pctype pctype,
         */
        static const uint64_t valid_fdir_inset_table[] = {
                [I40E_FILTER_PCTYPE_FRAG_IPV4] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
-               I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_PROTO |
-               I40E_INSET_IPV4_TTL,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV4_SRC |
+               I40E_INSET_IPV4_DST | I40E_INSET_IPV4_TOS |
+               I40E_INSET_IPV4_PROTO | I40E_INSET_IPV4_TTL,
                [I40E_FILTER_PCTYPE_NONF_IPV4_UDP] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
-               I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_TTL |
-               I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV4_SRC |
+               I40E_INSET_IPV4_DST | I40E_INSET_IPV4_TOS |
+               I40E_INSET_IPV4_TTL | I40E_INSET_SRC_PORT |
+               I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_UNICAST_IPV4_UDP] =
                I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
                I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
@@ -9357,36 +9358,38 @@ i40e_get_valid_input_set(enum i40e_filter_pctype pctype,
                I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_TTL |
                I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_IPV4_TCP] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
-               I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_TTL |
-               I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV4_SRC |
+               I40E_INSET_IPV4_DST | I40E_INSET_IPV4_TOS |
+               I40E_INSET_IPV4_TTL | I40E_INSET_SRC_PORT |
+               I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_IPV4_TCP_SYN_NO_ACK] =
                I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
                I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
                I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_TTL |
                I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_IPV4_SCTP] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
-               I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_TTL |
-               I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT |
-               I40E_INSET_SCTP_VT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV4_SRC |
+               I40E_INSET_IPV4_DST | I40E_INSET_IPV4_TOS |
+               I40E_INSET_IPV4_TTL | I40E_INSET_SRC_PORT |
+               I40E_INSET_DST_PORT | I40E_INSET_SCTP_VT,
                [I40E_FILTER_PCTYPE_NONF_IPV4_OTHER] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV4_SRC | I40E_INSET_IPV4_DST |
-               I40E_INSET_IPV4_TOS | I40E_INSET_IPV4_PROTO |
-               I40E_INSET_IPV4_TTL,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV4_SRC |
+               I40E_INSET_IPV4_DST | I40E_INSET_IPV4_TOS |
+               I40E_INSET_IPV4_PROTO | I40E_INSET_IPV4_TTL,
                [I40E_FILTER_PCTYPE_FRAG_IPV6] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
-               I40E_INSET_IPV6_TC | I40E_INSET_IPV6_NEXT_HDR |
-               I40E_INSET_IPV6_HOP_LIMIT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV6_SRC |
+               I40E_INSET_IPV6_DST | I40E_INSET_IPV6_TC |
+               I40E_INSET_IPV6_NEXT_HDR | I40E_INSET_IPV6_HOP_LIMIT,
                [I40E_FILTER_PCTYPE_NONF_IPV6_UDP] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
-               I40E_INSET_IPV6_TC | I40E_INSET_IPV6_HOP_LIMIT |
-               I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV6_SRC |
+               I40E_INSET_IPV6_DST | I40E_INSET_IPV6_TC |
+               I40E_INSET_IPV6_HOP_LIMIT | I40E_INSET_SRC_PORT |
+               I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_UNICAST_IPV6_UDP] =
                I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
                I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
@@ -9398,29 +9401,30 @@ i40e_get_valid_input_set(enum i40e_filter_pctype pctype,
                I40E_INSET_IPV6_TC | I40E_INSET_IPV6_HOP_LIMIT |
                I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_IPV6_TCP] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
-               I40E_INSET_IPV6_TC | I40E_INSET_IPV6_HOP_LIMIT |
-               I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV6_SRC |
+               I40E_INSET_IPV6_DST | I40E_INSET_IPV6_TC |
+               I40E_INSET_IPV6_HOP_LIMIT | I40E_INSET_SRC_PORT |
+               I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_IPV6_TCP_SYN_NO_ACK] =
                I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
                I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
                I40E_INSET_IPV6_TC | I40E_INSET_IPV6_HOP_LIMIT |
                I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT,
                [I40E_FILTER_PCTYPE_NONF_IPV6_SCTP] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
-               I40E_INSET_IPV6_TC | I40E_INSET_IPV6_HOP_LIMIT |
-               I40E_INSET_SRC_PORT | I40E_INSET_DST_PORT |
-               I40E_INSET_SCTP_VT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV6_SRC |
+               I40E_INSET_IPV6_DST | I40E_INSET_IPV6_TC |
+               I40E_INSET_IPV6_HOP_LIMIT | I40E_INSET_SRC_PORT |
+               I40E_INSET_DST_PORT | I40E_INSET_SCTP_VT,
                [I40E_FILTER_PCTYPE_NONF_IPV6_OTHER] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_IPV6_SRC | I40E_INSET_IPV6_DST |
-               I40E_INSET_IPV6_TC | I40E_INSET_IPV6_NEXT_HDR |
-               I40E_INSET_IPV6_HOP_LIMIT,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_IPV6_SRC |
+               I40E_INSET_IPV6_DST | I40E_INSET_IPV6_TC |
+               I40E_INSET_IPV6_NEXT_HDR | I40E_INSET_IPV6_HOP_LIMIT,
                [I40E_FILTER_PCTYPE_L2_PAYLOAD] =
-               I40E_INSET_VLAN_OUTER | I40E_INSET_VLAN_INNER |
-               I40E_INSET_LAST_ETHER_TYPE,
+               I40E_INSET_DMAC | I40E_INSET_VLAN_OUTER |
+               I40E_INSET_VLAN_INNER | I40E_INSET_LAST_ETHER_TYPE,
        };
 
        if (pctype > I40E_FILTER_PCTYPE_L2_PAYLOAD)
diff --git a/drivers/net/i40e/i40e_ethdev.h b/drivers/net/i40e/i40e_ethdev.h
index aac89de91..e4f445081 100644
--- a/drivers/net/i40e/i40e_ethdev.h
+++ b/drivers/net/i40e/i40e_ethdev.h
@@ -544,12 +544,19 @@ struct i40e_ipv6_l2tpv3oip_flow {
        uint32_t session_id; /* Session ID in big endian. */
 };
 
+/* A structure used to define the input for l2 dst type flow */
+struct i40e_eth_l2_flow {
+       struct rte_ether_addr src;
+       struct rte_ether_addr dst;
+       struct rte_eth_l2_flow     i40e_l2_flow;
+};
+
 /*
  * A union contains the inputs for all types of flow
  * items in flows need to be in big endian
  */
 union i40e_fdir_flow {
-       struct rte_eth_l2_flow          l2_flow;
+       struct i40e_eth_l2_flow         l2_flow;
        struct rte_eth_udpv4_flow       udp4_flow;
        struct rte_eth_tcpv4_flow       tcp4_flow;
        struct rte_eth_sctpv4_flow      sctp4_flow;
@@ -637,6 +644,7 @@ struct i40e_fdir_filter_conf {
        /* ID, an unique value is required when deal with FDIR entry */
        struct i40e_fdir_input input;    /* Input set */
        struct i40e_fdir_action action;  /* Action taken when match */
+       bool fdir_dst_mac_rule; /* rule which only includes dst mac pattern */
 };
 
 /*
diff --git a/drivers/net/i40e/i40e_fdir.c b/drivers/net/i40e/i40e_fdir.c
index 931f25976..d926b1eda 100644
--- a/drivers/net/i40e/i40e_fdir.c
+++ b/drivers/net/i40e/i40e_fdir.c
@@ -1062,7 +1062,9 @@ i40e_flow_fdir_fill_eth_ip_head(struct i40e_pf *pf,
                [I40E_FILTER_PCTYPE_NONF_IPV6_OTHER] = IPPROTO_NONE,
        };
 
-       raw_pkt += 2 * sizeof(struct rte_ether_addr);
+       rte_memcpy(raw_pkt, &fdir_input->flow.l2_flow.dst,
+                       sizeof(struct rte_ether_addr));
+       raw_pkt += sizeof(struct rte_ether_addr);
        if (vlan && fdir_input->flow_ext.vlan_tci) {
                rte_memcpy(raw_pkt, vlan_frame, sizeof(vlan_frame));
                rte_memcpy(raw_pkt + sizeof(uint16_t),
@@ -1085,7 +1087,7 @@ i40e_flow_fdir_fill_eth_ip_head(struct i40e_pf *pf,
        }
 
        if (pctype == I40E_FILTER_PCTYPE_L2_PAYLOAD)
-               *ether_type = fdir_input->flow.l2_flow.ether_type;
+               *ether_type = fdir_input->flow.l2_flow.i40e_l2_flow.ether_type;
        else if (pctype == I40E_FILTER_PCTYPE_NONF_IPV4_TCP ||
                 pctype == I40E_FILTER_PCTYPE_NONF_IPV4_UDP ||
                 pctype == I40E_FILTER_PCTYPE_NONF_IPV4_SCTP ||
@@ -1271,7 +1273,7 @@ i40e_flow_fdir_construct_pkt(struct i40e_pf *pf,
                 * ARP packet is a special case on which the payload
                 * starts after the whole ARP header
                 */
-               if (fdir_input->flow.l2_flow.ether_type ==
+               if (fdir_input->flow.l2_flow.i40e_l2_flow.ether_type ==
                                rte_cpu_to_be_16(RTE_ETHER_TYPE_ARP))
                        payload += sizeof(struct rte_arp_hdr);
                set_idx = I40E_FLXPLD_L2_IDX;
diff --git a/drivers/net/i40e/i40e_flow.c b/drivers/net/i40e/i40e_flow.c
index d877ac250..cfd530a87 100644
--- a/drivers/net/i40e/i40e_flow.c
+++ b/drivers/net/i40e/i40e_flow.c
@@ -2595,6 +2595,7 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
        uint32_t vtc_flow_cpu;
        bool outer_ip = true;
        int ret;
+       uint16_t flow_type;
 
        memset(off_arr, 0, sizeof(off_arr));
        memset(len_arr, 0, sizeof(len_arr));
@@ -2626,8 +2627,16 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
                        }
 
                        if (eth_spec && eth_mask) {
-                               if (!rte_is_zero_ether_addr(&eth_mask->src) ||
-                                   !rte_is_zero_ether_addr(&eth_mask->dst)) {
+                               if (next_type == RTE_FLOW_ITEM_TYPE_END &&
+                               rte_is_broadcast_ether_addr(&eth_mask->dst) &&
+                               rte_is_zero_ether_addr(&eth_mask->src) &&
+                               rte_is_zero_ether_addr(&eth_spec->src)) {
+                                       
cons_filter.fdir_filter.fdir_dst_mac_rule = true;
+                                       filter->input.flow.l2_flow.dst =
+                                               eth_spec->dst;
+                                       input_set |= I40E_INSET_DMAC;
+                               } else if 
(!rte_is_zero_ether_addr(&eth_mask->src) ||
+                               !rte_is_zero_ether_addr(&eth_mask->dst)) {
                                        rte_flow_error_set(error, EINVAL,
                                                      RTE_FLOW_ERROR_TYPE_ITEM,
                                                      item,
@@ -2658,7 +2667,7 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
                                        return -rte_errno;
                                }
                                input_set |= I40E_INSET_LAST_ETHER_TYPE;
-                               filter->input.flow.l2_flow.ether_type =
+                               
filter->input.flow.l2_flow.i40e_l2_flow.ether_type =
                                        eth_spec->type;
                        }
 
@@ -2703,7 +2712,7 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
                                        return -rte_errno;
                                }
                                input_set |= I40E_INSET_LAST_ETHER_TYPE;
-                               filter->input.flow.l2_flow.ether_type =
+                               
filter->input.flow.l2_flow.i40e_l2_flow.ether_type =
                                        vlan_spec->inner_type;
                        }
 
@@ -3217,41 +3226,96 @@ i40e_flow_parse_fdir_pattern(struct rte_eth_dev *dev,
 
        /* If customized pctype is not used, set fdir configuration.*/
        if (!filter->input.flow_ext.customized_pctype) {
-               ret = i40e_flow_set_fdir_inset(pf, pctype, input_set);
-               if (ret == -1) {
-                       rte_flow_error_set(error, EINVAL,
-                                          RTE_FLOW_ERROR_TYPE_ITEM, item,
-                                          "Conflict with the first rule's 
input set.");
-                       return -rte_errno;
-               } else if (ret == -EINVAL) {
-                       rte_flow_error_set(error, EINVAL,
-                                          RTE_FLOW_ERROR_TYPE_ITEM, item,
-                                          "Invalid pattern mask.");
-                       return -rte_errno;
-               }
+               if (filter->fdir_dst_mac_rule) {
+                       for (pctype = I40E_FILTER_PCTYPE_NONF_IPV4_UDP;
+                               pctype <= I40E_FILTER_PCTYPE_L2_PAYLOAD; 
pctype++) {
+                               flow_type = 
i40e_pctype_to_flowtype(pf->adapter, pctype);
+
+                               if (flow_type == RTE_ETH_FLOW_UNKNOWN)
+                                       continue;
+
+                               ret = i40e_flow_set_fdir_inset(pf,
+                                       pctype,
+                                       input_set);
+                               if (ret == -1) {
+                                       rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                               "Conflict with the first rule's 
input set.");
+                                       return -rte_errno;
+                               } else if (ret == -EINVAL) {
+                                       rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                               "Invalid pattern mask.");
+                                       return -rte_errno;
+                               }
 
-               /* Store flex mask to SW */
-               ret = i40e_flow_store_flex_mask(pf, pctype, flex_mask);
-               if (ret == -1) {
-                       rte_flow_error_set(error, EINVAL,
-                                          RTE_FLOW_ERROR_TYPE_ITEM,
-                                          item,
-                                          "Exceed maximal number of bitmasks");
-                       return -rte_errno;
-               } else if (ret == -2) {
-                       rte_flow_error_set(error, EINVAL,
-                                          RTE_FLOW_ERROR_TYPE_ITEM,
-                                          item,
-                                          "Conflict with the first flexible 
rule");
-                       return -rte_errno;
-               } else if (ret > 0)
-                       cfg_flex_msk = false;
+                               /* Store flex mask to SW */
+                               ret = i40e_flow_store_flex_mask(pf,
+                                       pctype,
+                                       flex_mask);
+                               if (ret == -1) {
+                                       rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM,
+                                               item,
+                                               "Exceed maximal number of 
bitmasks");
+                                       return -rte_errno;
+                               } else if (ret == -2) {
+                                       rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM,
+                                               item,
+                                               "Conflict with the first 
flexible rule");
+                                       return -rte_errno;
+                               } else if (ret > 0) {
+                                       cfg_flex_msk = false;
+                               }
 
-               if (cfg_flex_pit)
-                       i40e_flow_set_fdir_flex_pit(pf, layer_idx, raw_id);
+                               if (cfg_flex_pit)
+                                       i40e_flow_set_fdir_flex_pit(pf,
+                                               layer_idx, raw_id);
 
-               if (cfg_flex_msk)
-                       i40e_flow_set_fdir_flex_msk(pf, pctype);
+                               if (cfg_flex_msk)
+                                       i40e_flow_set_fdir_flex_msk(pf,
+                                       pctype);
+                       }
+               } else {
+                       ret = i40e_flow_set_fdir_inset(pf, pctype, input_set);
+                       if (ret == -1) {
+                               rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                               "Conflict with the first rule's 
input set.");
+                               return -rte_errno;
+                       } else if (ret == -EINVAL) {
+                               rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM, item,
+                                               "Invalid pattern mask.");
+                               return -rte_errno;
+                       }
+
+                       /* Store flex mask to SW */
+                       ret = i40e_flow_store_flex_mask(pf, pctype, flex_mask);
+                       if (ret == -1) {
+                               rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM,
+                                               item,
+                                               "Exceed maximal number of 
bitmasks");
+                               return -rte_errno;
+                       } else if (ret == -2) {
+                               rte_flow_error_set(error, EINVAL,
+                                               RTE_FLOW_ERROR_TYPE_ITEM,
+                                               item,
+                                               "Conflict with the first 
flexible rule");
+                               return -rte_errno;
+                       } else if (ret > 0) {
+                               cfg_flex_msk = false;
+                       }
+
+                       if (cfg_flex_pit)
+                               i40e_flow_set_fdir_flex_pit(pf,
+                                       layer_idx, raw_id);
+
+                       if (cfg_flex_msk)
+                               i40e_flow_set_fdir_flex_msk(pf, pctype);
+               }
        }
 
        filter->input.pctype = pctype;
@@ -4897,6 +4961,8 @@ i40e_flow_create(struct rte_eth_dev *dev,
        struct i40e_pf *pf = I40E_DEV_PRIVATE_TO_PF(dev->data->dev_private);
        struct rte_flow *flow;
        int ret;
+       uint16_t flow_type;
+       enum i40e_filter_pctype pctype;
 
        flow = rte_zmalloc("i40e_flow", sizeof(struct rte_flow), 0);
        if (!flow) {
@@ -4920,10 +4986,32 @@ i40e_flow_create(struct rte_eth_dev *dev,
                                        i40e_ethertype_filter_list);
                break;
        case RTE_ETH_FILTER_FDIR:
-               ret = i40e_flow_add_del_fdir_filter(dev,
-                                      &cons_filter.fdir_filter, 1);
-               if (ret)
-                       goto free_flow;
+               if (!cons_filter.fdir_filter.fdir_dst_mac_rule) {
+                       ret = i40e_flow_add_del_fdir_filter(dev,
+                                               &cons_filter.fdir_filter, 1);
+                       if (ret)
+                               goto free_flow;
+
+                       flow->rule = TAILQ_LAST(&pf->fdir.fdir_list,
+                                       i40e_fdir_filter_list);
+                       break;
+               }
+
+               for (pctype = I40E_FILTER_PCTYPE_NONF_IPV4_UDP;
+                       pctype <= I40E_FILTER_PCTYPE_L2_PAYLOAD; pctype++) {
+                       flow_type = i40e_pctype_to_flowtype(pf->adapter, 
pctype);
+
+                       if (flow_type == RTE_ETH_FLOW_UNKNOWN)
+                               continue;
+
+                       cons_filter.fdir_filter.input.pctype =
+                                       pctype;
+                       ret = i40e_flow_add_del_fdir_filter(dev,
+                                               &cons_filter.fdir_filter, 1);
+                       if (ret)
+                               goto free_flow;
+               }
+
                flow->rule = TAILQ_LAST(&pf->fdir.fdir_list,
                                        i40e_fdir_filter_list);
                break;
@@ -4965,7 +5053,10 @@ i40e_flow_destroy(struct rte_eth_dev *dev,
 {
        struct i40e_pf *pf = I40E_DEV_PRIVATE_TO_PF(dev->data->dev_private);
        enum rte_filter_type filter_type = flow->filter_type;
+       struct i40e_fdir_filter *fdir_rule = NULL;
        int ret = 0;
+       uint16_t flow_type;
+       enum i40e_filter_pctype pctype;
 
        switch (filter_type) {
        case RTE_ETH_FILTER_ETHERTYPE:
@@ -4977,15 +5068,39 @@ i40e_flow_destroy(struct rte_eth_dev *dev,
                              (struct i40e_tunnel_filter *)flow->rule);
                break;
        case RTE_ETH_FILTER_FDIR:
-               ret = i40e_flow_add_del_fdir_filter(dev,
-                      &((struct i40e_fdir_filter *)flow->rule)->fdir, 0);
-
-               /* If the last flow is destroyed, disable fdir. */
-               if (!ret && TAILQ_EMPTY(&pf->fdir.fdir_list)) {
-                       i40e_fdir_teardown(pf);
-                       dev->data->dev_conf.fdir_conf.mode =
-                                  RTE_FDIR_MODE_NONE;
-                       i40e_fdir_rx_proc_enable(dev, 0);
+               fdir_rule = (struct i40e_fdir_filter *)flow->rule;
+               if (!fdir_rule->fdir.fdir_dst_mac_rule) {
+                       ret = i40e_flow_add_del_fdir_filter(dev,
+                       &fdir_rule->fdir, 0);
+
+                       /* If the last flow is destroyed, disable fdir. */
+                       if (!ret && TAILQ_EMPTY(&pf->fdir.fdir_list)) {
+                               i40e_fdir_teardown(pf);
+                               dev->data->dev_conf.fdir_conf.mode =
+                                       RTE_FDIR_MODE_NONE;
+                               i40e_fdir_rx_proc_enable(dev, 0);
+                       }
+                       break;
+               }
+
+               for (pctype = I40E_FILTER_PCTYPE_NONF_IPV4_UDP;
+                       pctype <= I40E_FILTER_PCTYPE_L2_PAYLOAD; pctype++) {
+                       flow_type = i40e_pctype_to_flowtype(pf->adapter, 
pctype);
+
+                       if (flow_type == RTE_ETH_FLOW_UNKNOWN)
+                               continue;
+
+                       fdir_rule->fdir.input.pctype = pctype;
+                       ret = i40e_flow_add_del_fdir_filter(dev,
+                               &fdir_rule->fdir, 0);
+
+                       /* Last flow is destroyed, disable fdir. */
+                       if (!ret && TAILQ_EMPTY(&pf->fdir.fdir_list)) {
+                               i40e_fdir_teardown(pf);
+                               dev->data->dev_conf.fdir_conf.mode =
+                                       RTE_FDIR_MODE_NONE;
+                               i40e_fdir_rx_proc_enable(dev, 0);
+                       }
                }
                break;
        case RTE_ETH_FILTER_HASH:
-- 
2.17.1

Reply via email to