aboutsummaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorLennert Buytenhek <buytenh@wantstofly.org>2008-08-27 23:53:18 -0400
committerLennert Buytenhek <buytenh@marvell.com>2008-09-14 08:09:06 -0400
commit99ab08e091df65ce8e191d85ebdc23f78b8355cb (patch)
tree08ec6a76f90f52858be703210a76b906953ed54a /drivers
parenta418950c1378c4a3722baecdbe98df78ea23f231 (diff)
mv643xx_eth: replace array of skbs awaiting transmit completion with a queue
Get rid of the skb pointer array that we currently use for transmit reclaim, and replace it with an skb queue, to which skbuffs are appended when they are passed to the xmit function, and removed from the front and freed when we do transmit queue reclaim and hit a descriptor with the 'owned by device' bit clear and 'last descriptor' bit set. Signed-off-by: Lennert Buytenhek <buytenh@marvell.com>
Diffstat (limited to 'drivers')
-rw-r--r--drivers/net/mv643xx_eth.c39
1 files changed, 10 insertions, 29 deletions
diff --git a/drivers/net/mv643xx_eth.c b/drivers/net/mv643xx_eth.c
index 7410eca87823..099e0be0d8c3 100644
--- a/drivers/net/mv643xx_eth.c
+++ b/drivers/net/mv643xx_eth.c
@@ -336,7 +336,8 @@ struct tx_queue {
336 struct tx_desc *tx_desc_area; 336 struct tx_desc *tx_desc_area;
337 dma_addr_t tx_desc_dma; 337 dma_addr_t tx_desc_dma;
338 int tx_desc_area_size; 338 int tx_desc_area_size;
339 struct sk_buff **tx_skb; 339
340 struct sk_buff_head tx_skb;
340 341
341 unsigned long tx_packets; 342 unsigned long tx_packets;
342 unsigned long tx_bytes; 343 unsigned long tx_bytes;
@@ -676,10 +677,8 @@ static void txq_submit_frag_skb(struct tx_queue *txq, struct sk_buff *skb)
676 desc->cmd_sts = BUFFER_OWNED_BY_DMA | 677 desc->cmd_sts = BUFFER_OWNED_BY_DMA |
677 ZERO_PADDING | TX_LAST_DESC | 678 ZERO_PADDING | TX_LAST_DESC |
678 TX_ENABLE_INTERRUPT; 679 TX_ENABLE_INTERRUPT;
679 txq->tx_skb[tx_index] = skb;
680 } else { 680 } else {
681 desc->cmd_sts = BUFFER_OWNED_BY_DMA; 681 desc->cmd_sts = BUFFER_OWNED_BY_DMA;
682 txq->tx_skb[tx_index] = NULL;
683 } 682 }
684 683
685 desc->l4i_chk = 0; 684 desc->l4i_chk = 0;
@@ -712,13 +711,10 @@ static void txq_submit_skb(struct tx_queue *txq, struct sk_buff *skb)
712 711
713 if (nr_frags) { 712 if (nr_frags) {
714 txq_submit_frag_skb(txq, skb); 713 txq_submit_frag_skb(txq, skb);
715
716 length = skb_headlen(skb); 714 length = skb_headlen(skb);
717 txq->tx_skb[tx_index] = NULL;
718 } else { 715 } else {
719 cmd_sts |= ZERO_PADDING | TX_LAST_DESC | TX_ENABLE_INTERRUPT; 716 cmd_sts |= ZERO_PADDING | TX_LAST_DESC | TX_ENABLE_INTERRUPT;
720 length = skb->len; 717 length = skb->len;
721 txq->tx_skb[tx_index] = skb;
722 } 718 }
723 719
724 desc->byte_cnt = length; 720 desc->byte_cnt = length;
@@ -772,6 +768,8 @@ static void txq_submit_skb(struct tx_queue *txq, struct sk_buff *skb)
772 desc->l4i_chk = 0; 768 desc->l4i_chk = 0;
773 } 769 }
774 770
771 __skb_queue_tail(&txq->tx_skb, skb);
772
775 /* ensure all other descriptors are written before first cmd_sts */ 773 /* ensure all other descriptors are written before first cmd_sts */
776 wmb(); 774 wmb();
777 desc->cmd_sts = cmd_sts; 775 desc->cmd_sts = cmd_sts;
@@ -884,8 +882,9 @@ static int txq_reclaim(struct tx_queue *txq, int budget, int force)
884 reclaimed++; 882 reclaimed++;
885 txq->tx_desc_count--; 883 txq->tx_desc_count--;
886 884
887 skb = txq->tx_skb[tx_index]; 885 skb = NULL;
888 txq->tx_skb[tx_index] = NULL; 886 if (cmd_sts & TX_LAST_DESC)
887 skb = __skb_dequeue(&txq->tx_skb);
889 888
890 if (cmd_sts & ERROR_SUMMARY) { 889 if (cmd_sts & ERROR_SUMMARY) {
891 dev_printk(KERN_INFO, &mp->dev->dev, "tx error\n"); 890 dev_printk(KERN_INFO, &mp->dev->dev, "tx error\n");
@@ -1692,18 +1691,11 @@ static int txq_init(struct mv643xx_eth_private *mp, int index)
1692 if (txq->tx_desc_area == NULL) { 1691 if (txq->tx_desc_area == NULL) {
1693 dev_printk(KERN_ERR, &mp->dev->dev, 1692 dev_printk(KERN_ERR, &mp->dev->dev,
1694 "can't allocate tx ring (%d bytes)\n", size); 1693 "can't allocate tx ring (%d bytes)\n", size);
1695 goto out; 1694 return -ENOMEM;
1696 } 1695 }
1697 memset(txq->tx_desc_area, 0, size); 1696 memset(txq->tx_desc_area, 0, size);
1698 1697
1699 txq->tx_desc_area_size = size; 1698 txq->tx_desc_area_size = size;
1700 txq->tx_skb = kmalloc(txq->tx_ring_size * sizeof(*txq->tx_skb),
1701 GFP_KERNEL);
1702 if (txq->tx_skb == NULL) {
1703 dev_printk(KERN_ERR, &mp->dev->dev,
1704 "can't allocate tx skb ring\n");
1705 goto out_free;
1706 }
1707 1699
1708 tx_desc = (struct tx_desc *)txq->tx_desc_area; 1700 tx_desc = (struct tx_desc *)txq->tx_desc_area;
1709 for (i = 0; i < txq->tx_ring_size; i++) { 1701 for (i = 0; i < txq->tx_ring_size; i++) {
@@ -1719,18 +1711,9 @@ static int txq_init(struct mv643xx_eth_private *mp, int index)
1719 nexti * sizeof(struct tx_desc); 1711 nexti * sizeof(struct tx_desc);
1720 } 1712 }
1721 1713
1722 return 0; 1714 skb_queue_head_init(&txq->tx_skb);
1723 1715
1724out_free: 1716 return 0;
1725 if (index == 0 && size <= mp->tx_desc_sram_size)
1726 iounmap(txq->tx_desc_area);
1727 else
1728 dma_free_coherent(NULL, size,
1729 txq->tx_desc_area,
1730 txq->tx_desc_dma);
1731
1732out:
1733 return -ENOMEM;
1734} 1717}
1735 1718
1736static void txq_deinit(struct tx_queue *txq) 1719static void txq_deinit(struct tx_queue *txq)
@@ -1748,8 +1731,6 @@ static void txq_deinit(struct tx_queue *txq)
1748 else 1731 else
1749 dma_free_coherent(NULL, txq->tx_desc_area_size, 1732 dma_free_coherent(NULL, txq->tx_desc_area_size,
1750 txq->tx_desc_area, txq->tx_desc_dma); 1733 txq->tx_desc_area, txq->tx_desc_dma);
1751
1752 kfree(txq->tx_skb);
1753} 1734}
1754 1735
1755 1736