diff options
Diffstat (limited to 'drivers/net/ethernet/intel/igc/igc_main.c')
| -rw-r--r-- | drivers/net/ethernet/intel/igc/igc_main.c | 42 | 
1 files changed, 35 insertions, 7 deletions
| diff --git a/drivers/net/ethernet/intel/igc/igc_main.c b/drivers/net/ethernet/intel/igc/igc_main.c index c6968fdb6caa..9112dff075cf 100644 --- a/drivers/net/ethernet/intel/igc/igc_main.c +++ b/drivers/net/ethernet/intel/igc/igc_main.c @@ -47,6 +47,11 @@ static const struct pci_device_id igc_pci_tbl[] = {  	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I225_K2), board_base },  	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I225_LMVP), board_base },  	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I225_IT), board_base }, +	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I226_LM), board_base }, +	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I226_V), board_base }, +	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I226_IT), board_base }, +	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I221_V), board_base }, +	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I226_BLANK_NVM), board_base },  	{ PCI_VDEVICE(INTEL, IGC_DEV_ID_I225_BLANK_NVM), board_base },  	/* required last entry */  	{0, } @@ -1428,7 +1433,7 @@ static void igc_rx_checksum(struct igc_ring *ring,  	/* TCP/UDP checksum error bit is set */  	if (igc_test_staterr(rx_desc, -			     IGC_RXDEXT_STATERR_TCPE | +			     IGC_RXDEXT_STATERR_L4E |  			     IGC_RXDEXT_STATERR_IPE)) {  		/* work around errata with sctp packets where the TCPE aka  		 * L4E bit is set incorrectly on 64 byte (60 byte w/o crc) @@ -1737,8 +1742,7 @@ static bool igc_cleanup_headers(struct igc_ring *rx_ring,  				union igc_adv_rx_desc *rx_desc,  				struct sk_buff *skb)  { -	if (unlikely((igc_test_staterr(rx_desc, -				       IGC_RXDEXT_ERR_FRAME_ERR_MASK)))) { +	if (unlikely(igc_test_staterr(rx_desc, IGC_RXDEXT_STATERR_RXE))) {  		struct net_device *netdev = rx_ring->netdev;  		if (!(netdev->features & NETIF_F_RXALL)) { @@ -3679,6 +3683,8 @@ void igc_update_stats(struct igc_adapter *adapter)  	adapter->stats.prc511 += rd32(IGC_PRC511);  	adapter->stats.prc1023 += rd32(IGC_PRC1023);  	adapter->stats.prc1522 += rd32(IGC_PRC1522); +	adapter->stats.tlpic += rd32(IGC_TLPIC); +	adapter->stats.rlpic += rd32(IGC_RLPIC);  	mpc = rd32(IGC_MPC);  	adapter->stats.mpc += mpc; @@ -3772,6 +3778,8 @@ void igc_down(struct igc_adapter *adapter)  	set_bit(__IGC_DOWN, &adapter->state); +	igc_ptp_suspend(adapter); +  	/* disable receives in the hardware */  	rctl = rd32(IGC_RCTL);  	wr32(IGC_RCTL, rctl & ~IGC_RCTL_EN); @@ -3825,7 +3833,6 @@ void igc_down(struct igc_adapter *adapter)  void igc_reinit_locked(struct igc_adapter *adapter)  { -	WARN_ON(in_interrupt());  	while (test_and_set_bit(__IGC_RESETTING, &adapter->state))  		usleep_range(1000, 2000);  	igc_down(adapter); @@ -4653,7 +4660,7 @@ int igc_close(struct net_device *netdev)  /**   * igc_ioctl - Access the hwtstamp interface   * @netdev: network interface device structure - * @ifreq: interface request data + * @ifr: interface request data   * @cmd: ioctl command   **/  static int igc_ioctl(struct net_device *netdev, struct ifreq *ifr, int cmd) @@ -4694,14 +4701,35 @@ static int igc_save_launchtime_params(struct igc_adapter *adapter, int queue,  	return 0;  } -static bool validate_schedule(const struct tc_taprio_qopt_offload *qopt) +static bool is_base_time_past(ktime_t base_time, const struct timespec64 *now) +{ +	struct timespec64 b; + +	b = ktime_to_timespec64(base_time); + +	return timespec64_compare(now, &b) > 0; +} + +static bool validate_schedule(struct igc_adapter *adapter, +			      const struct tc_taprio_qopt_offload *qopt)  {  	int queue_uses[IGC_MAX_TX_QUEUES] = { }; +	struct timespec64 now;  	size_t n;  	if (qopt->cycle_time_extension)  		return false; +	igc_ptp_read(adapter, &now); + +	/* If we program the controller's BASET registers with a time +	 * in the future, it will hold all the packets until that +	 * time, causing a lot of TX Hangs, so to avoid that, we +	 * reject schedules that would start in the future. +	 */ +	if (!is_base_time_past(qopt->base_time, &now)) +		return false; +  	for (n = 0; n < qopt->num_entries; n++) {  		const struct tc_taprio_sched_entry *e;  		int i; @@ -4756,7 +4784,7 @@ static int igc_save_qbv_schedule(struct igc_adapter *adapter,  	if (adapter->base_time)  		return -EALREADY; -	if (!validate_schedule(qopt)) +	if (!validate_schedule(adapter, qopt))  		return -EINVAL;  	adapter->cycle_time = qopt->cycle_time; | 
