diff options
Diffstat (limited to 'drivers/net/ethernet/intel')
| -rw-r--r-- | drivers/net/ethernet/intel/idpf/idpf_ptp.c | 3 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/idpf/idpf_virtchnl_ptp.c | 1 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbe/ixgbe_main.c | 3 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbe/ixgbe_mbx.h | 15 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbe/ixgbe_sriov.c | 79 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/defines.h | 1 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/ipsec.c | 10 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/ixgbevf.h | 7 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c | 34 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/mbx.h | 8 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/vf.c | 182 | ||||
| -rw-r--r-- | drivers/net/ethernet/intel/ixgbevf/vf.h | 1 | 
12 files changed, 310 insertions, 34 deletions
| diff --git a/drivers/net/ethernet/intel/idpf/idpf_ptp.c b/drivers/net/ethernet/intel/idpf/idpf_ptp.c index 142823af1f9e..3e1052d070cf 100644 --- a/drivers/net/ethernet/intel/idpf/idpf_ptp.c +++ b/drivers/net/ethernet/intel/idpf/idpf_ptp.c @@ -863,6 +863,9 @@ static void idpf_ptp_release_vport_tstamp(struct idpf_vport *vport)  		u64_stats_inc(&vport->tstamp_stats.flushed);  		list_del(&ptp_tx_tstamp->list_member); +		if (ptp_tx_tstamp->skb) +			consume_skb(ptp_tx_tstamp->skb); +  		kfree(ptp_tx_tstamp);  	}  	u64_stats_update_end(&vport->tstamp_stats.stats_sync); diff --git a/drivers/net/ethernet/intel/idpf/idpf_virtchnl_ptp.c b/drivers/net/ethernet/intel/idpf/idpf_virtchnl_ptp.c index 8a2e0f8c5e36..61cedb6f2854 100644 --- a/drivers/net/ethernet/intel/idpf/idpf_virtchnl_ptp.c +++ b/drivers/net/ethernet/intel/idpf/idpf_virtchnl_ptp.c @@ -517,6 +517,7 @@ idpf_ptp_get_tstamp_value(struct idpf_vport *vport,  	shhwtstamps.hwtstamp = ns_to_ktime(tstamp);  	skb_tstamp_tx(ptp_tx_tstamp->skb, &shhwtstamps);  	consume_skb(ptp_tx_tstamp->skb); +	ptp_tx_tstamp->skb = NULL;  	list_add(&ptp_tx_tstamp->list_member,  		 &tx_tstamp_caps->latches_free); diff --git a/drivers/net/ethernet/intel/ixgbe/ixgbe_main.c b/drivers/net/ethernet/intel/ixgbe/ixgbe_main.c index 90d4e57b1c93..ca1ccc630001 100644 --- a/drivers/net/ethernet/intel/ixgbe/ixgbe_main.c +++ b/drivers/net/ethernet/intel/ixgbe/ixgbe_main.c @@ -12101,7 +12101,6 @@ static void ixgbe_remove(struct pci_dev *pdev)  	devl_port_unregister(&adapter->devlink_port);  	devl_unlock(adapter->devlink); -	devlink_free(adapter->devlink);  	ixgbe_stop_ipsec_offload(adapter);  	ixgbe_clear_interrupt_scheme(adapter); @@ -12137,6 +12136,8 @@ static void ixgbe_remove(struct pci_dev *pdev)  	if (disable_dev)  		pci_disable_device(pdev); + +	devlink_free(adapter->devlink);  }  /** diff --git a/drivers/net/ethernet/intel/ixgbe/ixgbe_mbx.h b/drivers/net/ethernet/intel/ixgbe/ixgbe_mbx.h index 4af149b63a39..0334ed4b8fa3 100644 --- a/drivers/net/ethernet/intel/ixgbe/ixgbe_mbx.h +++ b/drivers/net/ethernet/intel/ixgbe/ixgbe_mbx.h @@ -50,6 +50,9 @@ enum ixgbe_pfvf_api_rev {  	ixgbe_mbox_api_12,	/* API version 1.2, linux/freebsd VF driver */  	ixgbe_mbox_api_13,	/* API version 1.3, linux/freebsd VF driver */  	ixgbe_mbox_api_14,	/* API version 1.4, linux/freebsd VF driver */ +	ixgbe_mbox_api_15,	/* API version 1.5, linux/freebsd VF driver */ +	ixgbe_mbox_api_16,	/* API version 1.6, linux/freebsd VF driver */ +	ixgbe_mbox_api_17,	/* API version 1.7, linux/freebsd VF driver */  	/* This value should always be last */  	ixgbe_mbox_api_unknown,	/* indicates that API version is not known */  }; @@ -86,6 +89,12 @@ enum ixgbe_pfvf_api_rev {  #define IXGBE_VF_GET_LINK_STATE 0x10 /* get vf link state */ +/* mailbox API, version 1.6 VF requests */ +#define IXGBE_VF_GET_PF_LINK_STATE	0x11 /* request PF to send link info */ + +/* mailbox API, version 1.7 VF requests */ +#define IXGBE_VF_FEATURES_NEGOTIATE	0x12 /* get features supported by PF */ +  /* length of permanent address message returned from PF */  #define IXGBE_VF_PERMADDR_MSG_LEN 4  /* word in permanent address message with the current multicast type */ @@ -96,6 +105,12 @@ enum ixgbe_pfvf_api_rev {  #define IXGBE_VF_MBX_INIT_TIMEOUT 2000 /* number of retries on mailbox */  #define IXGBE_VF_MBX_INIT_DELAY   500  /* microseconds between retries */ +/* features negotiated between PF/VF */ +#define IXGBEVF_PF_SUP_IPSEC		BIT(0) +#define IXGBEVF_PF_SUP_ESX_MBX		BIT(1) + +#define IXGBE_SUPPORTED_FEATURES	IXGBEVF_PF_SUP_IPSEC +  struct ixgbe_hw;  int ixgbe_read_mbx(struct ixgbe_hw *, u32 *, u16, u16); diff --git a/drivers/net/ethernet/intel/ixgbe/ixgbe_sriov.c b/drivers/net/ethernet/intel/ixgbe/ixgbe_sriov.c index 32ac1e020d91..ee133d6749b3 100644 --- a/drivers/net/ethernet/intel/ixgbe/ixgbe_sriov.c +++ b/drivers/net/ethernet/intel/ixgbe/ixgbe_sriov.c @@ -510,6 +510,8 @@ static int ixgbe_set_vf_lpe(struct ixgbe_adapter *adapter, u32 max_frame, u32 vf  		case ixgbe_mbox_api_12:  		case ixgbe_mbox_api_13:  		case ixgbe_mbox_api_14: +		case ixgbe_mbox_api_16: +		case ixgbe_mbox_api_17:  			/* Version 1.1 supports jumbo frames on VFs if PF has  			 * jumbo frames enabled which means legacy VFs are  			 * disabled @@ -1046,6 +1048,8 @@ static int ixgbe_negotiate_vf_api(struct ixgbe_adapter *adapter,  	case ixgbe_mbox_api_12:  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		adapter->vfinfo[vf].vf_api = api;  		return 0;  	default: @@ -1072,6 +1076,8 @@ static int ixgbe_get_vf_queues(struct ixgbe_adapter *adapter,  	case ixgbe_mbox_api_12:  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		break;  	default:  		return -1; @@ -1112,6 +1118,8 @@ static int ixgbe_get_vf_reta(struct ixgbe_adapter *adapter, u32 *msgbuf, u32 vf)  	/* verify the PF is supporting the correct API */  	switch (adapter->vfinfo[vf].vf_api) { +	case ixgbe_mbox_api_17: +	case ixgbe_mbox_api_16:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_12: @@ -1145,6 +1153,8 @@ static int ixgbe_get_vf_rss_key(struct ixgbe_adapter *adapter,  	/* verify the PF is supporting the correct API */  	switch (adapter->vfinfo[vf].vf_api) { +	case ixgbe_mbox_api_17: +	case ixgbe_mbox_api_16:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_12: @@ -1174,6 +1184,8 @@ static int ixgbe_update_vf_xcast_mode(struct ixgbe_adapter *adapter,  		fallthrough;  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		break;  	default:  		return -EOPNOTSUPP; @@ -1244,6 +1256,8 @@ static int ixgbe_get_vf_link_state(struct ixgbe_adapter *adapter,  	case ixgbe_mbox_api_12:  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		break;  	default:  		return -EOPNOTSUPP; @@ -1254,6 +1268,65 @@ static int ixgbe_get_vf_link_state(struct ixgbe_adapter *adapter,  	return 0;  } +/** + * ixgbe_send_vf_link_status - send link status data to VF + * @adapter: pointer to adapter struct + * @msgbuf: pointer to message buffers + * @vf: VF identifier + * + * Reply for IXGBE_VF_GET_PF_LINK_STATE mbox command sending link status data. + * + * Return: 0 on success or -EOPNOTSUPP when operation is not supported. + */ +static int ixgbe_send_vf_link_status(struct ixgbe_adapter *adapter, +				     u32 *msgbuf, u32 vf) +{ +	struct ixgbe_hw *hw = &adapter->hw; + +	switch (adapter->vfinfo[vf].vf_api) { +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17: +		if (hw->mac.type != ixgbe_mac_e610) +			return -EOPNOTSUPP; +		break; +	default: +		return -EOPNOTSUPP; +	} +	/* Simply provide stored values as watchdog & link status events take +	 * care of its freshness. +	 */ +	msgbuf[1] = adapter->link_speed; +	msgbuf[2] = adapter->link_up; + +	return 0; +} + +/** + * ixgbe_negotiate_vf_features -  negotiate supported features with VF driver + * @adapter: pointer to adapter struct + * @msgbuf: pointer to message buffers + * @vf: VF identifier + * + * Return: 0 on success or -EOPNOTSUPP when operation is not supported. + */ +static int ixgbe_negotiate_vf_features(struct ixgbe_adapter *adapter, +				       u32 *msgbuf, u32 vf) +{ +	u32 features = msgbuf[1]; + +	switch (adapter->vfinfo[vf].vf_api) { +	case ixgbe_mbox_api_17: +		break; +	default: +		return -EOPNOTSUPP; +	} + +	features &= IXGBE_SUPPORTED_FEATURES; +	msgbuf[1] = features; + +	return 0; +} +  static int ixgbe_rcv_msg_from_vf(struct ixgbe_adapter *adapter, u32 vf)  {  	u32 mbx_size = IXGBE_VFMAILBOX_SIZE; @@ -1328,6 +1401,12 @@ static int ixgbe_rcv_msg_from_vf(struct ixgbe_adapter *adapter, u32 vf)  	case IXGBE_VF_IPSEC_DEL:  		retval = ixgbe_ipsec_vf_del_sa(adapter, msgbuf, vf);  		break; +	case IXGBE_VF_GET_PF_LINK_STATE: +		retval = ixgbe_send_vf_link_status(adapter, msgbuf, vf); +		break; +	case IXGBE_VF_FEATURES_NEGOTIATE: +		retval = ixgbe_negotiate_vf_features(adapter, msgbuf, vf); +		break;  	default:  		e_err(drv, "Unhandled Msg %8.8x\n", msgbuf[0]);  		retval = -EIO; diff --git a/drivers/net/ethernet/intel/ixgbevf/defines.h b/drivers/net/ethernet/intel/ixgbevf/defines.h index a9bc96f6399d..e177d1d58696 100644 --- a/drivers/net/ethernet/intel/ixgbevf/defines.h +++ b/drivers/net/ethernet/intel/ixgbevf/defines.h @@ -28,6 +28,7 @@  /* Link speed */  typedef u32 ixgbe_link_speed; +#define IXGBE_LINK_SPEED_UNKNOWN	0  #define IXGBE_LINK_SPEED_1GB_FULL	0x0020  #define IXGBE_LINK_SPEED_10GB_FULL	0x0080  #define IXGBE_LINK_SPEED_100_FULL	0x0008 diff --git a/drivers/net/ethernet/intel/ixgbevf/ipsec.c b/drivers/net/ethernet/intel/ixgbevf/ipsec.c index 65580b9cb06f..fce35924ff8b 100644 --- a/drivers/net/ethernet/intel/ixgbevf/ipsec.c +++ b/drivers/net/ethernet/intel/ixgbevf/ipsec.c @@ -273,6 +273,9 @@ static int ixgbevf_ipsec_add_sa(struct net_device *dev,  	adapter = netdev_priv(dev);  	ipsec = adapter->ipsec; +	if (!(adapter->pf_features & IXGBEVF_PF_SUP_IPSEC)) +		return -EOPNOTSUPP; +  	if (xs->id.proto != IPPROTO_ESP && xs->id.proto != IPPROTO_AH) {  		NL_SET_ERR_MSG_MOD(extack, "Unsupported protocol for IPsec offload");  		return -EINVAL; @@ -405,6 +408,9 @@ static void ixgbevf_ipsec_del_sa(struct net_device *dev,  	adapter = netdev_priv(dev);  	ipsec = adapter->ipsec; +	if (!(adapter->pf_features & IXGBEVF_PF_SUP_IPSEC)) +		return; +  	if (xs->xso.dir == XFRM_DEV_OFFLOAD_IN) {  		sa_idx = xs->xso.offload_handle - IXGBE_IPSEC_BASE_RX_INDEX; @@ -612,6 +618,10 @@ void ixgbevf_init_ipsec_offload(struct ixgbevf_adapter *adapter)  	size_t size;  	switch (adapter->hw.api_version) { +	case ixgbe_mbox_api_17: +		if (!(adapter->pf_features & IXGBEVF_PF_SUP_IPSEC)) +			return; +		break;  	case ixgbe_mbox_api_14:  		break;  	default: diff --git a/drivers/net/ethernet/intel/ixgbevf/ixgbevf.h b/drivers/net/ethernet/intel/ixgbevf/ixgbevf.h index 3a379e6a3a2a..039187607e98 100644 --- a/drivers/net/ethernet/intel/ixgbevf/ixgbevf.h +++ b/drivers/net/ethernet/intel/ixgbevf/ixgbevf.h @@ -363,6 +363,13 @@ struct ixgbevf_adapter {  	struct ixgbe_hw hw;  	u16 msg_enable; +	u32 pf_features; +#define IXGBEVF_PF_SUP_IPSEC		BIT(0) +#define IXGBEVF_PF_SUP_ESX_MBX		BIT(1) + +#define IXGBEVF_SUPPORTED_FEATURES	(IXGBEVF_PF_SUP_IPSEC | \ +					IXGBEVF_PF_SUP_ESX_MBX) +  	struct ixgbevf_hw_stats stats;  	unsigned long state; diff --git a/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c b/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c index 28e25641b167..d5ce20f47def 100644 --- a/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c +++ b/drivers/net/ethernet/intel/ixgbevf/ixgbevf_main.c @@ -2271,10 +2271,36 @@ static void ixgbevf_init_last_counter_stats(struct ixgbevf_adapter *adapter)  	adapter->stats.base_vfmprc = adapter->stats.last_vfmprc;  } +/** + * ixgbevf_set_features - Set features supported by PF + * @adapter: pointer to the adapter struct + * + * Negotiate with PF supported features and then set pf_features accordingly. + */ +static void ixgbevf_set_features(struct ixgbevf_adapter *adapter) +{ +	u32 *pf_features = &adapter->pf_features; +	struct ixgbe_hw *hw = &adapter->hw; +	int err; + +	err = hw->mac.ops.negotiate_features(hw, pf_features); +	if (err && err != -EOPNOTSUPP) +		netdev_dbg(adapter->netdev, +			   "PF feature negotiation failed.\n"); + +	/* Address also pre API 1.7 cases */ +	if (hw->api_version == ixgbe_mbox_api_14) +		*pf_features |= IXGBEVF_PF_SUP_IPSEC; +	else if (hw->api_version == ixgbe_mbox_api_15) +		*pf_features |= IXGBEVF_PF_SUP_ESX_MBX; +} +  static void ixgbevf_negotiate_api(struct ixgbevf_adapter *adapter)  {  	struct ixgbe_hw *hw = &adapter->hw;  	static const int api[] = { +		ixgbe_mbox_api_17, +		ixgbe_mbox_api_16,  		ixgbe_mbox_api_15,  		ixgbe_mbox_api_14,  		ixgbe_mbox_api_13, @@ -2294,7 +2320,9 @@ static void ixgbevf_negotiate_api(struct ixgbevf_adapter *adapter)  		idx++;  	} -	if (hw->api_version >= ixgbe_mbox_api_15) { +	ixgbevf_set_features(adapter); + +	if (adapter->pf_features & IXGBEVF_PF_SUP_ESX_MBX) {  		hw->mbx.ops.init_params(hw);  		memcpy(&hw->mbx.ops, &ixgbevf_mbx_ops,  		       sizeof(struct ixgbe_mbx_operations)); @@ -2651,6 +2679,8 @@ static void ixgbevf_set_num_queues(struct ixgbevf_adapter *adapter)  		case ixgbe_mbox_api_13:  		case ixgbe_mbox_api_14:  		case ixgbe_mbox_api_15: +		case ixgbe_mbox_api_16: +		case ixgbe_mbox_api_17:  			if (adapter->xdp_prog &&  			    hw->mac.max_tx_queues == rss)  				rss = rss > 3 ? 2 : 1; @@ -4645,6 +4675,8 @@ static int ixgbevf_probe(struct pci_dev *pdev, const struct pci_device_id *ent)  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_15: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		netdev->max_mtu = IXGBE_MAX_JUMBO_FRAME_SIZE -  				  (ETH_HLEN + ETH_FCS_LEN);  		break; diff --git a/drivers/net/ethernet/intel/ixgbevf/mbx.h b/drivers/net/ethernet/intel/ixgbevf/mbx.h index 835bbcc5cc8e..a8ed23ee66aa 100644 --- a/drivers/net/ethernet/intel/ixgbevf/mbx.h +++ b/drivers/net/ethernet/intel/ixgbevf/mbx.h @@ -66,6 +66,8 @@ enum ixgbe_pfvf_api_rev {  	ixgbe_mbox_api_13,	/* API version 1.3, linux/freebsd VF driver */  	ixgbe_mbox_api_14,	/* API version 1.4, linux/freebsd VF driver */  	ixgbe_mbox_api_15,	/* API version 1.5, linux/freebsd VF driver */ +	ixgbe_mbox_api_16,      /* API version 1.6, linux/freebsd VF driver */ +	ixgbe_mbox_api_17,	/* API version 1.7, linux/freebsd VF driver */  	/* This value should always be last */  	ixgbe_mbox_api_unknown,	/* indicates that API version is not known */  }; @@ -102,6 +104,12 @@ enum ixgbe_pfvf_api_rev {  #define IXGBE_VF_GET_LINK_STATE 0x10 /* get vf link state */ +/* mailbox API, version 1.6 VF requests */ +#define IXGBE_VF_GET_PF_LINK_STATE	0x11 /* request PF to send link info */ + +/* mailbox API, version 1.7 VF requests */ +#define IXGBE_VF_FEATURES_NEGOTIATE	0x12 /* get features supported by PF*/ +  /* length of permanent address message returned from PF */  #define IXGBE_VF_PERMADDR_MSG_LEN	4  /* word in permanent address message with the current multicast type */ diff --git a/drivers/net/ethernet/intel/ixgbevf/vf.c b/drivers/net/ethernet/intel/ixgbevf/vf.c index dcaef34b88b6..74d320879513 100644 --- a/drivers/net/ethernet/intel/ixgbevf/vf.c +++ b/drivers/net/ethernet/intel/ixgbevf/vf.c @@ -313,6 +313,8 @@ int ixgbevf_get_reta_locked(struct ixgbe_hw *hw, u32 *reta, int num_rx_queues)  	 * is not supported for this device type.  	 */  	switch (hw->api_version) { +	case ixgbe_mbox_api_17: +	case ixgbe_mbox_api_16:  	case ixgbe_mbox_api_15:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_13: @@ -382,6 +384,8 @@ int ixgbevf_get_rss_key_locked(struct ixgbe_hw *hw, u8 *rss_key)  	 * or if the operation is not supported for this device type.  	 */  	switch (hw->api_version) { +	case ixgbe_mbox_api_17: +	case ixgbe_mbox_api_16:  	case ixgbe_mbox_api_15:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_13: @@ -552,6 +556,8 @@ static s32 ixgbevf_update_xcast_mode(struct ixgbe_hw *hw, int xcast_mode)  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_15: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		break;  	default:  		return -EOPNOTSUPP; @@ -625,6 +631,85 @@ static s32 ixgbevf_hv_get_link_state_vf(struct ixgbe_hw *hw, bool *link_state)  }  /** + * ixgbevf_get_pf_link_state - Get PF's link status + * @hw: pointer to the HW structure + * @speed: link speed + * @link_up: indicate if link is up/down + * + * Ask PF to provide link_up state and speed of the link. + * + * Return: IXGBE_ERR_MBX in the case of mailbox error, + * -EOPNOTSUPP if the op is not supported or 0 on success. + */ +static int ixgbevf_get_pf_link_state(struct ixgbe_hw *hw, ixgbe_link_speed *speed, +				     bool *link_up) +{ +	u32 msgbuf[3] = {}; +	int err; + +	switch (hw->api_version) { +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17: +		break; +	default: +		return -EOPNOTSUPP; +	} + +	msgbuf[0] = IXGBE_VF_GET_PF_LINK_STATE; + +	err = ixgbevf_write_msg_read_ack(hw, msgbuf, msgbuf, +					 ARRAY_SIZE(msgbuf)); +	if (err || (msgbuf[0] & IXGBE_VT_MSGTYPE_FAILURE)) { +		err = IXGBE_ERR_MBX; +		*speed = IXGBE_LINK_SPEED_UNKNOWN; +		/* No need to set @link_up to false as it will be done by +		 * ixgbe_check_mac_link_vf(). +		 */ +	} else { +		*speed = msgbuf[1]; +		*link_up = msgbuf[2]; +	} + +	return err; +} + +/** + * ixgbevf_negotiate_features_vf - negotiate supported features with PF driver + * @hw: pointer to the HW structure + * @pf_features: bitmask of features supported by PF + * + * Return: IXGBE_ERR_MBX in the  case of mailbox error, + * -EOPNOTSUPP if the op is not supported or 0 on success. + */ +static int ixgbevf_negotiate_features_vf(struct ixgbe_hw *hw, u32 *pf_features) +{ +	u32 msgbuf[2] = {}; +	int err; + +	switch (hw->api_version) { +	case ixgbe_mbox_api_17: +		break; +	default: +		return -EOPNOTSUPP; +	} + +	msgbuf[0] = IXGBE_VF_FEATURES_NEGOTIATE; +	msgbuf[1] = IXGBEVF_SUPPORTED_FEATURES; + +	err = ixgbevf_write_msg_read_ack(hw, msgbuf, msgbuf, +					 ARRAY_SIZE(msgbuf)); + +	if (err || (msgbuf[0] & IXGBE_VT_MSGTYPE_FAILURE)) { +		err = IXGBE_ERR_MBX; +		*pf_features = 0x0; +	} else { +		*pf_features = msgbuf[1]; +	} + +	return err; +} + +/**   *  ixgbevf_set_vfta_vf - Set/Unset VLAN filter table address   *  @hw: pointer to the HW structure   *  @vlan: 12 bit VLAN ID @@ -659,6 +744,58 @@ mbx_err:  }  /** + * ixgbe_read_vflinks - Read VFLINKS register + * @hw: pointer to the HW structure + * @speed: link speed + * @link_up: indicate if link is up/down + * + * Get linkup status and link speed from the VFLINKS register. + */ +static void ixgbe_read_vflinks(struct ixgbe_hw *hw, ixgbe_link_speed *speed, +			       bool *link_up) +{ +	u32 vflinks = IXGBE_READ_REG(hw, IXGBE_VFLINKS); + +	/* if link status is down no point in checking to see if PF is up */ +	if (!(vflinks & IXGBE_LINKS_UP)) { +		*link_up = false; +		return; +	} + +	/* for SFP+ modules and DA cables on 82599 it can take up to 500usecs +	 * before the link status is correct +	 */ +	if (hw->mac.type == ixgbe_mac_82599_vf) { +		for (int i = 0; i < 5; i++) { +			udelay(100); +			vflinks = IXGBE_READ_REG(hw, IXGBE_VFLINKS); + +			if (!(vflinks & IXGBE_LINKS_UP)) { +				*link_up = false; +				return; +			} +		} +	} + +	/* We reached this point so there's link */ +	*link_up = true; + +	switch (vflinks & IXGBE_LINKS_SPEED_82599) { +	case IXGBE_LINKS_SPEED_10G_82599: +		*speed = IXGBE_LINK_SPEED_10GB_FULL; +		break; +	case IXGBE_LINKS_SPEED_1G_82599: +		*speed = IXGBE_LINK_SPEED_1GB_FULL; +		break; +	case IXGBE_LINKS_SPEED_100_82599: +		*speed = IXGBE_LINK_SPEED_100_FULL; +		break; +	default: +		*speed = IXGBE_LINK_SPEED_UNKNOWN; +	} +} + +/**   * ixgbevf_hv_set_vfta_vf - * Hyper-V variant - just a stub.   * @hw: unused   * @vlan: unused @@ -702,10 +839,10 @@ static s32 ixgbevf_check_mac_link_vf(struct ixgbe_hw *hw,  				     bool *link_up,  				     bool autoneg_wait_to_complete)  { +	struct ixgbevf_adapter *adapter = hw->back;  	struct ixgbe_mbx_info *mbx = &hw->mbx;  	struct ixgbe_mac_info *mac = &hw->mac;  	s32 ret_val = 0; -	u32 links_reg;  	u32 in_msg = 0;  	/* If we were hit with a reset drop the link */ @@ -715,43 +852,21 @@ static s32 ixgbevf_check_mac_link_vf(struct ixgbe_hw *hw,  	if (!mac->get_link_status)  		goto out; -	/* if link status is down no point in checking to see if pf is up */ -	links_reg = IXGBE_READ_REG(hw, IXGBE_VFLINKS); -	if (!(links_reg & IXGBE_LINKS_UP)) -		goto out; - -	/* for SFP+ modules and DA cables on 82599 it can take up to 500usecs -	 * before the link status is correct -	 */ -	if (mac->type == ixgbe_mac_82599_vf) { -		int i; - -		for (i = 0; i < 5; i++) { -			udelay(100); -			links_reg = IXGBE_READ_REG(hw, IXGBE_VFLINKS); - -			if (!(links_reg & IXGBE_LINKS_UP)) -				goto out; -		} -	} - -	switch (links_reg & IXGBE_LINKS_SPEED_82599) { -	case IXGBE_LINKS_SPEED_10G_82599: -		*speed = IXGBE_LINK_SPEED_10GB_FULL; -		break; -	case IXGBE_LINKS_SPEED_1G_82599: -		*speed = IXGBE_LINK_SPEED_1GB_FULL; -		break; -	case IXGBE_LINKS_SPEED_100_82599: -		*speed = IXGBE_LINK_SPEED_100_FULL; -		break; +	if (hw->mac.type == ixgbe_mac_e610_vf) { +		ret_val = ixgbevf_get_pf_link_state(hw, speed, link_up); +		if (ret_val) +			goto out; +	} else { +		ixgbe_read_vflinks(hw, speed, link_up); +		if (*link_up == false) +			goto out;  	}  	/* if the read failed it could just be a mailbox collision, best wait  	 * until we are called again and don't report an error  	 */  	if (mbx->ops.read(hw, &in_msg, 1)) { -		if (hw->api_version >= ixgbe_mbox_api_15) +		if (adapter->pf_features & IXGBEVF_PF_SUP_ESX_MBX)  			mac->get_link_status = false;  		goto out;  	} @@ -951,6 +1066,8 @@ int ixgbevf_get_queues(struct ixgbe_hw *hw, unsigned int *num_tcs,  	case ixgbe_mbox_api_13:  	case ixgbe_mbox_api_14:  	case ixgbe_mbox_api_15: +	case ixgbe_mbox_api_16: +	case ixgbe_mbox_api_17:  		break;  	default:  		return 0; @@ -1005,6 +1122,7 @@ static const struct ixgbe_mac_operations ixgbevf_mac_ops = {  	.setup_link		= ixgbevf_setup_mac_link_vf,  	.check_link		= ixgbevf_check_mac_link_vf,  	.negotiate_api_version	= ixgbevf_negotiate_api_version_vf, +	.negotiate_features	= ixgbevf_negotiate_features_vf,  	.set_rar		= ixgbevf_set_rar_vf,  	.update_mc_addr_list	= ixgbevf_update_mc_addr_list_vf,  	.update_xcast_mode	= ixgbevf_update_xcast_mode, diff --git a/drivers/net/ethernet/intel/ixgbevf/vf.h b/drivers/net/ethernet/intel/ixgbevf/vf.h index 2d791bc26ae4..4f19b8900c29 100644 --- a/drivers/net/ethernet/intel/ixgbevf/vf.h +++ b/drivers/net/ethernet/intel/ixgbevf/vf.h @@ -26,6 +26,7 @@ struct ixgbe_mac_operations {  	s32 (*stop_adapter)(struct ixgbe_hw *);  	s32 (*get_bus_info)(struct ixgbe_hw *);  	s32 (*negotiate_api_version)(struct ixgbe_hw *hw, int api); +	int (*negotiate_features)(struct ixgbe_hw *hw, u32 *pf_features);  	/* Link */  	s32 (*setup_link)(struct ixgbe_hw *, ixgbe_link_speed, bool, bool); | 
