]> www.infradead.org Git - users/willy/linux.git/commitdiff
ice: manage profiles and field vectors
authorDan Nowlin <dan.nowlin@intel.com>
Fri, 6 Aug 2021 08:48:59 +0000 (10:48 +0200)
committerTony Nguyen <anthony.l.nguyen@intel.com>
Mon, 11 Oct 2021 15:36:25 +0000 (08:36 -0700)
Implement functions to manage profiles and field vectors in hardware.

In hardware, there are up to 256 profiles and each of these profiles can
have 48 field vector words. Each field vector word is described by
protocol id and offset in the packet. To add a new recipe all used
profiles need to be searched. If the profile contains all required
protocol ids and offsets from the recipe it can be used. The driver has
to add this profile to recipe association to tell hardware that newly
added recipe is going to be associated with this profile.

The amount of used profiles depend on the package. To avoid searching
across not used profile, max profile id value is calculated at init flow.
The profile is considered as unused when all field vector words in the
profile are invalid (protocol id 0xff and offset 0x1ff).

Profiles are read from the package section ICE_SID_FLD_VEC_SW. Empty
field vector words can be used for recipe results. Store all unused field
vector words in prof_res_bm. It is a 256 elements array (max number of
profiles) each element is a 48 bit bitmap (max number of field vector
words).

For now, support only non-tunnel profiles type.

Co-developed-by: Grishma Kotecha <grishma.kotecha@intel.com>
Signed-off-by: Grishma Kotecha <grishma.kotecha@intel.com>
Signed-off-by: Dan Nowlin <dan.nowlin@intel.com>
Signed-off-by: Wojciech Drewek <wojciech.drewek@intel.com>
Tested-by: Sandeep Penigalapati <sandeep.penigalapati@intel.com>
Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com>
drivers/net/ethernet/intel/ice/ice_flex_pipe.c
drivers/net/ethernet/intel/ice/ice_flex_pipe.h
drivers/net/ethernet/intel/ice/ice_flex_type.h
drivers/net/ethernet/intel/ice/ice_switch.c
drivers/net/ethernet/intel/ice/ice_type.h

index 06ac9badee7748e57a99e14a075fec85f1f5e8a6..cd98b1d2f1a30937a6f5d77a456d5ae25cc3f503 100644 (file)
@@ -1329,6 +1329,86 @@ fw_ddp_compat_free_alloc:
        return status;
 }
 
+/**
+ * ice_sw_fv_handler
+ * @sect_type: section type
+ * @section: pointer to section
+ * @index: index of the field vector entry to be returned
+ * @offset: ptr to variable that receives the offset in the field vector table
+ *
+ * This is a callback function that can be passed to ice_pkg_enum_entry.
+ * This function treats the given section as of type ice_sw_fv_section and
+ * enumerates offset field. "offset" is an index into the field vector table.
+ */
+static void *
+ice_sw_fv_handler(u32 sect_type, void *section, u32 index, u32 *offset)
+{
+       struct ice_sw_fv_section *fv_section = section;
+
+       if (!section || sect_type != ICE_SID_FLD_VEC_SW)
+               return NULL;
+       if (index >= le16_to_cpu(fv_section->count))
+               return NULL;
+       if (offset)
+               /* "index" passed in to this function is relative to a given
+                * 4k block. To get to the true index into the field vector
+                * table need to add the relative index to the base_offset
+                * field of this section
+                */
+               *offset = le16_to_cpu(fv_section->base_offset) + index;
+       return fv_section->fv + index;
+}
+
+/**
+ * ice_get_prof_index_max - get the max profile index for used profile
+ * @hw: pointer to the HW struct
+ *
+ * Calling this function will get the max profile index for used profile
+ * and store the index number in struct ice_switch_info *switch_info
+ * in HW for following use.
+ */
+static enum ice_status ice_get_prof_index_max(struct ice_hw *hw)
+{
+       u16 prof_index = 0, j, max_prof_index = 0;
+       struct ice_pkg_enum state;
+       struct ice_seg *ice_seg;
+       bool flag = false;
+       struct ice_fv *fv;
+       u32 offset;
+
+       memset(&state, 0, sizeof(state));
+
+       if (!hw->seg)
+               return ICE_ERR_PARAM;
+
+       ice_seg = hw->seg;
+
+       do {
+               fv = ice_pkg_enum_entry(ice_seg, &state, ICE_SID_FLD_VEC_SW,
+                                       &offset, ice_sw_fv_handler);
+               if (!fv)
+                       break;
+               ice_seg = NULL;
+
+               /* in the profile that not be used, the prot_id is set to 0xff
+                * and the off is set to 0x1ff for all the field vectors.
+                */
+               for (j = 0; j < hw->blk[ICE_BLK_SW].es.fvw; j++)
+                       if (fv->ew[j].prot_id != ICE_PROT_INVALID ||
+                           fv->ew[j].off != ICE_FV_OFFSET_INVAL)
+                               flag = true;
+               if (flag && prof_index > max_prof_index)
+                       max_prof_index = prof_index;
+
+               prof_index++;
+               flag = false;
+       } while (fv);
+
+       hw->switch_info->max_used_prof_index = max_prof_index;
+
+       return 0;
+}
+
 /**
  * ice_init_pkg - initialize/download package
  * @hw: pointer to the hardware structure
@@ -1408,6 +1488,7 @@ enum ice_status ice_init_pkg(struct ice_hw *hw, u8 *buf, u32 len)
                 */
                ice_init_pkg_regs(hw);
                ice_fill_blk_tbls(hw);
