#include <net/devlink.h>
 #include <net/ipv6.h>
 #include <net/xdp_sock.h>
+#include <net/geneve.h>
+#include <net/gre.h>
+#include <net/udp_tunnel.h>
+#include <net/vxlan.h>
 #include "ice_devids.h"
 #include "ice_type.h"
 #include "ice_txrx.h"
 
        status = ice_init_hw_tbls(hw);
        if (status)
                goto err_unroll_fltr_mgmt_struct;
+       mutex_init(&hw->tnl_lock);
        return 0;
 
 err_unroll_fltr_mgmt_struct:
        ice_sched_clear_agg(hw);
        ice_free_seg(hw);
        ice_free_hw_tbls(hw);
+       mutex_destroy(&hw->tnl_lock);
 
        if (hw->port_info) {
                devm_kfree(ice_hw_to_dev(hw), hw->port_info);
 
 #include "ice_flex_pipe.h"
 #include "ice_flow.h"
 
+/* To support tunneling entries by PF, the package will append the PF number to
+ * the label; for example TNL_VXLAN_PF0, TNL_VXLAN_PF1, TNL_VXLAN_PF2, etc.
+ */
+static const struct ice_tunnel_type_scan tnls[] = {
+       { TNL_VXLAN,            "TNL_VXLAN_PF" },
+       { TNL_GENEVE,           "TNL_GENEVE_PF" },
+       { TNL_LAST,             "" }
+};
+
 static const u32 ice_sect_lkup[ICE_BLK_COUNT][ICE_SECT_COUNT] = {
        /* SWITCH */
        {
        return state->sect;
 }
 
+/**
+ * ice_pkg_enum_entry
+ * @ice_seg: pointer to the ice segment (or NULL on subsequent calls)
+ * @state: pointer to the enum state
+ * @sect_type: section type to enumerate
+ * @offset: pointer to variable that receives the offset in the table (optional)
+ * @handler: function that handles access to the entries into the section type
+ *
+ * This function will enumerate all the entries in particular section type in
+ * the ice segment. The first call is made with the ice_seg parameter non-NULL;
+ * on subsequent calls, ice_seg is set to NULL which continues the enumeration.
+ * When the function returns a NULL pointer, then the end of the entries has
+ * been reached.
+ *
+ * Since each section may have a different header and entry size, the handler
+ * function is needed to determine the number and location entries in each
+ * section.
+ *
+ * The offset parameter is optional, but should be used for sections that
+ * contain an offset for each section table. For such cases, the section handler
+ * function must return the appropriate offset + index to give the absolution
+ * offset for each entry. For example, if the base for a section's header
+ * indicates a base offset of 10, and the index for the entry is 2, then
+ * section handler function should set the offset to 10 + 2 = 12.
+ */
+static void *
+ice_pkg_enum_entry(struct ice_seg *ice_seg, struct ice_pkg_enum *state,
+                  u32 sect_type, u32 *offset,
+                  void *(*handler)(u32 sect_type, void *section,
+                                   u32 index, u32 *offset))
+{
+       void *entry;
+
+       if (ice_seg) {
+               if (!handler)
+                       return NULL;
+
+               if (!ice_pkg_enum_section(ice_seg, state, sect_type))
+                       return NULL;
+
+               state->entry_idx = 0;
+               state->handler = handler;
+       } else {
+               state->entry_idx++;
+       }
+
+       if (!state->handler)
+               return NULL;
+
+       /* get entry */
+       entry = state->handler(state->sect_type, state->sect, state->entry_idx,
+                              offset);
+       if (!entry) {
+               /* end of a section, look for another section of this type */
+               if (!ice_pkg_enum_section(NULL, state, 0))
+                       return NULL;
+
+               state->entry_idx = 0;
+               entry = state->handler(state->sect_type, state->sect,
+                                      state->entry_idx, offset);
+       }
+
+       return entry;
+}
+
+/**
+ * ice_boost_tcam_handler
+ * @sect_type: section type
+ * @section: pointer to section
+ * @index: index of the boost TCAM entry to be returned
+ * @offset: pointer to receive absolute offset, always 0 for boost TCAM sections
+ *
+ * This is a callback function that can be passed to ice_pkg_enum_entry.
+ * Handles enumeration of individual boost TCAM entries.
+ */
+static void *
+ice_boost_tcam_handler(u32 sect_type, void *section, u32 index, u32 *offset)
+{
+       struct ice_boost_tcam_section *boost;
+
+       if (!section)
+               return NULL;
+
+       if (sect_type != ICE_SID_RXPARSER_BOOST_TCAM)
+               return NULL;
+
+       if (index > ICE_MAX_BST_TCAMS_IN_BUF)
+               return NULL;
+
+       if (offset)
+               *offset = 0;
+
+       boost = section;
+       if (index >= le16_to_cpu(boost->count))
+               return NULL;
+
+       return boost->tcam + index;
+}
+
+/**
+ * ice_find_boost_entry
+ * @ice_seg: pointer to the ice segment (non-NULL)
+ * @addr: Boost TCAM address of entry to search for
+ * @entry: returns pointer to the entry
+ *
+ * Finds a particular Boost TCAM entry and returns a pointer to that entry
+ * if it is found. The ice_seg parameter must not be NULL since the first call
+ * to ice_pkg_enum_entry requires a pointer to an actual ice_segment structure.
+ */
+static enum ice_status
+ice_find_boost_entry(struct ice_seg *ice_seg, u16 addr,
+                    struct ice_boost_tcam_entry **entry)
+{
+       struct ice_boost_tcam_entry *tcam;
+       struct ice_pkg_enum state;
+
+       memset(&state, 0, sizeof(state));
+
+       if (!ice_seg)
+               return ICE_ERR_PARAM;
+
+       do {
+               tcam = ice_pkg_enum_entry(ice_seg, &state,
+                                         ICE_SID_RXPARSER_BOOST_TCAM, NULL,
+                                         ice_boost_tcam_handler);
+               if (tcam && le16_to_cpu(tcam->addr) == addr) {
+                       *entry = tcam;
+                       return 0;
+               }
+
+               ice_seg = NULL;
+       } while (tcam);
+
+       *entry = NULL;
+       return ICE_ERR_CFG;
+}
+
+/**
+ * ice_label_enum_handler
+ * @sect_type: section type
+ * @section: pointer to section
+ * @index: index of the label entry to be returned
+ * @offset: pointer to receive absolute offset, always zero for label sections
+ *
+ * This is a callback function that can be passed to ice_pkg_enum_entry.
+ * Handles enumeration of individual label entries.
+ */
+static void *
+ice_label_enum_handler(u32 __always_unused sect_type, void *section, u32 index,
+                      u32 *offset)
+{
+       struct ice_label_section *labels;
+
+       if (!section)
+               return NULL;
+
+       if (index > ICE_MAX_LABELS_IN_BUF)
+               return NULL;
+
+       if (offset)
+               *offset = 0;
+
+       labels = section;
+       if (index >= le16_to_cpu(labels->count))
+               return NULL;
+
+       return labels->label + index;
+}
+
+/**
+ * ice_enum_labels
+ * @ice_seg: pointer to the ice segment (NULL on subsequent calls)
+ * @type: the section type that will contain the label (0 on subsequent calls)
+ * @state: ice_pkg_enum structure that will hold the state of the enumeration
+ * @value: pointer to a value that will return the label's value if found
+ *
+ * Enumerates a list of labels in the package. The caller will call
+ * ice_enum_labels(ice_seg, type, ...) to start the enumeration, then call
+ * ice_enum_labels(NULL, 0, ...) to continue. When the function returns a NULL
+ * the end of the list has been reached.
+ */
+static char *
+ice_enum_labels(struct ice_seg *ice_seg, u32 type, struct ice_pkg_enum *state,
+               u16 *value)
+{
+       struct ice_label *label;
+
+       /* Check for valid label section on first call */
+       if (type && !(type >= ICE_SID_LBL_FIRST && type <= ICE_SID_LBL_LAST))
+               return NULL;
+
+       label = ice_pkg_enum_entry(ice_seg, state, type, NULL,
+                                  ice_label_enum_handler);
+       if (!label)
+               return NULL;
+
+       *value = le16_to_cpu(label->value);
+       return label->name;
+}
+
+/**
+ * ice_init_pkg_hints
+ * @hw: pointer to the HW structure
+ * @ice_seg: pointer to the segment of the package scan (non-NULL)
+ *
+ * This function will scan the package and save off relevant information
+ * (hints or metadata) for driver use. The ice_seg parameter must not be NULL
+ * since the first call to ice_enum_labels requires a pointer to an actual
+ * ice_seg structure.
+ */
+static void ice_init_pkg_hints(struct ice_hw *hw, struct ice_seg *ice_seg)
+{
+       struct ice_pkg_enum state;
+       char *label_name;
+       u16 val;
+       int i;
+
+       memset(&hw->tnl, 0, sizeof(hw->tnl));
+       memset(&state, 0, sizeof(state));
+
+       if (!ice_seg)
+               return;
+
+       label_name = ice_enum_labels(ice_seg, ICE_SID_LBL_RXPARSER_TMEM, &state,
+                                    &val);
+
+       while (label_name && hw->tnl.count < ICE_TUNNEL_MAX_ENTRIES) {
+               for (i = 0; tnls[i].type != TNL_LAST; i++) {
+                       size_t len = strlen(tnls[i].label_prefix);
+
+                       /* Look for matching label start, before continuing */
+                       if (strncmp(label_name, tnls[i].label_prefix, len))
+                               continue;
+
+                       /* Make sure this label matches our PF. Note that the PF
+                        * character ('0' - '7') will be located where our
+                        * prefix string's null terminator is located.
+                        */
+                       if ((label_name[len] - '0') == hw->pf_id) {
+                               hw->tnl.tbl[hw->tnl.count].type = tnls[i].type;
+                               hw->tnl.tbl[hw->tnl.count].valid = false;
+                               hw->tnl.tbl[hw->tnl.count].in_use = false;
+                               hw->tnl.tbl[hw->tnl.count].marked = false;
+                               hw->tnl.tbl[hw->tnl.count].boost_addr = val;
+                               hw->tnl.tbl[hw->tnl.count].port = 0;
+                               hw->tnl.count++;
+                               break;
+                       }
+               }
+
+               label_name = ice_enum_labels(NULL, 0, &state, &val);
+       }
+
+       /* Cache the appropriate boost TCAM entry pointers */
+       for (i = 0; i < hw->tnl.count; i++) {
+               ice_find_boost_entry(ice_seg, hw->tnl.tbl[i].boost_addr,
+                                    &hw->tnl.tbl[i].boost_entry);
+               if (hw->tnl.tbl[i].boost_entry)
+                       hw->tnl.tbl[i].valid = true;
+       }
+}
+
 /* Key creation */
 
 #define ICE_DC_KEY     0x1     /* don't care */
                return ICE_ERR_CFG;
        }
 
-       /* download package */
+       /* initialize package hints and then download package */
+       ice_init_pkg_hints(hw, seg);
        status = ice_download_pkg(hw, seg);
        if (status == ICE_ERR_AQ_NO_WORK) {
                ice_debug(hw, ICE_DBG_INIT,
        return &bld->buf;
 }
 
+/**
+ * ice_tunnel_port_in_use_hlpr - helper function to determine tunnel usage
+ * @hw: pointer to the HW structure
+ * @port: port to search for
+ * @index: optionally returns index
+ *
+ * Returns whether a port is already in use as a tunnel, and optionally its
+ * index
+ */
+static bool ice_tunnel_port_in_use_hlpr(struct ice_hw *hw, u16 port, u16 *index)
+{
+       u16 i;
+
+       for (i = 0; i < hw->tnl.count && i < ICE_TUNNEL_MAX_ENTRIES; i++)
+               if (hw->tnl.tbl[i].in_use && hw->tnl.tbl[i].port == port) {
+                       if (index)
+                               *index = i;
+                       return true;
+               }
+
+       return false;
+}
+
+/**
+ * ice_tunnel_port_in_use
+ * @hw: pointer to the HW structure
+ * @port: port to search for
+ * @index: optionally returns index
+ *
+ * Returns whether a port is already in use as a tunnel, and optionally its
+ * index
+ */
+bool ice_tunnel_port_in_use(struct ice_hw *hw, u16 port, u16 *index)
+{
+       bool res;
+
+       mutex_lock(&hw->tnl_lock);
+       res = ice_tunnel_port_in_use_hlpr(hw, port, index);
+       mutex_unlock(&hw->tnl_lock);
+
+       return res;
+}
+
+/**
+ * ice_find_free_tunnel_entry
+ * @hw: pointer to the HW structure
+ * @type: tunnel type
+ * @index: optionally returns index
+ *
+ * Returns whether there is a free tunnel entry, and optionally its index
+ */
+static bool
+ice_find_free_tunnel_entry(struct ice_hw *hw, enum ice_tunnel_type type,
+                          u16 *index)
+{
+       u16 i;
+
+       for (i = 0; i < hw->tnl.count && i < ICE_TUNNEL_MAX_ENTRIES; i++)
+               if (hw->tnl.tbl[i].valid && !hw->tnl.tbl[i].in_use &&
+                   hw->tnl.tbl[i].type == type) {
+                       if (index)
+                               *index = i;
+                       return true;
+               }
+
+       return false;
+}
+
+/**
+ * ice_create_tunnel
+ * @hw: pointer to the HW structure
+ * @type: type of tunnel
+ * @port: port of tunnel to create
+ *
+ * Create a tunnel by updating the parse graph in the parser. We do that by
+ * creating a package buffer with the tunnel info and issuing an update package
+ * command.
+ */
+enum ice_status
+ice_create_tunnel(struct ice_hw *hw, enum ice_tunnel_type type, u16 port)
+{
+       struct ice_boost_tcam_section *sect_rx, *sect_tx;
+       enum ice_status status = ICE_ERR_MAX_LIMIT;
+       struct ice_buf_build *bld;
+       u16 index;
+
+       mutex_lock(&hw->tnl_lock);
+
+       if (ice_tunnel_port_in_use_hlpr(hw, port, &index)) {
+               hw->tnl.tbl[index].ref++;
+               status = 0;
+               goto ice_create_tunnel_end;
+       }
+
+       if (!ice_find_free_tunnel_entry(hw, type, &index)) {
+               status = ICE_ERR_OUT_OF_RANGE;
+               goto ice_create_tunnel_end;
+       }
+
+       bld = ice_pkg_buf_alloc(hw);
+       if (!bld) {
+               status = ICE_ERR_NO_MEMORY;
+               goto ice_create_tunnel_end;
+       }
+
+       /* allocate 2 sections, one for Rx parser, one for Tx parser */
+       if (ice_pkg_buf_reserve_section(bld, 2))
+               goto ice_create_tunnel_err;
+
+       sect_rx = ice_pkg_buf_alloc_section(bld, ICE_SID_RXPARSER_BOOST_TCAM,
+                                           sizeof(*sect_rx));
+       if (!sect_rx)
+               goto ice_create_tunnel_err;
+       sect_rx->count = cpu_to_le16(1);
+
+       sect_tx = ice_pkg_buf_alloc_section(bld, ICE_SID_TXPARSER_BOOST_TCAM,
+                                           sizeof(*sect_tx));
+       if (!sect_tx)
+               goto ice_create_tunnel_err;
+       sect_tx->count = cpu_to_le16(1);
+
+       /* copy original boost entry to update package buffer */
+       memcpy(sect_rx->tcam, hw->tnl.tbl[index].boost_entry,
+              sizeof(*sect_rx->tcam));
+
+       /* over-write the never-match dest port key bits with the encoded port
+        * bits
+        */
+       ice_set_key((u8 *)§_rx->tcam[0].key, sizeof(sect_rx->tcam[0].key),
+                   (u8 *)&port, NULL, NULL, NULL,
+                   offsetof(struct ice_boost_key_value, hv_dst_port_key),
+                   sizeof(sect_rx->tcam[0].key.key.hv_dst_port_key));
+
+       /* exact copy of entry to Tx section entry */
+       memcpy(sect_tx->tcam, sect_rx->tcam, sizeof(*sect_tx->tcam));
+
+       status = ice_update_pkg(hw, ice_pkg_buf(bld), 1);
+       if (!status) {
+               hw->tnl.tbl[index].port = port;
+               hw->tnl.tbl[index].in_use = true;
+               hw->tnl.tbl[index].ref = 1;
+       }
+
+ice_create_tunnel_err:
+       ice_pkg_buf_free(hw, bld);
+
+ice_create_tunnel_end:
+       mutex_unlock(&hw->tnl_lock);
+
+       return status;
+}
+
+/**
+ * ice_destroy_tunnel
+ * @hw: pointer to the HW structure
+ * @port: port of tunnel to destroy (ignored if the all parameter is true)
+ * @all: flag that states to destroy all tunnels
+ *
+ * Destroys a tunnel or all tunnels by creating an update package buffer
+ * targeting the specific updates requested and then performing an update
+ * package.
+ */
+enum ice_status ice_destroy_tunnel(struct ice_hw *hw, u16 port, bool all)
+{
+       struct ice_boost_tcam_section *sect_rx, *sect_tx;
+       enum ice_status status = ICE_ERR_MAX_LIMIT;
+       struct ice_buf_build *bld;
+       u16 count = 0;
+       u16 index;
+       u16 size;
+       u16 i;
+
+       mutex_lock(&hw->tnl_lock);
+
+       if (!all && ice_tunnel_port_in_use_hlpr(hw, port, &index))
+               if (hw->tnl.tbl[index].ref > 1) {
+                       hw->tnl.tbl[index].ref--;
+                       status = 0;
+                       goto ice_destroy_tunnel_end;
+               }
+
+       /* determine count */
+       for (i = 0; i < hw->tnl.count && i < ICE_TUNNEL_MAX_ENTRIES; i++)
+               if (hw->tnl.tbl[i].valid && hw->tnl.tbl[i].in_use &&
+                   (all || hw->tnl.tbl[i].port == port))
+                       count++;
+
+       if (!count) {
+               status = ICE_ERR_PARAM;
+               goto ice_destroy_tunnel_end;
+       }
+
+       /* size of section - there is at least one entry */
+       size = struct_size(sect_rx, tcam, count - 1);
+
+       bld = ice_pkg_buf_alloc(hw);
+       if (!bld) {
+               status = ICE_ERR_NO_MEMORY;
+               goto ice_destroy_tunnel_end;
+       }
+
+       /* allocate 2 sections, one for Rx parser, one for Tx parser */
+       if (ice_pkg_buf_reserve_section(bld, 2))
+               goto ice_destroy_tunnel_err;
+
+       sect_rx = ice_pkg_buf_alloc_section(bld, ICE_SID_RXPARSER_BOOST_TCAM,
+                                           size);
+       if (!sect_rx)
+               goto ice_destroy_tunnel_err;
+       sect_rx->count = cpu_to_le16(1);
+
+       sect_tx = ice_pkg_buf_alloc_section(bld, ICE_SID_TXPARSER_BOOST_TCAM,
+                                           size);
+       if (!sect_tx)
+               goto ice_destroy_tunnel_err;
+       sect_tx->count = cpu_to_le16(1);
+
+       /* copy original boost entry to update package buffer, one copy to Rx
+        * section, another copy to the Tx section
+        */
+       for (i = 0; i < hw->tnl.count && i < ICE_TUNNEL_MAX_ENTRIES; i++)
+               if (hw->tnl.tbl[i].valid && hw->tnl.tbl[i].in_use &&
+                   (all || hw->tnl.tbl[i].port == port)) {
+                       memcpy(sect_rx->tcam + i, hw->tnl.tbl[i].boost_entry,
+                              sizeof(*sect_rx->tcam));
+                       memcpy(sect_tx->tcam + i, hw->tnl.tbl[i].boost_entry,
+                              sizeof(*sect_tx->tcam));
+                       hw->tnl.tbl[i].marked = true;
+               }
+
+       status = ice_update_pkg(hw, ice_pkg_buf(bld), 1);
+       if (!status)
+               for (i = 0; i < hw->tnl.count &&
+                    i < ICE_TUNNEL_MAX_ENTRIES; i++)
+                       if (hw->tnl.tbl[i].marked) {
+                               hw->tnl.tbl[i].ref = 0;
+                               hw->tnl.tbl[i].port = 0;
+                               hw->tnl.tbl[i].in_use = false;
+                               hw->tnl.tbl[i].marked = false;
+                       }
+
+ice_destroy_tunnel_err:
+       ice_pkg_buf_free(hw, bld);
+
+ice_destroy_tunnel_end:
+       mutex_unlock(&hw->tnl_lock);
+
+       return status;
+}
+
 /* PTG Management */
 
 /**
 
 
 #define ICE_PKG_CNT 4
 
+enum ice_status
+ice_create_tunnel(struct ice_hw *hw, enum ice_tunnel_type type, u16 port);
+enum ice_status ice_destroy_tunnel(struct ice_hw *hw, u16 port, bool all);
+bool ice_tunnel_port_in_use(struct ice_hw *hw, u16 port, u16 *index);
+
 enum ice_status
 ice_add_prof(struct ice_hw *hw, enum ice_block blk, u64 id, u8 ptypes[],
             struct ice_fv_word *es);
 
 #define ICE_SID_CDID_REDIR_RSS         48
 
 #define ICE_SID_RXPARSER_BOOST_TCAM    56
+#define ICE_SID_TXPARSER_BOOST_TCAM    66
 
 #define ICE_SID_XLT0_PE                        80
 #define ICE_SID_XLT_KEY_BUILDER_PE     81
        void *(*handler)(u32 sect_type, void *section, u32 index, u32 *offset);
 };
 
+/* Tunnel enabling */
+
+enum ice_tunnel_type {
+       TNL_VXLAN = 0,
+       TNL_GENEVE,
+       TNL_LAST = 0xFF,
+       TNL_ALL = 0xFF,
+};
+
+struct ice_tunnel_type_scan {
+       enum ice_tunnel_type type;
+       const char *label_prefix;
+};
+
+struct ice_tunnel_entry {
+       enum ice_tunnel_type type;
+       u16 boost_addr;
+       u16 port;
+       u16 ref;
+       struct ice_boost_tcam_entry *boost_entry;
+       u8 valid;
+       u8 in_use;
+       u8 marked;
+};
+
+#define ICE_TUNNEL_MAX_ENTRIES 16
+
+struct ice_tunnel_table {
+       struct ice_tunnel_entry tbl[ICE_TUNNEL_MAX_ENTRIES];
+       u16 count;
+};
+
 struct ice_pkg_es {
        __le16 count;
        __le16 offset;
 
        ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_SCTP, 0, sizeof(__be16)),
        /* ICE_FLOW_FIELD_IDX_SCTP_DST_PORT */
        ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_SCTP, 2, sizeof(__be16)),
-
+       /* GRE */
+       /* ICE_FLOW_FIELD_IDX_GRE_KEYID */
+       ICE_FLOW_FLD_INFO(ICE_FLOW_SEG_HDR_GRE, 12,
+                         sizeof_field(struct gre_full_hdr, key)),
 };
 
 /* Bitmaps indicating relevant packet types for a particular protocol header
        0x00000000, 0x00000000, 0x00000000, 0x00000000,
 };
 
+/* Packet types for packets with an Outermost/First GRE header */
+static const u32 ice_ptypes_gre_of[] = {
+       0x00000000, 0xBFBF7800, 0x000001DF, 0xFEFDE000,
+       0x0000017E, 0x00000000, 0x00000000, 0x00000000,
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+       0x00000000, 0x00000000, 0x00000000, 0x00000000,
+};
+
 /* Manage parameters and info. used during the creation of a flow profile */
 struct ice_flow_prof_params {
        enum ice_block blk;
                        src = (const unsigned long *)ice_ptypes_sctp_il;
                        bitmap_and(params->ptypes, params->ptypes, src,
                                   ICE_FLOW_PTYPE_MAX);
+               } else if (hdrs & ICE_FLOW_SEG_HDR_GRE) {
+                       if (!i) {
+                               src = (const unsigned long *)ice_ptypes_gre_of;
+                               bitmap_and(params->ptypes, params->ptypes,
+                                          src, ICE_FLOW_PTYPE_MAX);
+                       }
                }
        }
 
        case ICE_FLOW_FIELD_IDX_SCTP_DST_PORT:
                prot_id = ICE_PROT_SCTP_IL;
                break;
