Skip to content

Commit

Permalink
net: mv643xx_eth: reclaim TX skbs only when released by the HW
Browse files Browse the repository at this point in the history
ATM, txq_reclaim will dequeue and free an skb for each tx desc released
by the hw that has TX_LAST_DESC set. However, in case of TSO, each
hw desc embedding the last part of a segment has TX_LAST_DESC set,
losing the one-to-one 'last skb frag'/'TX_LAST_DESC set' correspondance,
which causes data corruption.

Fix this by checking TX_ENABLE_INTERRUPT instead of TX_LAST_DESC, and
warn when trying to dequeue from an empty txq (which can be symptomatic
of releasing skbs prematurely).

Fixes: 3ae8f4e ('net: mv643xx_eth: Implement software TSO')
Reported-by: Slawomir Gajzner <[email protected]>
Reported-by: Julien D'Ascenzio <[email protected]>
Signed-off-by: Karl Beldan <[email protected]>
Cc: Ian Campbell <[email protected]>
Cc: Eric Dumazet <[email protected]>
Cc: Ezequiel Garcia <[email protected]>
Cc: Sebastian Hesselbarth <[email protected]>
Signed-off-by: David S. Miller <[email protected]>
  • Loading branch information
Karl Beldan authored and davem330 committed Nov 6, 2014
1 parent 66f1c44 commit 2c2a9cb
Showing 1 changed file with 10 additions and 8 deletions.
18 changes: 10 additions & 8 deletions drivers/net/ethernet/marvell/mv643xx_eth.c
Original file line number Diff line number Diff line change
Expand Up @@ -1047,7 +1047,6 @@ static int txq_reclaim(struct tx_queue *txq, int budget, int force)
int tx_index;
struct tx_desc *desc;
u32 cmd_sts;
struct sk_buff *skb;

tx_index = txq->tx_used_desc;
desc = &txq->tx_desc_area[tx_index];
Expand All @@ -1066,19 +1065,22 @@ static int txq_reclaim(struct tx_queue *txq, int budget, int force)
reclaimed++;
txq->tx_desc_count--;

skb = NULL;
if (cmd_sts & TX_LAST_DESC)
skb = __skb_dequeue(&txq->tx_skb);
if (!IS_TSO_HEADER(txq, desc->buf_ptr))
dma_unmap_single(mp->dev->dev.parent, desc->buf_ptr,
desc->byte_cnt, DMA_TO_DEVICE);

if (cmd_sts & TX_ENABLE_INTERRUPT) {
struct sk_buff *skb = __skb_dequeue(&txq->tx_skb);

if (!WARN_ON(!skb))
dev_kfree_skb(skb);
}

if (cmd_sts & ERROR_SUMMARY) {
netdev_info(mp->dev, "tx error\n");
mp->dev->stats.tx_errors++;
}

if (!IS_TSO_HEADER(txq, desc->buf_ptr))
dma_unmap_single(mp->dev->dev.parent, desc->buf_ptr,
desc->byte_cnt, DMA_TO_DEVICE);
dev_kfree_skb(skb);
}

__netif_tx_unlock_bh(nq);
Expand Down

0 comments on commit 2c2a9cb

Please sign in to comment.