ice: adjust profile ID map locks

The profile ID map lock should be held till the caller completes
all references of that profile entries.

The current code releases the lock right after the match search.
This caused a driver issue when the profile map entries were
referenced after it was freed in other thread after the lock was
released earlier.

Signed-off-by: Victor Raj <victor.raj@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com>
This commit is contained in:
Victor Raj 2020-07-29 17:19:21 -07:00 committed by Tony Nguyen
parent eddbee9b94
commit 6a2c2b2c1b
1 changed files with 45 additions and 45 deletions

View File

@ -3878,16 +3878,16 @@ ice_add_prof(struct ice_hw *hw, enum ice_block blk, u64 id, u8 ptypes[],
} }
/** /**
* ice_search_prof_id_low - Search for a profile tracking ID low level * ice_search_prof_id - Search for a profile tracking ID
* @hw: pointer to the HW struct * @hw: pointer to the HW struct
* @blk: hardware block * @blk: hardware block
* @id: profile tracking ID * @id: profile tracking ID
* *
* This will search for a profile tracking ID which was previously added. This * This will search for a profile tracking ID which was previously added.
* version assumes that the caller has already acquired the prof map lock. * The profile map lock should be held before calling this function.
*/ */
static struct ice_prof_map * static struct ice_prof_map *
ice_search_prof_id_low(struct ice_hw *hw, enum ice_block blk, u64 id) ice_search_prof_id(struct ice_hw *hw, enum ice_block blk, u64 id)
{ {
struct ice_prof_map *entry = NULL; struct ice_prof_map *entry = NULL;
struct ice_prof_map *map; struct ice_prof_map *map;
@ -3901,26 +3901,6 @@ ice_search_prof_id_low(struct ice_hw *hw, enum ice_block blk, u64 id)
return entry; return entry;
} }
/**
* ice_search_prof_id - Search for a profile tracking ID
* @hw: pointer to the HW struct
* @blk: hardware block
* @id: profile tracking ID
*
* This will search for a profile tracking ID which was previously added.
*/
static struct ice_prof_map *
ice_search_prof_id(struct ice_hw *hw, enum ice_block blk, u64 id)
{
struct ice_prof_map *entry;
mutex_lock(&hw->blk[blk].es.prof_map_lock);
entry = ice_search_prof_id_low(hw, blk, id);
mutex_unlock(&hw->blk[blk].es.prof_map_lock);
return entry;
}
/** /**
* ice_vsig_prof_id_count - count profiles in a VSIG * ice_vsig_prof_id_count - count profiles in a VSIG
* @hw: pointer to the HW struct * @hw: pointer to the HW struct
@ -4137,7 +4117,7 @@ enum ice_status ice_rem_prof(struct ice_hw *hw, enum ice_block blk, u64 id)
mutex_lock(&hw->blk[blk].es.prof_map_lock); mutex_lock(&hw->blk[blk].es.prof_map_lock);
pmap = ice_search_prof_id_low(hw, blk, id); pmap = ice_search_prof_id(hw, blk, id);
if (!pmap) { if (!pmap) {
status = ICE_ERR_DOES_NOT_EXIST; status = ICE_ERR_DOES_NOT_EXIST;
goto err_ice_rem_prof; goto err_ice_rem_prof;
@ -4170,22 +4150,28 @@ static enum ice_status
ice_get_prof(struct ice_hw *hw, enum ice_block blk, u64 hdl, ice_get_prof(struct ice_hw *hw, enum ice_block blk, u64 hdl,
struct list_head *chg) struct list_head *chg)
{ {
enum ice_status status = 0;
struct ice_prof_map *map; struct ice_prof_map *map;
struct ice_chs_chg *p; struct ice_chs_chg *p;
u16 i; u16 i;
mutex_lock(&hw->blk[blk].es.prof_map_lock);
/* Get the details on the profile specified by the handle ID */ /* Get the details on the profile specified by the handle ID */
map = ice_search_prof_id(hw, blk, hdl); map = ice_search_prof_id(hw, blk, hdl);
if (!map) if (!map) {
return ICE_ERR_DOES_NOT_EXIST; status = ICE_ERR_DOES_NOT_EXIST;
goto err_ice_get_prof;
}
for (i = 0; i < map->ptg_cnt; i++) for (i = 0; i < map->ptg_cnt; i++)
if (!hw->blk[blk].es.written[map->prof_id]) { if (!hw->blk[blk].es.written[map->prof_id]) {
/* add ES to change list */ /* add ES to change list */
p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p),
GFP_KERNEL); GFP_KERNEL);
if (!p) if (!p) {
status = ICE_ERR_NO_MEMORY;
goto err_ice_get_prof; goto err_ice_get_prof;
}
p->type = ICE_PTG_ES_ADD; p->type = ICE_PTG_ES_ADD;
p->ptype = 0; p->ptype = 0;
@ -4200,11 +4186,10 @@ ice_get_prof(struct ice_hw *hw, enum ice_block blk, u64 hdl,
list_add(&p->list_entry, chg); list_add(&p->list_entry, chg);
} }
return 0;
err_ice_get_prof: err_ice_get_prof:
mutex_unlock(&hw->blk[blk].es.prof_map_lock);
/* let caller clean up the change list */ /* let caller clean up the change list */
return ICE_ERR_NO_MEMORY; return status;
} }
/** /**
@ -4258,17 +4243,23 @@ static enum ice_status
ice_add_prof_to_lst(struct ice_hw *hw, enum ice_block blk, ice_add_prof_to_lst(struct ice_hw *hw, enum ice_block blk,
struct list_head *lst, u64 hdl) struct list_head *lst, u64 hdl)
{ {
enum ice_status status = 0;
struct ice_prof_map *map; struct ice_prof_map *map;
struct ice_vsig_prof *p; struct ice_vsig_prof *p;
u16 i; u16 i;
mutex_lock(&hw->blk[blk].es.prof_map_lock);
map = ice_search_prof_id(hw, blk, hdl); map = ice_search_prof_id(hw, blk, hdl);
if (!map) if (!map) {
return ICE_ERR_DOES_NOT_EXIST; status = ICE_ERR_DOES_NOT_EXIST;
goto err_ice_add_prof_to_lst;
}
p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL);
if (!p) if (!p) {
return ICE_ERR_NO_MEMORY; status = ICE_ERR_NO_MEMORY;
goto err_ice_add_prof_to_lst;
}
p->profile_cookie = map->profile_cookie; p->profile_cookie = map->profile_cookie;
p->prof_id = map->prof_id; p->prof_id = map->prof_id;
@ -4282,7 +4273,9 @@ ice_add_prof_to_lst(struct ice_hw *hw, enum ice_block blk,
list_add(&p->list, lst); list_add(&p->list, lst);
return 0; err_ice_add_prof_to_lst:
mutex_unlock(&hw->blk[blk].es.prof_map_lock);
return status;
} }
/** /**
@ -4500,16 +4493,12 @@ ice_add_prof_id_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, u64 hdl,
u8 vl_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF }; u8 vl_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
u8 dc_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0x00, 0x00, 0x00 }; u8 dc_msk[ICE_TCAM_KEY_VAL_SZ] = { 0xFF, 0xFF, 0x00, 0x00, 0x00 };
u8 nm_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x00, 0x00, 0x00, 0x00, 0x00 }; u8 nm_msk[ICE_TCAM_KEY_VAL_SZ] = { 0x00, 0x00, 0x00, 0x00, 0x00 };
enum ice_status status = 0;
struct ice_prof_map *map; struct ice_prof_map *map;
struct ice_vsig_prof *t; struct ice_vsig_prof *t;
struct ice_chs_chg *p; struct ice_chs_chg *p;
u16 vsig_idx, i; u16 vsig_idx, i;
/* Get the details on the profile specified by the handle ID */
map = ice_search_prof_id(hw, blk, hdl);
if (!map)
return ICE_ERR_DOES_NOT_EXIST;
/* Error, if this VSIG already has this profile */ /* Error, if this VSIG already has this profile */
if (ice_has_prof_vsig(hw, blk, vsig, hdl)) if (ice_has_prof_vsig(hw, blk, vsig, hdl))
return ICE_ERR_ALREADY_EXISTS; return ICE_ERR_ALREADY_EXISTS;
@ -4519,19 +4508,28 @@ ice_add_prof_id_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, u64 hdl,
if (!t) if (!t)
return ICE_ERR_NO_MEMORY; return ICE_ERR_NO_MEMORY;
mutex_lock(&hw->blk[blk].es.prof_map_lock);
/* Get the details on the profile specified by the handle ID */
map = ice_search_prof_id(hw, blk, hdl);
if (!map) {
status = ICE_ERR_DOES_NOT_EXIST;
goto err_ice_add_prof_id_vsig;
}
t->profile_cookie = map->profile_cookie; t->profile_cookie = map->profile_cookie;
t->prof_id = map->prof_id; t->prof_id = map->prof_id;
t->tcam_count = map->ptg_cnt; t->tcam_count = map->ptg_cnt;
/* create TCAM entries */ /* create TCAM entries */
for (i = 0; i < map->ptg_cnt; i++) { for (i = 0; i < map->ptg_cnt; i++) {
enum ice_status status;
u16 tcam_idx; u16 tcam_idx;
/* add TCAM to change list */ /* add TCAM to change list */
p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL); p = devm_kzalloc(ice_hw_to_dev(hw), sizeof(*p), GFP_KERNEL);
if (!p) if (!p) {
status = ICE_ERR_NO_MEMORY;
goto err_ice_add_prof_id_vsig; goto err_ice_add_prof_id_vsig;
}
/* allocate the TCAM entry index */ /* allocate the TCAM entry index */
status = ice_alloc_tcam_ent(hw, blk, &tcam_idx); status = ice_alloc_tcam_ent(hw, blk, &tcam_idx);
@ -4575,12 +4573,14 @@ ice_add_prof_id_vsig(struct ice_hw *hw, enum ice_block blk, u16 vsig, u64 hdl,
list_add(&t->list, list_add(&t->list,
&hw->blk[blk].xlt2.vsig_tbl[vsig_idx].prop_lst); &hw->blk[blk].xlt2.vsig_tbl[vsig_idx].prop_lst);
return 0; mutex_unlock(&hw->blk[blk].es.prof_map_lock);
return status;
err_ice_add_prof_id_vsig: err_ice_add_prof_id_vsig:
mutex_unlock(&hw->blk[blk].es.prof_map_lock);
/* let caller clean up the change list */ /* let caller clean up the change list */
devm_kfree(ice_hw_to_dev(hw), t); devm_kfree(ice_hw_to_dev(hw), t);
return ICE_ERR_NO_MEMORY; return status;
} }
/** /**