+       case ICE_FLOW_FIELD_IDX_GRE_KEYID:
+               prot_id = ICE_PROT_GRE_OF;
+               break;
        default:
                return ICE_ERR_NOT_IMPL;
        }
 #define ICE_FLOW_PROF_ENCAP_M  (BIT_ULL(ICE_FLOW_PROF_ENCAP_S))
 
 #define ICE_RSS_OUTER_HEADERS  1
+#define ICE_RSS_INNER_HEADERS  2
 
 /* Flow profile ID format:
  * [0:31] - Packet match fields
        mutex_lock(&hw->rss_locks);
        status = ice_add_rss_cfg_sync(hw, vsi_handle, hashed_flds, addl_hdrs,
                                      ICE_RSS_OUTER_HEADERS);
+       if (!status)
+               status = ice_add_rss_cfg_sync(hw, vsi_handle, hashed_flds,
+                                             addl_hdrs, ICE_RSS_INNER_HEADERS);
        mutex_unlock(&hw->rss_locks);
 
        return status;
                                                      ICE_RSS_OUTER_HEADERS);
                        if (status)
                                break;
+                       status = ice_add_rss_cfg_sync(hw, vsi_handle,
+                                                     r->hashed_flds,
+                                                     r->packet_hdr,
+                                                     ICE_RSS_INNER_HEADERS);
+                       if (status)
+                               break;
                }
        }
        mutex_unlock(&hw->rss_locks);
 
        ICE_FLOW_SEG_HDR_TCP            = 0x00000040,
        ICE_FLOW_SEG_HDR_UDP            = 0x00000080,
        ICE_FLOW_SEG_HDR_SCTP           = 0x00000100,
+       ICE_FLOW_SEG_HDR_GRE            = 0x00000200,
 };
 
 enum ice_flow_field {
        ICE_FLOW_FIELD_IDX_UDP_DST_PORT,
        ICE_FLOW_FIELD_IDX_SCTP_SRC_PORT,
        ICE_FLOW_FIELD_IDX_SCTP_DST_PORT,
+       /* GRE */
+       ICE_FLOW_FIELD_IDX_GRE_KEYID,
        /* The total number of enums must not exceed 64 */
        ICE_FLOW_FIELD_IDX_MAX
 };
 
        ICE_RX_FLEX_DESC_STATUS0_LAST /* this entry must be last!!! */
 };
 
