diff options
Diffstat (limited to 'drivers/net')
-rw-r--r-- | drivers/net/tg3.c | 34 |
1 files changed, 25 insertions, 9 deletions
diff --git a/drivers/net/tg3.c b/drivers/net/tg3.c index 9975cdb38831..52dd516ba786 100644 --- a/drivers/net/tg3.c +++ b/drivers/net/tg3.c | |||
@@ -10935,7 +10935,7 @@ static int tg3_test_memory(struct tg3 *tp) | |||
10935 | #define TG3_MAC_LOOPBACK 0 | 10935 | #define TG3_MAC_LOOPBACK 0 |
10936 | #define TG3_PHY_LOOPBACK 1 | 10936 | #define TG3_PHY_LOOPBACK 1 |
10937 | 10937 | ||
10938 | static int tg3_run_loopback(struct tg3 *tp, int loopback_mode) | 10938 | static int tg3_run_loopback(struct tg3 *tp, u32 pktsz, int loopback_mode) |
10939 | { | 10939 | { |
10940 | u32 mac_mode, rx_start_idx, rx_idx, tx_idx, opaque_key; | 10940 | u32 mac_mode, rx_start_idx, rx_idx, tx_idx, opaque_key; |
10941 | u32 desc_idx, coal_now; | 10941 | u32 desc_idx, coal_now; |
@@ -11033,7 +11033,7 @@ static int tg3_run_loopback(struct tg3 *tp, int loopback_mode) | |||
11033 | 11033 | ||
11034 | err = -EIO; | 11034 | err = -EIO; |
11035 | 11035 | ||
11036 | tx_len = 1514; | 11036 | tx_len = pktsz; |
11037 | skb = netdev_alloc_skb(tp->dev, tx_len); | 11037 | skb = netdev_alloc_skb(tp->dev, tx_len); |
11038 | if (!skb) | 11038 | if (!skb) |
11039 | return -ENOMEM; | 11039 | return -ENOMEM; |
@@ -11042,7 +11042,7 @@ static int tg3_run_loopback(struct tg3 *tp, int loopback_mode) | |||
11042 | memcpy(tx_data, tp->dev->dev_addr, 6); | 11042 | memcpy(tx_data, tp->dev->dev_addr, 6); |
11043 | memset(tx_data + 6, 0x0, 8); | 11043 | memset(tx_data + 6, 0x0, 8); |
11044 | 11044 | ||
11045 | tw32(MAC_RX_MTU_SIZE, tx_len + 4); | 11045 | tw32(MAC_RX_MTU_SIZE, tx_len + ETH_FCS_LEN); |
11046 | 11046 | ||
11047 | for (i = 14; i < tx_len; i++) | 11047 | for (i = 14; i < tx_len; i++) |
11048 | tx_data[i] = (u8) (i & 0xff); | 11048 | tx_data[i] = (u8) (i & 0xff); |
@@ -11098,8 +11098,6 @@ static int tg3_run_loopback(struct tg3 *tp, int loopback_mode) | |||
11098 | desc = &rnapi->rx_rcb[rx_start_idx]; | 11098 | desc = &rnapi->rx_rcb[rx_start_idx]; |
11099 | desc_idx = desc->opaque & RXD_OPAQUE_INDEX_MASK; | 11099 | desc_idx = desc->opaque & RXD_OPAQUE_INDEX_MASK; |
11100 | opaque_key = desc->opaque & RXD_OPAQUE_RING_MASK; | 11100 | opaque_key = desc->opaque & RXD_OPAQUE_RING_MASK; |
11101 | if (opaque_key != RXD_OPAQUE_RING_STD) | ||
11102 | goto out; | ||
11103 | 11101 | ||
11104 | if ((desc->err_vlan & RXD_ERR_MASK) != 0 && | 11102 | if ((desc->err_vlan & RXD_ERR_MASK) != 0 && |
11105 | (desc->err_vlan != RXD_ERR_ODD_NIBBLE_RCVD_MII)) | 11103 | (desc->err_vlan != RXD_ERR_ODD_NIBBLE_RCVD_MII)) |
@@ -11109,9 +11107,20 @@ static int tg3_run_loopback(struct tg3 *tp, int loopback_mode) | |||
11109 | if (rx_len != tx_len) | 11107 | if (rx_len != tx_len) |
11110 | goto out; | 11108 | goto out; |
11111 | 11109 | ||
11112 | rx_skb = tpr->rx_std_buffers[desc_idx].skb; | 11110 | if (pktsz <= TG3_RX_STD_DMA_SZ - ETH_FCS_LEN) { |
11111 | if (opaque_key != RXD_OPAQUE_RING_STD) | ||
11112 | goto out; | ||
11113 | |||
11114 | rx_skb = tpr->rx_std_buffers[desc_idx].skb; | ||
11115 | map = dma_unmap_addr(&tpr->rx_std_buffers[desc_idx], mapping); | ||
11116 | } else { | ||
11117 | if (opaque_key != RXD_OPAQUE_RING_JUMBO) | ||
11118 | goto out; | ||
11119 | |||
11120 | rx_skb = tpr->rx_jmb_buffers[desc_idx].skb; | ||
11121 | map = dma_unmap_addr(&tpr->rx_jmb_buffers[desc_idx], mapping); | ||
11122 | } | ||
11113 | 11123 | ||
11114 | map = dma_unmap_addr(&tpr->rx_std_buffers[desc_idx], mapping); | ||
11115 | pci_dma_sync_single_for_cpu(tp->pdev, map, rx_len, PCI_DMA_FROMDEVICE); | 11124 | pci_dma_sync_single_for_cpu(tp->pdev, map, rx_len, PCI_DMA_FROMDEVICE); |
11116 | 11125 | ||
11117 | for (i = 14; i < tx_len; i++) { | 11126 | for (i = 14; i < tx_len; i++) { |
@@ -11177,9 +11186,13 @@ static int tg3_test_loopback(struct tg3 *tp) | |||
11177 | CPMU_CTRL_LINK_AWARE_MODE)); | 11186 | CPMU_CTRL_LINK_AWARE_MODE)); |
11178 | } | 11187 | } |
11179 | 11188 | ||
11180 | if (tg3_run_loopback(tp, TG3_MAC_LOOPBACK)) | 11189 | if (tg3_run_loopback(tp, ETH_FRAME_LEN, TG3_MAC_LOOPBACK)) |
11181 | err |= TG3_MAC_LOOPBACK_FAILED; | 11190 | err |= TG3_MAC_LOOPBACK_FAILED; |
11182 | 11191 | ||
11192 | if ((tp->tg3_flags & TG3_FLAG_JUMBO_RING_ENABLE) && | ||
11193 | tg3_run_loopback(tp, 9000 + ETH_HLEN, TG3_MAC_LOOPBACK)) | ||
11194 | err |= (TG3_MAC_LOOPBACK_FAILED << 2); | ||
11195 | |||
11183 | if (tp->tg3_flags & TG3_FLAG_CPMU_PRESENT) { | 11196 | if (tp->tg3_flags & TG3_FLAG_CPMU_PRESENT) { |
11184 | tw32(TG3_CPMU_CTRL, cpmuctrl); | 11197 | tw32(TG3_CPMU_CTRL, cpmuctrl); |
11185 | 11198 | ||
@@ -11189,8 +11202,11 @@ static int tg3_test_loopback(struct tg3 *tp) | |||
11189 | 11202 | ||
11190 | if (!(tp->phy_flags & TG3_PHYFLG_PHY_SERDES) && | 11203 | if (!(tp->phy_flags & TG3_PHYFLG_PHY_SERDES) && |
11191 | !(tp->tg3_flags3 & TG3_FLG3_USE_PHYLIB)) { | 11204 | !(tp->tg3_flags3 & TG3_FLG3_USE_PHYLIB)) { |
11192 | if (tg3_run_loopback(tp, TG3_PHY_LOOPBACK)) | 11205 | if (tg3_run_loopback(tp, ETH_FRAME_LEN, TG3_PHY_LOOPBACK)) |
11193 | err |= TG3_PHY_LOOPBACK_FAILED; | 11206 | err |= TG3_PHY_LOOPBACK_FAILED; |
11207 | if ((tp->tg3_flags & TG3_FLAG_JUMBO_RING_ENABLE) && | ||
11208 | tg3_run_loopback(tp, 9000 + ETH_HLEN, TG3_PHY_LOOPBACK)) | ||
11209 | err |= (TG3_PHY_LOOPBACK_FAILED << 2); | ||
11194 | } | 11210 | } |
11195 | 11211 | ||
11196 | /* Re-enable gphy autopowerdown. */ | 11212 | /* Re-enable gphy autopowerdown. */ |