+               ice_get_prof_index_max(hw);
        } else {
                ice_debug(hw, ICE_DBG_INIT, "package load failed, %d\n",
                          status);
@@ -1484,6 +1565,167 @@ static struct ice_buf_build *ice_pkg_buf_alloc(struct ice_hw *hw)
        return bld;
 }
 
+/**
+ * ice_get_sw_fv_bitmap - Get switch field vector bitmap based on profile type
+ * @hw: pointer to hardware structure
+ * @req_profs: type of profiles requested
+ * @bm: pointer to memory for returning the bitmap of field vectors
+ */
+void
+ice_get_sw_fv_bitmap(struct ice_hw *hw, enum ice_prof_type req_profs,
+                    unsigned long *bm)
+{
+       struct ice_pkg_enum state;
+       struct ice_seg *ice_seg;
+       struct ice_fv *fv;
+
+       if (req_profs == ICE_PROF_ALL) {
+               bitmap_set(bm, 0, ICE_MAX_NUM_PROFILES);
+               return;
+       }
+
+       memset(&state, 0, sizeof(state));
+       bitmap_zero(bm, ICE_MAX_NUM_PROFILES);
+       ice_seg = hw->seg;
+       do {
+               u32 offset;
+
+               fv = ice_pkg_enum_entry(ice_seg, &state, ICE_SID_FLD_VEC_SW,
+                                       &offset, ice_sw_fv_handler);
+               ice_seg = NULL;
+
+               if (fv) {
+                       if (req_profs & ICE_PROF_NON_TUN)
+                               set_bit((u16)offset, bm);
+               }
+       } while (fv);
+}
+
+/**
+ * ice_get_sw_fv_list
+ * @hw: pointer to the HW structure
+ * @prot_ids: field vector to search for with a given protocol ID
+ * @ids_cnt: lookup/protocol count
+ * @bm: bitmap of field vectors to consider
+ * @fv_list: Head of a list
+ *
+ * Finds all the field vector entries from switch block that contain
+ * a given protocol ID and returns a list of structures of type
+ * "ice_sw_fv_list_entry". Every structure in the list has a field vector
+ * definition and profile ID information
+ * NOTE: The caller of the function is responsible for freeing the memory
+ * allocated for every list entry.
+ */
+enum ice_status
+ice_get_sw_fv_list(struct ice_hw *hw, u8 *prot_ids, u16 ids_cnt,
+                  unsigned long *bm, struct list_head *fv_list)
+{
+       struct ice_sw_fv_list_entry *fvl;
+       struct ice_sw_fv_list_entry *tmp;
+       struct ice_pkg_enum state;
+       struct ice_seg *ice_seg;
+       struct ice_fv *fv;
+       u32 offset;
+
+       memset(&state, 0, sizeof(state));
+
+       if (!ids_cnt || !hw->seg)
+               return ICE_ERR_PARAM;
+
+       ice_seg = hw->seg;
+       do {
+               u16 i;
+
+               fv = ice_pkg_enum_entry(ice_seg, &state, ICE_SID_FLD_VEC_SW,
+                                       &offset, ice_sw_fv_handler);
+               if (!fv)
+                       break;
+               ice_seg = NULL;
+
+               /* If field vector is not in the bitmap list, then skip this
+                * profile.
+                */
+               if (!test_bit((u16)offset, bm))
+                       continue;
+
+               for (i = 0; i < ids_cnt; i++) {
+                       int j;
+
+                       /* This code assumes that if a switch field vector line
+                        * has a matching protocol, then this line will contain
+                        * the entries necessary to represent every field in
+                        * that protocol header.
+                        */
+                       for (j = 0; j < hw->blk[ICE_BLK_SW].es.fvw; j++)
+                               if (fv->ew[j].prot_id == prot_ids[i])
+                                       break;
+                       if (j >= hw->blk[ICE_BLK_SW].es.fvw)
+                               break;
+                       if (i + 1 == ids_cnt) {
+                               fvl = devm_kzalloc(ice_hw_to_dev(hw),
+                                                  sizeof(*fvl), GFP_KERNEL);
+                               if (!fvl)
+                                       goto err;
+                               fvl->fv_ptr = fv;
+                               fvl->profile_id = offset;
+                               list_add(&fvl->list_entry, fv_list);
+                               break;
+                       }
+               }
+       } while (fv);
+       if (list_empty(fv_list))
+               return ICE_ERR_CFG;
+       return 0;
+
+err:
+       list_for_each_entry_safe(fvl, tmp, fv_list, list_entry) {
+               list_del(&fvl->list_entry);
+               devm_kfree(ice_hw_to_dev(hw), fvl);
+       }
+
+       return ICE_ERR_NO_MEMORY;
+}
+
+/**
+ * ice_init_prof_result_bm - Initialize the profile result index bitmap
+ * @hw: pointer to hardware structure
+ */
+void ice_init_prof_result_bm(struct ice_hw *hw)
+{
+       struct ice_pkg_enum state;
+       struct ice_seg *ice_seg;
+       struct ice_fv *fv;
+
+       memset(&state, 0, sizeof(state));
+
+       if (!hw->seg)
+               return;
+
+       ice_seg = hw->seg;
+       do {
+               u32 off;
+               u16 i;
+
+               fv = ice_pkg_enum_entry(ice_seg, &state, ICE_SID_FLD_VEC_SW,
+                                       &off, ice_sw_fv_handler);
+               ice_seg = NULL;
+               if (!fv)
+                       break;
+
+               bitmap_zero(hw->switch_info->prof_res_bm[off],
+                           ICE_MAX_FV_WORDS);
+
+               /* Determine empty field vector indices, these can be
+                * used for recipe results. Skip index 0, since it is
+                * always used for Switch ID.
+                */
+               for (i = 1; i < ICE_MAX_FV_WORDS; i++)
+                       if (fv->ew[i].prot_id == ICE_PROT_INVALID &&
+                           fv->ew[i].off == ICE_FV_OFFSET_INVAL)
+                               set_bit(i, hw->switch_info->prof_res_bm[off]);
+       } while (fv);
+}
+
 /**
  * ice_pkg_buf_free
  * @hw: pointer to the HW structure
index 8a58e79729b983589b39b54f174122113f0d06ec..0cf5f0c241782a5adfdd9b6101be17cc3340881f 100644 (file)
 
 #define ICE_PKG_CNT 4
 
+void
+ice_get_sw_fv_bitmap(struct ice_hw *hw, enum ice_prof_type type,
+                    unsigned long *bm);
+void
+ice_init_prof_result_bm(struct ice_hw *hw);
+enum ice_status
+ice_get_sw_fv_list(struct ice_hw *hw, u8 *prot_ids, u16 ids_cnt,
+                  unsigned long *bm, struct list_head *fv_list);
 bool
 ice_get_open_tunnel_port(struct ice_hw *hw, u16 *port);
 int ice_udp_tunnel_set_port(struct net_device *netdev, unsigned int table,
index 7d8b517a63c9c2d876492d17623c0cb0a8e9072e..120bcebaa080c8e380dd371475f07201873da98e 100644 (file)
@@ -13,6 +13,8 @@ struct ice_fv_word {
        u8 resvrd;
 } __packed;
 
+#define ICE_MAX_NUM_PROFILES 256
+
 #define ICE_MAX_FV_WORDS 48
 struct ice_fv {
        struct ice_fv_word ew[ICE_MAX_FV_WORDS];
@@ -279,6 +281,12 @@ struct ice_sw_fv_section {
        struct ice_fv fv[];
 };
 
+struct ice_sw_fv_list_entry {
+       struct list_head list_entry;
+       u32 profile_id;
+       struct ice_fv *fv_ptr;
+};
+
 /* The BOOST TCAM stores the match packet header in reverse order, meaning
  * the fields are reversed; in addition, this means that the normally big endian
  * fields of the packet are now little endian.
@@ -603,4 +611,9 @@ struct ice_chs_chg {
 };
 
 #define ICE_FLOW_PTYPE_MAX             ICE_XLT1_CNT
+
+enum ice_prof_type {
+       ICE_PROF_NON_TUN = 0x1,
+       ICE_PROF_ALL = 0xFF,
+};
 #endif /* _ICE_FLEX_TYPE_H_ */