+enum ice_rx_flex_desc_status_error_1_bits {
+       /* Note: These are predefined bit offsets */
+       ICE_RX_FLEX_DESC_STATUS1_NAT_S = 4,
+       ICE_RX_FLEX_DESC_STATUS1_LAST /* this entry must be last!!! */
+};
+
 #define ICE_RXQ_CTX_SIZE_DWORDS                8
 #define ICE_RXQ_CTX_SZ                 (ICE_RXQ_CTX_SIZE_DWORDS * sizeof(u32))
 #define ICE_TX_CMPLTNQ_CTX_SIZE_DWORDS 22
        ICE_TX_CTX_DESC_RESERVED        = 0x40
 };
 
+enum ice_tx_ctx_desc_eipt_offload {
+       ICE_TX_CTX_EIPT_NONE            = 0x0,
+       ICE_TX_CTX_EIPT_IPV6            = 0x1,
+       ICE_TX_CTX_EIPT_IPV4_NO_CSUM    = 0x2,
+       ICE_TX_CTX_EIPT_IPV4            = 0x3
+};
+
+#define ICE_TXD_CTX_QW0_EIPLEN_S       2
+
+#define ICE_TXD_CTX_QW0_L4TUNT_S       9
+
+#define ICE_TXD_CTX_UDP_TUNNELING      BIT_ULL(ICE_TXD_CTX_QW0_L4TUNT_S)
+#define ICE_TXD_CTX_GRE_TUNNELING      (0x2ULL << ICE_TXD_CTX_QW0_L4TUNT_S)
+
+#define ICE_TXD_CTX_QW0_NATLEN_S       12
+
+#define ICE_TXD_CTX_QW0_L4T_CS_S       23
+#define ICE_TXD_CTX_QW0_L4T_CS_M       BIT_ULL(ICE_TXD_CTX_QW0_L4T_CS_S)
+
 #define ICE_LAN_TXQ_MAX_QGRPS  127
 #define ICE_LAN_TXQ_MAX_QDIS   1023
 
 
                         NETIF_F_HW_VLAN_CTAG_TX     |
                         NETIF_F_HW_VLAN_CTAG_RX;
 
