aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlexander Duyck <alexander.h.duyck@intel.com>2009-12-02 11:47:57 -0500
committerDavid S. Miller <davem@davemloft.net>2009-12-02 22:57:13 -0500
commite95524a726904a1d2b91552f0577838f67d53c6c (patch)
treeae04c29ad8125fb43bbf34f30d9a90e120737233
parenta7d5ca40ff56e2cd4e30bbe91f2d0deab6bfc006 (diff)
bnx2: remove skb_dma_map/unmap calls from driver
Due to the fact that skb_dma_map/unmap do not work correctly when a HW IOMMU is enabled it has been recommended to go about removing the calls from the network device drivers. [ Fix bnx2_free_tx_skbs() ring indexing and use NETDEV_TX_OK return code in bnx2_start_xmit() after cleaning up DMA mapping errors. -Mchan ] Signed-off-by: Alexander Duyck <alexander.h.duyck@intel.com> Signed-off-by: Michael Chan <mchan@broadcom.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--drivers/net/bnx2.c72
-rw-r--r--drivers/net/bnx2.h1
2 files changed, 61 insertions, 12 deletions
diff --git a/drivers/net/bnx2.c b/drivers/net/bnx2.c
index 539d23b594ce..f47bf50602d9 100644
--- a/drivers/net/bnx2.c
+++ b/drivers/net/bnx2.c
@@ -2815,13 +2815,21 @@ bnx2_tx_int(struct bnx2 *bp, struct bnx2_napi *bnapi, int budget)
2815 } 2815 }
2816 } 2816 }
2817 2817
2818 skb_dma_unmap(&bp->pdev->dev, skb, DMA_TO_DEVICE); 2818 pci_unmap_single(bp->pdev, pci_unmap_addr(tx_buf, mapping),
2819 skb_headlen(skb), PCI_DMA_TODEVICE);
2819 2820
2820 tx_buf->skb = NULL; 2821 tx_buf->skb = NULL;
2821 last = tx_buf->nr_frags; 2822 last = tx_buf->nr_frags;
2822 2823
2823 for (i = 0; i < last; i++) { 2824 for (i = 0; i < last; i++) {
2824 sw_cons = NEXT_TX_BD(sw_cons); 2825 sw_cons = NEXT_TX_BD(sw_cons);
2826
2827 pci_unmap_page(bp->pdev,
2828 pci_unmap_addr(
2829 &txr->tx_buf_ring[TX_RING_IDX(sw_cons)],
2830 mapping),
2831 skb_shinfo(skb)->frags[i].size,
2832 PCI_DMA_TODEVICE);
2825 } 2833 }
2826 2834
2827 sw_cons = NEXT_TX_BD(sw_cons); 2835 sw_cons = NEXT_TX_BD(sw_cons);
@@ -5295,17 +5303,29 @@ bnx2_free_tx_skbs(struct bnx2 *bp)
5295 for (j = 0; j < TX_DESC_CNT; ) { 5303 for (j = 0; j < TX_DESC_CNT; ) {
5296 struct sw_tx_bd *tx_buf = &txr->tx_buf_ring[j]; 5304 struct sw_tx_bd *tx_buf = &txr->tx_buf_ring[j];
5297 struct sk_buff *skb = tx_buf->skb; 5305 struct sk_buff *skb = tx_buf->skb;
5306 int k, last;
5298 5307
5299 if (skb == NULL) { 5308 if (skb == NULL) {
5300 j++; 5309 j++;
5301 continue; 5310 continue;
5302 } 5311 }
5303 5312
5304 skb_dma_unmap(&bp->pdev->dev, skb, DMA_TO_DEVICE); 5313 pci_unmap_single(bp->pdev,
5314 pci_unmap_addr(tx_buf, mapping),
5315 skb_headlen(skb),
5316 PCI_DMA_TODEVICE);
5305 5317
5306 tx_buf->skb = NULL; 5318 tx_buf->skb = NULL;
5307 5319
5308 j += skb_shinfo(skb)->nr_frags + 1; 5320 last = tx_buf->nr_frags;
5321 j++;
5322 for (k = 0; k < last; k++, j++) {
5323 tx_buf = &txr->tx_buf_ring[TX_RING_IDX(j)];
5324 pci_unmap_page(bp->pdev,
5325 pci_unmap_addr(tx_buf, mapping),
5326 skb_shinfo(skb)->frags[k].size,
5327 PCI_DMA_TODEVICE);
5328 }
5309 dev_kfree_skb(skb); 5329 dev_kfree_skb(skb);
5310 } 5330 }
5311 } 5331 }
@@ -5684,11 +5704,12 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode)
5684 for (i = 14; i < pkt_size; i++) 5704 for (i = 14; i < pkt_size; i++)
5685 packet[i] = (unsigned char) (i & 0xff); 5705 packet[i] = (unsigned char) (i & 0xff);
5686 5706
5687 if (skb_dma_map(&bp->pdev->dev, skb, DMA_TO_DEVICE)) { 5707 map = pci_map_single(bp->pdev, skb->data, pkt_size,
5708 PCI_DMA_TODEVICE);
5709 if (pci_dma_mapping_error(bp->pdev, map)) {
5688 dev_kfree_skb(skb); 5710 dev_kfree_skb(skb);
5689 return -EIO; 5711 return -EIO;
5690 } 5712 }
5691 map = skb_shinfo(skb)->dma_head;
5692 5713
5693 REG_WR(bp, BNX2_HC_COMMAND, 5714 REG_WR(bp, BNX2_HC_COMMAND,
5694 bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW_WO_INT); 5715 bp->hc_cmd | BNX2_HC_COMMAND_COAL_NOW_WO_INT);
@@ -5723,7 +5744,7 @@ bnx2_run_loopback(struct bnx2 *bp, int loopback_mode)
5723 5744
5724 udelay(5); 5745 udelay(5);
5725 5746
5726 skb_dma_unmap(&bp->pdev->dev, skb, DMA_TO_DEVICE); 5747 pci_unmap_single(bp->pdev, map, pkt_size, PCI_DMA_TODEVICE);
5727 dev_kfree_skb(skb); 5748 dev_kfree_skb(skb);
5728 5749
5729 if (bnx2_get_hw_tx_cons(tx_napi) != txr->tx_prod) 5750 if (bnx2_get_hw_tx_cons(tx_napi) != txr->tx_prod)
@@ -6302,7 +6323,6 @@ bnx2_start_xmit(struct sk_buff *skb, struct net_device *dev)
6302 struct bnx2_napi *bnapi; 6323 struct bnx2_napi *bnapi;
6303 struct bnx2_tx_ring_info *txr; 6324 struct bnx2_tx_ring_info *txr;
6304 struct netdev_queue *txq; 6325 struct netdev_queue *txq;
6305 struct skb_shared_info *sp;
6306 6326
6307 /* Determine which tx ring we will be placed on */ 6327 /* Determine which tx ring we will be placed on */
6308 i = skb_get_queue_mapping(skb); 6328 i = skb_get_queue_mapping(skb);
@@ -6367,16 +6387,15 @@ bnx2_start_xmit(struct sk_buff *skb, struct net_device *dev)
6367 } else 6387 } else
6368 mss = 0; 6388 mss = 0;
6369 6389
6370 if (skb_dma_map(&bp->pdev->dev, skb, DMA_TO_DEVICE)) { 6390 mapping = pci_map_single(bp->pdev, skb->data, len, PCI_DMA_TODEVICE);
6391 if (pci_dma_mapping_error(bp->pdev, mapping)) {
6371 dev_kfree_skb(skb); 6392 dev_kfree_skb(skb);
6372 return NETDEV_TX_OK; 6393 return NETDEV_TX_OK;
6373 } 6394 }
6374 6395
6375 sp = skb_shinfo(skb);
6376 mapping = sp->dma_head;
6377
6378 tx_buf = &txr->tx_buf_ring[ring_prod]; 6396 tx_buf = &txr->tx_buf_ring[ring_prod];
6379 tx_buf->skb = skb; 6397 tx_buf->skb = skb;
6398 pci_unmap_addr_set(tx_buf, mapping, mapping);
6380 6399
6381 txbd = &txr->tx_desc_ring[ring_prod]; 6400 txbd = &txr->tx_desc_ring[ring_prod];
6382 6401
@@ -6397,7 +6416,12 @@ bnx2_start_xmit(struct sk_buff *skb, struct net_device *dev)
6397 txbd = &txr->tx_desc_ring[ring_prod]; 6416 txbd = &txr->tx_desc_ring[ring_prod];
6398 6417
6399 len = frag->size; 6418 len = frag->size;
6400 mapping = sp->dma_maps[i]; 6419 mapping = pci_map_page(bp->pdev, frag->page, frag->page_offset,
6420 len, PCI_DMA_TODEVICE);
6421 if (pci_dma_mapping_error(bp->pdev, mapping))
6422 goto dma_error;
6423 pci_unmap_addr_set(&txr->tx_buf_ring[ring_prod], mapping,
6424 mapping);
6401 6425
6402 txbd->tx_bd_haddr_hi = (u64) mapping >> 32; 6426 txbd->tx_bd_haddr_hi = (u64) mapping >> 32;
6403 txbd->tx_bd_haddr_lo = (u64) mapping & 0xffffffff; 6427 txbd->tx_bd_haddr_lo = (u64) mapping & 0xffffffff;
@@ -6424,6 +6448,30 @@ bnx2_start_xmit(struct sk_buff *skb, struct net_device *dev)
6424 } 6448 }
6425 6449
6426 return NETDEV_TX_OK; 6450 return NETDEV_TX_OK;
6451dma_error:
6452 /* save value of frag that failed */
6453 last_frag = i;
6454
6455 /* start back at beginning and unmap skb */
6456 prod = txr->tx_prod;
6457 ring_prod = TX_RING_IDX(prod);
6458 tx_buf = &txr->tx_buf_ring[ring_prod];
6459 tx_buf->skb = NULL;
6460 pci_unmap_single(bp->pdev, pci_unmap_addr(tx_buf, mapping),
6461 skb_headlen(skb), PCI_DMA_TODEVICE);
6462
6463 /* unmap remaining mapped pages */
6464 for (i = 0; i < last_frag; i++) {
6465 prod = NEXT_TX_BD(prod);
6466 ring_prod = TX_RING_IDX(prod);
6467 tx_buf = &txr->tx_buf_ring[ring_prod];
6468 pci_unmap_page(bp->pdev, pci_unmap_addr(tx_buf, mapping),
6469 skb_shinfo(skb)->frags[i].size,
6470 PCI_DMA_TODEVICE);
6471 }
6472
6473 dev_kfree_skb(skb);
6474 return NETDEV_TX_OK;
6427} 6475}
6428 6476
6429/* Called with rtnl_lock */ 6477/* Called with rtnl_lock */
diff --git a/drivers/net/bnx2.h b/drivers/net/bnx2.h
index a4d83409f205..4908b9f74260 100644
--- a/drivers/net/bnx2.h
+++ b/drivers/net/bnx2.h
@@ -6559,6 +6559,7 @@ struct sw_pg {
6559 6559
6560struct sw_tx_bd { 6560struct sw_tx_bd {
6561 struct sk_buff *skb; 6561 struct sk_buff *skb;
6562 DECLARE_PCI_UNMAP_ADDR(mapping)
6562 unsigned short is_gso; 6563 unsigned short is_gso;
6563 unsigned short nr_frags; 6564 unsigned short nr_frags;
6564}; 6565};