index becc4ff6b7c602e11fb7a348a74cfbcecf7412c6..dc0e79e77c490e5e2bfdd66d9960688b86dbc09c 100644 (file)
@@ -59,10 +59,11 @@ enum ice_status ice_init_def_sw_recp(struct ice_hw *hw)
        if (!recps)
                return ICE_ERR_NO_MEMORY;
 
-       for (i = 0; i < ICE_SW_LKUP_LAST; i++) {
+       for (i = 0; i < ICE_MAX_NUM_RECIPES; i++) {
                recps[i].root_rid = i;
                INIT_LIST_HEAD(&recps[i].filt_rules);
                INIT_LIST_HEAD(&recps[i].filt_replay_rules);
+               INIT_LIST_HEAD(&recps[i].rg_list);
                mutex_init(&recps[i].filt_rule_lock);
        }
 
index e064439fc1a92ca34a3877eb184384b8386169da..b2786783b80f057acf11f6eebc3981a7713be7f0 100644 (file)
@@ -677,6 +677,9 @@ struct ice_port_info {
 struct ice_switch_info {
        struct list_head vsi_list_map_head;
        struct ice_sw_recipe *recp_list;
+       u16 max_used_prof_index;
+
+       DECLARE_BITMAP(prof_res_bm[ICE_MAX_NUM_PROFILES], ICE_MAX_FV_WORDS);
 };
 
 /* FW logging configuration */