-       tso_features = NETIF_F_TSO              |
+       tso_features = NETIF_F_TSO                      |
+                      NETIF_F_TSO_ECN                  |
+                      NETIF_F_TSO6                     |
+                      NETIF_F_GSO_GRE                  |
+                      NETIF_F_GSO_UDP_TUNNEL           |
+                      NETIF_F_GSO_GRE_CSUM             |
+                      NETIF_F_GSO_UDP_TUNNEL_CSUM      |
+                      NETIF_F_GSO_PARTIAL              |
+                      NETIF_F_GSO_IPXIP4               |
+                      NETIF_F_GSO_IPXIP6               |
                       NETIF_F_GSO_UDP_L4;
 
+       netdev->gso_partial_features |= NETIF_F_GSO_UDP_TUNNEL_CSUM |
+                                       NETIF_F_GSO_GRE_CSUM;
        /* set features that user can change */
        netdev->hw_features = dflt_features | csumo_features |
                              vlano_features | tso_features;
 
+       /* add support for HW_CSUM on packets with MPLS header */
+       netdev->mpls_features =  NETIF_F_HW_CSUM;
+
        /* enable features */
        netdev->features |= netdev->hw_features;
        /* encap and VLAN devices inherit default, csumo and tso features */
        pf->tx_timeout_recovery_level++;
 }
 
