diff options
-rw-r--r-- | drivers/net/e1000/e1000_main.c | 43 |
1 files changed, 27 insertions, 16 deletions
diff --git a/drivers/net/e1000/e1000_main.c b/drivers/net/e1000/e1000_main.c index 9c6c1fbbc412..4a8c069e6c7c 100644 --- a/drivers/net/e1000/e1000_main.c +++ b/drivers/net/e1000/e1000_main.c | |||
@@ -2384,11 +2384,10 @@ e1000_clean_tx_irq(struct e1000_adapter *adapter) | |||
2384 | eop_desc = E1000_TX_DESC(*tx_ring, eop); | 2384 | eop_desc = E1000_TX_DESC(*tx_ring, eop); |
2385 | 2385 | ||
2386 | while(eop_desc->upper.data & cpu_to_le32(E1000_TXD_STAT_DD)) { | 2386 | while(eop_desc->upper.data & cpu_to_le32(E1000_TXD_STAT_DD)) { |
2387 | /* pre-mature writeback of Tx descriptors */ | 2387 | /* Premature writeback of Tx descriptors clear (free buffers |
2388 | /* clear (free buffers and unmap pci_mapping) */ | 2388 | * and unmap pci_mapping) previous_buffer_info */ |
2389 | /* previous_buffer_info */ | ||
2390 | if (likely(adapter->previous_buffer_info.skb != NULL)) { | 2389 | if (likely(adapter->previous_buffer_info.skb != NULL)) { |
2391 | e1000_unmap_and_free_tx_resource(adapter, | 2390 | e1000_unmap_and_free_tx_resource(adapter, |
2392 | &adapter->previous_buffer_info); | 2391 | &adapter->previous_buffer_info); |
2393 | } | 2392 | } |
2394 | 2393 | ||
@@ -2397,20 +2396,25 @@ e1000_clean_tx_irq(struct e1000_adapter *adapter) | |||
2397 | buffer_info = &tx_ring->buffer_info[i]; | 2396 | buffer_info = &tx_ring->buffer_info[i]; |
2398 | cleaned = (i == eop); | 2397 | cleaned = (i == eop); |
2399 | 2398 | ||
2400 | /* pre-mature writeback of Tx descriptors */ | 2399 | #ifdef NETIF_F_TSO |
2401 | /* save the cleaning of the this for the */ | 2400 | if (!(netdev->features & NETIF_F_TSO)) { |
2402 | /* next iteration */ | 2401 | #endif |
2403 | if (cleaned) { | 2402 | e1000_unmap_and_free_tx_resource(adapter, |
2404 | memcpy(&adapter->previous_buffer_info, | 2403 | buffer_info); |
2405 | buffer_info, | 2404 | #ifdef NETIF_F_TSO |
2406 | sizeof(struct e1000_buffer)); | ||
2407 | memset(buffer_info, | ||
2408 | 0, | ||
2409 | sizeof(struct e1000_buffer)); | ||
2410 | } else { | 2405 | } else { |
2411 | e1000_unmap_and_free_tx_resource(adapter, | 2406 | if (cleaned) { |
2412 | buffer_info); | 2407 | memcpy(&adapter->previous_buffer_info, |
2408 | buffer_info, | ||
2409 | sizeof(struct e1000_buffer)); | ||
2410 | memset(buffer_info, 0, | ||
2411 | sizeof(struct e1000_buffer)); | ||
2412 | } else { | ||
2413 | e1000_unmap_and_free_tx_resource( | ||
2414 | adapter, buffer_info); | ||
2415 | } | ||
2413 | } | 2416 | } |
2417 | #endif | ||
2414 | 2418 | ||
2415 | tx_desc->buffer_addr = 0; | 2419 | tx_desc->buffer_addr = 0; |
2416 | tx_desc->lower.data = 0; | 2420 | tx_desc->lower.data = 0; |
@@ -2443,7 +2447,14 @@ e1000_clean_tx_irq(struct e1000_adapter *adapter) | |||
2443 | !(E1000_READ_REG(&adapter->hw, STATUS) & E1000_STATUS_TXOFF)) | 2447 | !(E1000_READ_REG(&adapter->hw, STATUS) & E1000_STATUS_TXOFF)) |
2444 | netif_stop_queue(netdev); | 2448 | netif_stop_queue(netdev); |
2445 | } | 2449 | } |
2450 | #ifdef NETIF_F_TSO | ||
2446 | 2451 | ||
2452 | if( unlikely(!(eop_desc->upper.data & cpu_to_le32(E1000_TXD_STAT_DD)) && | ||
2453 | time_after(jiffies, adapter->previous_buffer_info.time_stamp + HZ))) | ||
2454 | e1000_unmap_and_free_tx_resource( | ||
2455 | adapter, &adapter->previous_buffer_info); | ||
2456 | |||
2457 | #endif | ||
2447 | return cleaned; | 2458 | return cleaned; |
2448 | } | 2459 | } |
2449 | 2460 | ||