diff options
Diffstat (limited to 'drivers/net/ethernet/intel/i40e/i40e_virtchnl_pf.c')
| -rw-r--r-- | drivers/net/ethernet/intel/i40e/i40e_virtchnl_pf.c | 93 | 
1 files changed, 67 insertions, 26 deletions
| diff --git a/drivers/net/ethernet/intel/i40e/i40e_virtchnl_pf.c b/drivers/net/ethernet/intel/i40e/i40e_virtchnl_pf.c index 3d2440838822..6a3f0fc56c3b 100644 --- a/drivers/net/ethernet/intel/i40e/i40e_virtchnl_pf.c +++ b/drivers/net/ethernet/intel/i40e/i40e_virtchnl_pf.c @@ -955,7 +955,6 @@ static void i40e_free_vf_res(struct i40e_vf *vf)  		i40e_vsi_release(pf->vsi[vf->lan_vsi_idx]);  		vf->lan_vsi_idx = 0;  		vf->lan_vsi_id = 0; -		vf->num_mac = 0;  	}  	/* do the accounting and remove additional ADq VSI's */ @@ -2548,20 +2547,12 @@ static inline int i40e_check_vf_permission(struct i40e_vf *vf,  					   struct virtchnl_ether_addr_list *al)  {  	struct i40e_pf *pf = vf->pf; +	struct i40e_vsi *vsi = pf->vsi[vf->lan_vsi_idx]; +	int mac2add_cnt = 0;  	int i; -	/* If this VF is not privileged, then we can't add more than a limited -	 * number of addresses. Check to make sure that the additions do not -	 * push us over the limit. -	 */ -	if (!test_bit(I40E_VIRTCHNL_VF_CAP_PRIVILEGE, &vf->vf_caps) && -	    (vf->num_mac + al->num_elements) > I40E_VC_MAX_MAC_ADDR_PER_VF) { -		dev_err(&pf->pdev->dev, -			"Cannot add more MAC addresses, VF is not trusted, switch the VF to trusted to add more functionality\n"); -		return -EPERM; -	} -  	for (i = 0; i < al->num_elements; i++) { +		struct i40e_mac_filter *f;  		u8 *addr = al->list[i].addr;  		if (is_broadcast_ether_addr(addr) || @@ -2585,8 +2576,24 @@ static inline int i40e_check_vf_permission(struct i40e_vf *vf,  				"VF attempting to override administratively set MAC address, bring down and up the VF interface to resume normal operation\n");  			return -EPERM;  		} + +		/*count filters that really will be added*/ +		f = i40e_find_mac(vsi, addr); +		if (!f) +			++mac2add_cnt;  	} +	/* If this VF is not privileged, then we can't add more than a limited +	 * number of addresses. Check to make sure that the additions do not +	 * push us over the limit. +	 */ +	if (!test_bit(I40E_VIRTCHNL_VF_CAP_PRIVILEGE, &vf->vf_caps) && +	    (i40e_count_filters(vsi) + mac2add_cnt) > +		    I40E_VC_MAX_MAC_ADDR_PER_VF) { +		dev_err(&pf->pdev->dev, +			"Cannot add more MAC addresses, VF is not trusted, switch the VF to trusted to add more functionality\n"); +		return -EPERM; +	}  	return 0;  } @@ -2640,8 +2647,6 @@ static int i40e_vc_add_mac_addr_msg(struct i40e_vf *vf, u8 *msg)  				ret = I40E_ERR_PARAM;  				spin_unlock_bh(&vsi->mac_filter_hash_lock);  				goto error_param; -			} else { -				vf->num_mac++;  			}  		}  	} @@ -2689,16 +2694,6 @@ static int i40e_vc_del_mac_addr_msg(struct i40e_vf *vf, u8 *msg)  			ret = I40E_ERR_INVALID_MAC_ADDR;  			goto error_param;  		} - -		if (vf->pf_set_mac && -		    ether_addr_equal(al->list[i].addr, -				     vf->default_lan_addr.addr)) { -			dev_err(&pf->pdev->dev, -				"MAC addr %pM has been set by PF, cannot delete it for VF %d, reset VF to change MAC addr\n", -				vf->default_lan_addr.addr, vf->vf_id); -			ret = I40E_ERR_PARAM; -			goto error_param; -		}  	}  	vsi = pf->vsi[vf->lan_vsi_idx]; @@ -2709,8 +2704,6 @@ static int i40e_vc_del_mac_addr_msg(struct i40e_vf *vf, u8 *msg)  			ret = I40E_ERR_INVALID_MAC_ADDR;  			spin_unlock_bh(&vsi->mac_filter_hash_lock);  			goto error_param; -		} else { -			vf->num_mac--;  		}  	spin_unlock_bh(&vsi->mac_filter_hash_lock); @@ -4531,3 +4524,51 @@ out:  	clear_bit(__I40E_VIRTCHNL_OP_PENDING, pf->state);  	return ret;  } + +/** + * i40e_get_vf_stats - populate some stats for the VF + * @netdev: the netdev of the PF + * @vf_id: the host OS identifier (0-127) + * @vf_stats: pointer to the OS memory to be initialized + */ +int i40e_get_vf_stats(struct net_device *netdev, int vf_id, +		      struct ifla_vf_stats *vf_stats) +{ +	struct i40e_netdev_priv *np = netdev_priv(netdev); +	struct i40e_pf *pf = np->vsi->back; +	struct i40e_eth_stats *stats; +	struct i40e_vsi *vsi; +	struct i40e_vf *vf; + +	/* validate the request */ +	if (i40e_validate_vf(pf, vf_id)) +		return -EINVAL; + +	vf = &pf->vf[vf_id]; +	if (!test_bit(I40E_VF_STATE_INIT, &vf->vf_states)) { +		dev_err(&pf->pdev->dev, "VF %d in reset. Try again.\n", vf_id); +		return -EBUSY; +	} + +	vsi = pf->vsi[vf->lan_vsi_idx]; +	if (!vsi) +		return -EINVAL; + +	i40e_update_eth_stats(vsi); +	stats = &vsi->eth_stats; + +	memset(vf_stats, 0, sizeof(*vf_stats)); + +	vf_stats->rx_packets = stats->rx_unicast + stats->rx_broadcast + +		stats->rx_multicast; +	vf_stats->tx_packets = stats->tx_unicast + stats->tx_broadcast + +		stats->tx_multicast; +	vf_stats->rx_bytes   = stats->rx_bytes; +	vf_stats->tx_bytes   = stats->tx_bytes; +	vf_stats->broadcast  = stats->rx_broadcast; +	vf_stats->multicast  = stats->rx_multicast; +	vf_stats->rx_dropped = stats->rx_discards; +	vf_stats->tx_dropped = stats->tx_discards; + +	return 0; +} | 