+/**
+ * ice_udp_tunnel_add - Get notifications about UDP tunnel ports that come up
+ * @netdev: This physical port's netdev
+ * @ti: Tunnel endpoint information
+ */
+static void
+ice_udp_tunnel_add(struct net_device *netdev, struct udp_tunnel_info *ti)
+{
+       struct ice_netdev_priv *np = netdev_priv(netdev);
+       struct ice_vsi *vsi = np->vsi;
+       struct ice_pf *pf = vsi->back;
+       enum ice_tunnel_type tnl_type;
+       u16 port = ntohs(ti->port);
+       enum ice_status status;
+
+       switch (ti->type) {
+       case UDP_TUNNEL_TYPE_VXLAN:
+               tnl_type = TNL_VXLAN;
+               break;
+       case UDP_TUNNEL_TYPE_GENEVE:
+               tnl_type = TNL_GENEVE;
+               break;
+       default:
+               netdev_err(netdev, "Unknown tunnel type\n");
+               return;
+       }
+
+       status = ice_create_tunnel(&pf->hw, tnl_type, port);
+       if (status == ICE_ERR_OUT_OF_RANGE)
+               netdev_info(netdev, "Max tunneled UDP ports reached, port %d not added\n",
+                           port);
+       else if (status)
+               netdev_err(netdev, "Error adding UDP tunnel - %d\n",
+                          status);
+}
+
+/**
+ * ice_udp_tunnel_del - Get notifications about UDP tunnel ports that go away
+ * @netdev: This physical port's netdev
+ * @ti: Tunnel endpoint information
+ */
+static void
+ice_udp_tunnel_del(struct net_device *netdev, struct udp_tunnel_info *ti)
+{
+       struct ice_netdev_priv *np = netdev_priv(netdev);
+       struct ice_vsi *vsi = np->vsi;
+       struct ice_pf *pf = vsi->back;
+       u16 port = ntohs(ti->port);
+       enum ice_status status;
+       bool retval;
+
+       retval = ice_tunnel_port_in_use(&pf->hw, port, NULL);
+       if (!retval) {
+               netdev_info(netdev, "port %d not found in UDP tunnels list\n",
+                           port);
+               return;
+       }
+
+       status = ice_destroy_tunnel(&pf->hw, port, false);
+       if (status)
+               netdev_err(netdev, "error deleting port %d from UDP tunnels list\n",
+                          port);
+}
+
 /**
  * ice_open - Called when a network interface becomes active
  * @netdev: network interface device structure
        if (err)
                netdev_err(netdev, "Failed to open VSI 0x%04X on switch 0x%04X\n",
                           vsi->vsi_num, vsi->vsw->sw_id);
+
+       /* Update existing tunnels information */
+       udp_tunnel_get_rx_info(netdev);
+
        return err;
 }
 
                features &= ~NETIF_F_GSO_MASK;
 
        len = skb_network_header(skb) - skb->data;
