forked from mirrors/linux
		
	ice: Fix deinitializing VF in error path
If ice_ena_vfs() fails after calling ice_create_vf_entries(), it frees
all VFs without removing them from snapshot PF-VF mailbox list, leading
to list corruption.
Reproducer:
  devlink dev eswitch set $PF1_PCI mode switchdev
  ip l s $PF1 up
  ip l s $PF1 promisc on
  sleep 1
  echo 1 > /sys/class/net/$PF1/device/sriov_numvfs
  sleep 1
  echo 1 > /sys/class/net/$PF1/device/sriov_numvfs
Trace (minimized):
  list_add corruption. next->prev should be prev (ffff8882e241c6f0), but was 0000000000000000. (next=ffff888455da1330).
  kernel BUG at lib/list_debug.c:29!
  RIP: 0010:__list_add_valid_or_report+0xa6/0x100
   ice_mbx_init_vf_info+0xa7/0x180 [ice]
   ice_initialize_vf_entry+0x1fa/0x250 [ice]
   ice_sriov_configure+0x8d7/0x1520 [ice]
   ? __percpu_ref_switch_mode+0x1b1/0x5d0
   ? __pfx_ice_sriov_configure+0x10/0x10 [ice]
Sometimes a KASAN report can be seen instead with a similar stack trace:
  BUG: KASAN: use-after-free in __list_add_valid_or_report+0xf1/0x100
VFs are added to this list in ice_mbx_init_vf_info(), but only removed
in ice_free_vfs(). Move the removing to ice_free_vf_entries(), which is
also being called in other places where VFs are being removed (including
ice_free_vfs() itself).
Fixes: 8cd8a6b17d ("ice: move VF overflow message count into struct ice_mbx_vf_info")
Reported-by: Sujai Buvaneswaran <sujai.buvaneswaran@intel.com>
Closes: https://lore.kernel.org/intel-wired-lan/PH0PR11MB50138B635F2E5CEB7075325D961F2@PH0PR11MB5013.namprd11.prod.outlook.com
Reviewed-by: Martyna Szapar-Mudlaw <martyna.szapar-mudlaw@linux.intel.com>
Signed-off-by: Marcin Szycik <marcin.szycik@linux.intel.com>
Reviewed-by: Simon Horman <horms@kernel.org>
Tested-by: Sujai Buvaneswaran <sujai.buvaneswaran@intel.com>
Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com>
Link: https://patch.msgid.link/20250224190647.3601930-2-anthony.l.nguyen@intel.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
			
			
This commit is contained in:
		
							parent
							
								
									5568e4ca9a
								
							
						
					
					
						commit
						79990cf5e7
					
				
					 3 changed files with 10 additions and 4 deletions
				
			
		|  | @ -36,6 +36,7 @@ static void ice_free_vf_entries(struct ice_pf *pf) | ||||||
| 
 | 
 | ||||||
| 	hash_for_each_safe(vfs->table, bkt, tmp, vf, entry) { | 	hash_for_each_safe(vfs->table, bkt, tmp, vf, entry) { | ||||||
| 		hash_del_rcu(&vf->entry); | 		hash_del_rcu(&vf->entry); | ||||||
|  | 		ice_deinitialize_vf_entry(vf); | ||||||
| 		ice_put_vf(vf); | 		ice_put_vf(vf); | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | @ -193,10 +194,6 @@ void ice_free_vfs(struct ice_pf *pf) | ||||||
| 			wr32(hw, GLGEN_VFLRSTAT(reg_idx), BIT(bit_idx)); | 			wr32(hw, GLGEN_VFLRSTAT(reg_idx), BIT(bit_idx)); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		/* clear malicious info since the VF is getting released */ |  | ||||||
| 		if (!ice_is_feature_supported(pf, ICE_F_MBX_LIMIT)) |  | ||||||
| 			list_del(&vf->mbx_info.list_entry); |  | ||||||
| 
 |  | ||||||
| 		mutex_unlock(&vf->cfg_lock); | 		mutex_unlock(&vf->cfg_lock); | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1036,6 +1036,14 @@ void ice_initialize_vf_entry(struct ice_vf *vf) | ||||||
| 	mutex_init(&vf->cfg_lock); | 	mutex_init(&vf->cfg_lock); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void ice_deinitialize_vf_entry(struct ice_vf *vf) | ||||||
|  | { | ||||||
|  | 	struct ice_pf *pf = vf->pf; | ||||||
|  | 
 | ||||||
|  | 	if (!ice_is_feature_supported(pf, ICE_F_MBX_LIMIT)) | ||||||
|  | 		list_del(&vf->mbx_info.list_entry); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /**
 | /**
 | ||||||
|  * ice_dis_vf_qs - Disable the VF queues |  * ice_dis_vf_qs - Disable the VF queues | ||||||
|  * @vf: pointer to the VF structure |  * @vf: pointer to the VF structure | ||||||
|  |  | ||||||
|  | @ -24,6 +24,7 @@ | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| void ice_initialize_vf_entry(struct ice_vf *vf); | void ice_initialize_vf_entry(struct ice_vf *vf); | ||||||
|  | void ice_deinitialize_vf_entry(struct ice_vf *vf); | ||||||
| void ice_dis_vf_qs(struct ice_vf *vf); | void ice_dis_vf_qs(struct ice_vf *vf); | ||||||
| int ice_check_vf_init(struct ice_vf *vf); | int ice_check_vf_init(struct ice_vf *vf); | ||||||
| enum virtchnl_status_code ice_err_to_virt_err(int err); | enum virtchnl_status_code ice_err_to_virt_err(int err); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Marcin Szycik
						Marcin Szycik