mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2025-01-11 00:04:33 +08:00
iavf: fix potential races for FDIR filters
commit0fb1d8eb23
upstream. Add fdir_fltr_lock locking in unprotected places. The change in iavf_fdir_is_dup_fltr adds a spinlock around a loop which iterates over all filters and looks for a duplicate. The filter can be removed from list and freed from memory at the same time it's being compared. All other places where filters are deleted are already protected with spinlock. The remaining changes protect adapter->fdir_active_fltr variable so now all its uses are under a spinlock. Fixes:527691bf06
("iavf: Support IPv4 Flow Director filters") Signed-off-by: Piotr Gardocki <piotrx.gardocki@intel.com> Tested-by: Rafal Romanowski <rafal.romanowski@intel.com> Signed-off-by: Tony Nguyen <anthony.l.nguyen@intel.com> Reviewed-by: Simon Horman <horms@kernel.org> Link: https://lore.kernel.org/r/20230807205011.3129224-1-anthony.l.nguyen@intel.com Signed-off-by: Jakub Kicinski <kuba@kernel.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
b1f985cf1c
commit
df21468bfd
@ -1387,14 +1387,15 @@ static int iavf_add_fdir_ethtool(struct iavf_adapter *adapter, struct ethtool_rx
|
||||
if (fsp->flow_type & FLOW_MAC_EXT)
|
||||
return -EINVAL;
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
if (adapter->fdir_active_fltr >= IAVF_MAX_FDIR_FILTERS) {
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
dev_err(&adapter->pdev->dev,
|
||||
"Unable to add Flow Director filter because VF reached the limit of max allowed filters (%u)\n",
|
||||
IAVF_MAX_FDIR_FILTERS);
|
||||
return -ENOSPC;
|
||||
}
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
if (iavf_find_fdir_fltr_by_loc(adapter, fsp->location)) {
|
||||
dev_err(&adapter->pdev->dev, "Failed to add Flow Director filter, it already exists\n");
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
@ -1767,7 +1768,9 @@ static int iavf_get_rxnfc(struct net_device *netdev, struct ethtool_rxnfc *cmd,
|
||||
case ETHTOOL_GRXCLSRLCNT:
|
||||
if (!FDIR_FLTR_SUPPORT(adapter))
|
||||
break;
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
cmd->rule_cnt = adapter->fdir_active_fltr;
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
cmd->data = IAVF_MAX_FDIR_FILTERS;
|
||||
ret = 0;
|
||||
break;
|
||||
|
@ -722,7 +722,9 @@ void iavf_print_fdir_fltr(struct iavf_adapter *adapter, struct iavf_fdir_fltr *f
|
||||
bool iavf_fdir_is_dup_fltr(struct iavf_adapter *adapter, struct iavf_fdir_fltr *fltr)
|
||||
{
|
||||
struct iavf_fdir_fltr *tmp;
|
||||
bool ret = false;
|
||||
|
||||
spin_lock_bh(&adapter->fdir_fltr_lock);
|
||||
list_for_each_entry(tmp, &adapter->fdir_list_head, list) {
|
||||
if (tmp->flow_type != fltr->flow_type)
|
||||
continue;
|
||||
@ -732,11 +734,14 @@ bool iavf_fdir_is_dup_fltr(struct iavf_adapter *adapter, struct iavf_fdir_fltr *
|
||||
!memcmp(&tmp->ip_data, &fltr->ip_data,
|
||||
sizeof(fltr->ip_data)) &&
|
||||
!memcmp(&tmp->ext_data, &fltr->ext_data,
|
||||
sizeof(fltr->ext_data)))
|
||||
return true;
|
||||
sizeof(fltr->ext_data))) {
|
||||
ret = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
spin_unlock_bh(&adapter->fdir_fltr_lock);
|
||||
|
||||
return false;
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
|
Loading…
Reference in New Issue
Block a user