-       if (len & ~(ICE_TXD_MACLEN_MAX))
+       if (len > ICE_TXD_MACLEN_MAX || len & 0x1)
                goto out_rm_features;
 
        len = skb_transport_header(skb) - skb_network_header(skb);
-       if (len & ~(ICE_TXD_IPLEN_MAX))
+       if (len > ICE_TXD_IPLEN_MAX || len & 0x1)
                goto out_rm_features;
 
        if (skb->encapsulation) {
                len = skb_inner_network_header(skb) - skb_transport_header(skb);
-               if (len & ~(ICE_TXD_L4LEN_MAX))
+               if (len > ICE_TXD_L4LEN_MAX || len & 0x1)
                        goto out_rm_features;
 
                len = skb_inner_transport_header(skb) -
                      skb_inner_network_header(skb);
-               if (len & ~(ICE_TXD_IPLEN_MAX))
+               if (len > ICE_TXD_IPLEN_MAX || len & 0x1)
                        goto out_rm_features;
        }
 
        .ndo_bpf = ice_xdp,
        .ndo_xdp_xmit = ice_xdp_xmit,
        .ndo_xsk_wakeup = ice_xsk_wakeup,
+       .ndo_udp_tunnel_add = ice_udp_tunnel_add,
+       .ndo_udp_tunnel_del = ice_udp_tunnel_del,
 };
 
        ICE_PROT_IPV6_IL        = 41,
        ICE_PROT_TCP_IL         = 49,
        ICE_PROT_UDP_IL_OR_S    = 53,
+       ICE_PROT_GRE_OF         = 64,
        ICE_PROT_SCTP_IL        = 96,
        ICE_PROT_META_ID        = 255, /* when offset == metadata */
        ICE_PROT_INVALID        = 255  /* when offset == ICE_FV_OFFSET_INVAL */
 
        l2_len = ip.hdr - skb->data;
        offset = (l2_len / 2) << ICE_TX_DESC_LEN_MACLEN_S;
 
-       if (skb->encapsulation)
-               return -1;
+       protocol = vlan_get_protocol(skb);
+
+       if (protocol == htons(ETH_P_IP))
+               first->tx_flags |= ICE_TX_FLAGS_IPV4;
+       else if (protocol == htons(ETH_P_IPV6))
+               first->tx_flags |= ICE_TX_FLAGS_IPV6;
+
+       if (skb->encapsulation) {
+               bool gso_ena = false;
+               u32 tunnel = 0;
+
+               /* define outer network header type */
+               if (first->tx_flags & ICE_TX_FLAGS_IPV4) {
+                       tunnel |= (first->tx_flags & ICE_TX_FLAGS_TSO) ?
+                                 ICE_TX_CTX_EIPT_IPV4 :
+                                 ICE_TX_CTX_EIPT_IPV4_NO_CSUM;
+                       l4_proto = ip.v4->protocol;
+               } else if (first->tx_flags & ICE_TX_FLAGS_IPV6) {
+                       tunnel |= ICE_TX_CTX_EIPT_IPV6;
+                       exthdr = ip.hdr + sizeof(*ip.v6);
+                       l4_proto = ip.v6->nexthdr;
+                       if (l4.hdr != exthdr)
+                               ipv6_skip_exthdr(skb, exthdr - skb->data,
+                                                &l4_proto, &frag_off);
+               }
+
+               /* define outer transport */
+               switch (l4_proto) {
+               case IPPROTO_UDP:
+                       tunnel |= ICE_TXD_CTX_UDP_TUNNELING;
+                       first->tx_flags |= ICE_TX_FLAGS_TUNNEL;
+                       break;
+               case IPPROTO_GRE:
+                       tunnel |= ICE_TXD_CTX_GRE_TUNNELING;
+                       first->tx_flags |= ICE_TX_FLAGS_TUNNEL;
+                       break;
+               case IPPROTO_IPIP:
+               case IPPROTO_IPV6:
+                       first->tx_flags |= ICE_TX_FLAGS_TUNNEL;
+                       l4.hdr = skb_inner_network_header(skb);
+                       break;
+               default:
+                       if (first->tx_flags & ICE_TX_FLAGS_TSO)
+                               return -1;
+
+                       skb_checksum_help(skb);
+                       return 0;
+               }
+
+               /* compute outer L3 header size */
+               tunnel |= ((l4.hdr - ip.hdr) / 4) <<
+                         ICE_TXD_CTX_QW0_EIPLEN_S;
+
+               /* switch IP header pointer from outer to inner header */
+               ip.hdr = skb_inner_network_header(skb);
+
+               /* compute tunnel header size */
+               tunnel |= ((ip.hdr - l4.hdr) / 2) <<
+                          ICE_TXD_CTX_QW0_NATLEN_S;
+
+               gso_ena = skb_shinfo(skb)->gso_type & SKB_GSO_PARTIAL;
+               /* indicate if we need to offload outer UDP header */
+               if ((first->tx_flags & ICE_TX_FLAGS_TSO) && !gso_ena &&
+                   (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM))
+                       tunnel |= ICE_TXD_CTX_QW0_L4T_CS_M;
+
+               /* record tunnel offload values */
+               off->cd_tunnel_params |= tunnel;
+
+               /* set DTYP=1 to indicate that it's an Tx context descriptor
+                * in IPsec tunnel mode with Tx offloads in Quad word 1
+                */
+               off->cd_qw1 |= (u64)ICE_TX_DESC_DTYPE_CTX;
+
+               /* switch L4 header pointer from outer to inner */
+               l4.hdr = skb_inner_transport_header(skb);
+               l4_proto = 0;
+
+               /* reset type as we transition from outer to inner headers */
+               first->tx_flags &= ~(ICE_TX_FLAGS_IPV4 | ICE_TX_FLAGS_IPV6);
+               if (ip.v4->version == 4)
+                       first->tx_flags |= ICE_TX_FLAGS_IPV4;
+               if (ip.v6->version == 6)
+                       first->tx_flags |= ICE_TX_FLAGS_IPV6;
+       }
 
        /* Enable IP checksum offloads */
-       protocol = vlan_get_protocol(skb);
-       if (protocol == htons(ETH_P_IP)) {
+       if (first->tx_flags & ICE_TX_FLAGS_IPV4) {
                l4_proto = ip.v4->protocol;
                /* the stack computes the IP header already, the only time we
                 * need the hardware to recompute it is in the case of TSO.
                else
                        cmd |= ICE_TX_DESC_CMD_IIPT_IPV4;
 
-       } else if (protocol == htons(ETH_P_IPV6)) {
+       } else if (first->tx_flags & ICE_TX_FLAGS_IPV6) {
                cmd |= ICE_TX_DESC_CMD_IIPT_IPV6;
                exthdr = ip.hdr + sizeof(*ip.v6);
                l4_proto = ip.v6->nexthdr;
                ip.v6->payload_len = 0;
        }
 
+       if (skb_shinfo(skb)->gso_type & (SKB_GSO_GRE |
+                                        SKB_GSO_GRE_CSUM |
+                                        SKB_GSO_IPXIP4 |
+                                        SKB_GSO_IPXIP6 |
+                                        SKB_GSO_UDP_TUNNEL |
+                                        SKB_GSO_UDP_TUNNEL_CSUM)) {
+               if (!(skb_shinfo(skb)->gso_type & SKB_GSO_PARTIAL) &&
+                   (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM)) {
+                       l4.udp->len = 0;
+
+                       /* determine offset of outer transport header */
+                       l4_start = l4.hdr - skb->data;
+
+                       /* remove payload length from outer checksum */
+                       paylen = skb->len - l4_start;
+                       csum_replace_by_diff(&l4.udp->check,
+                                            (__force __wsum)htonl(paylen));
+               }
+
+               /* reset pointers to inner headers */
+
+               /* cppcheck-suppress unreadVariable */
+               ip.hdr = skb_inner_network_header(skb);
+               l4.hdr = skb_inner_transport_header(skb);
+
+               /* initialize inner IP header fields */
+               if (ip.v4->version == 4) {
+                       ip.v4->tot_len = 0;
+                       ip.v4->check = 0;
+               } else {
+                       ip.v6->payload_len = 0;
+               }
+       }
+
        /* determine offset of transport header */
        l4_start = l4.hdr - skb->data;
 
 
 #define ICE_TX_FLAGS_TSO       BIT(0)
 #define ICE_TX_FLAGS_HW_VLAN   BIT(1)
 #define ICE_TX_FLAGS_SW_VLAN   BIT(2)
+#define ICE_TX_FLAGS_IPV4      BIT(5)
+#define ICE_TX_FLAGS_IPV6      BIT(6)
+#define ICE_TX_FLAGS_TUNNEL    BIT(7)
 #define ICE_TX_FLAGS_VLAN_M    0xffff0000
 #define ICE_TX_FLAGS_VLAN_PR_M 0xe0000000
 #define ICE_TX_FLAGS_VLAN_PR_S 29
 
            union ice_32b_rx_flex_desc *rx_desc, u8 ptype)
 {
        struct ice_rx_ptype_decoded decoded;
-       u32 rx_error, rx_status;
+       u16 rx_error, rx_status;
+       u16 rx_stat_err1;
        bool ipv4, ipv6;
 
        rx_status = le16_to_cpu(rx_desc->wb.status_error0);
-       rx_error = rx_status;
+       rx_error = rx_status & (BIT(ICE_RX_FLEX_DESC_STATUS0_XSUM_IPE_S) |
+                               BIT(ICE_RX_FLEX_DESC_STATUS0_XSUM_L4E_S) |
+                               BIT(ICE_RX_FLEX_DESC_STATUS0_XSUM_EIPE_S) |
+                               BIT(ICE_RX_FLEX_DESC_STATUS0_XSUM_EUDPE_S));
 
+       rx_stat_err1 = le16_to_cpu(rx_desc->wb.status_error1);
        decoded = ice_decode_rx_desc_ptype(ptype);
 
        /* Start with CHECKSUM_NONE and by default csum_level = 0 */
        if (rx_error & BIT(ICE_RX_FLEX_DESC_STATUS0_XSUM_L4E_S))
                goto checksum_fail;
 
+       /* check for outer UDP checksum error in tunneled packets */
+       if ((rx_stat_err1 & BIT(ICE_RX_FLEX_DESC_STATUS1_NAT_S)) &&
+           (rx_error & BIT(ICE_RX_FLEX_DESC_STATUS0_XSUM_EUDPE_S)))
+               goto checksum_fail;
+
+       /* If there is an outer header present that might contain a checksum
+        * we need to bump the checksum level by 1 to reflect the fact that
+        * we are indicating we validated the inner checksum.
+        */
+       if (decoded.tunnel_type >= ICE_RX_PTYPE_TUNNEL_IP_GRENAT)
+               skb->csum_level = 1;
+
        /* Only report checksum unnecessary for TCP, UDP, or SCTP */
        switch (decoded.inner_prot) {
        case ICE_RX_PTYPE_INNER_PROT_TCP:
 
        u8 *pkg_copy;
        u32 pkg_size;
 
+       /* tunneling info */
+       struct mutex tnl_lock;
+       struct ice_tunnel_table tnl;
+
        /* HW block tables */
        struct ice_blk_info blk[ICE_BLK_COUNT];
        struct mutex fl_profs_locks[ICE_BLK_COUNT];     /* lock fltr profiles */