aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/tg3.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/net/tg3.c')
-rw-r--r--drivers/net/tg3.c53
1 files changed, 49 insertions, 4 deletions
diff --git a/drivers/net/tg3.c b/drivers/net/tg3.c
index 2c36e70e37e5..63d4b2c797e8 100644
--- a/drivers/net/tg3.c
+++ b/drivers/net/tg3.c
@@ -3880,6 +3880,40 @@ out_unlock:
3880 return NETDEV_TX_OK; 3880 return NETDEV_TX_OK;
3881} 3881}
3882 3882
3883#if TG3_TSO_SUPPORT != 0
3884static int tg3_start_xmit_dma_bug(struct sk_buff *, struct net_device *);
3885
3886/* Use GSO to workaround a rare TSO bug that may be triggered when the
3887 * TSO header is greater than 80 bytes.
3888 */
3889static int tg3_tso_bug(struct tg3 *tp, struct sk_buff *skb)
3890{
3891 struct sk_buff *segs, *nskb;
3892
3893 /* Estimate the number of fragments in the worst case */
3894 if (unlikely(TX_BUFFS_AVAIL(tp) <= (skb_shinfo(skb)->gso_segs * 3))) {
3895 netif_stop_queue(tp->dev);
3896 return NETDEV_TX_BUSY;
3897 }
3898
3899 segs = skb_gso_segment(skb, tp->dev->features & ~NETIF_F_TSO);
3900 if (unlikely(IS_ERR(segs)))
3901 goto tg3_tso_bug_end;
3902
3903 do {
3904 nskb = segs;
3905 segs = segs->next;
3906 nskb->next = NULL;
3907 tg3_start_xmit_dma_bug(nskb, tp->dev);
3908 } while (segs);
3909
3910tg3_tso_bug_end:
3911 dev_kfree_skb(skb);
3912
3913 return NETDEV_TX_OK;
3914}
3915#endif
3916
3883/* hard_start_xmit for devices that have the 4G bug and/or 40-bit bug and 3917/* hard_start_xmit for devices that have the 4G bug and/or 40-bit bug and
3884 * support TG3_FLG2_HW_TSO_1 or firmware TSO only. 3918 * support TG3_FLG2_HW_TSO_1 or firmware TSO only.
3885 */ 3919 */
@@ -3916,7 +3950,7 @@ static int tg3_start_xmit_dma_bug(struct sk_buff *skb, struct net_device *dev)
3916 mss = 0; 3950 mss = 0;
3917 if (skb->len > (tp->dev->mtu + ETH_HLEN) && 3951 if (skb->len > (tp->dev->mtu + ETH_HLEN) &&
3918 (mss = skb_shinfo(skb)->gso_size) != 0) { 3952 (mss = skb_shinfo(skb)->gso_size) != 0) {
3919 int tcp_opt_len, ip_tcp_len; 3953 int tcp_opt_len, ip_tcp_len, hdr_len;
3920 3954
3921 if (skb_header_cloned(skb) && 3955 if (skb_header_cloned(skb) &&
3922 pskb_expand_head(skb, 0, 0, GFP_ATOMIC)) { 3956 pskb_expand_head(skb, 0, 0, GFP_ATOMIC)) {
@@ -3927,11 +3961,16 @@ static int tg3_start_xmit_dma_bug(struct sk_buff *skb, struct net_device *dev)
3927 tcp_opt_len = ((skb->h.th->doff - 5) * 4); 3961 tcp_opt_len = ((skb->h.th->doff - 5) * 4);
3928 ip_tcp_len = (skb->nh.iph->ihl * 4) + sizeof(struct tcphdr); 3962 ip_tcp_len = (skb->nh.iph->ihl * 4) + sizeof(struct tcphdr);
3929 3963
3964 hdr_len = ip_tcp_len + tcp_opt_len;
3965 if (unlikely((ETH_HLEN + hdr_len) > 80) &&
3966 (tp->tg3_flags2 & TG3_FLG2_HW_TSO_1_BUG))
3967 return (tg3_tso_bug(tp, skb));
3968
3930 base_flags |= (TXD_FLAG_CPU_PRE_DMA | 3969 base_flags |= (TXD_FLAG_CPU_PRE_DMA |
3931 TXD_FLAG_CPU_POST_DMA); 3970 TXD_FLAG_CPU_POST_DMA);
3932 3971
3933 skb->nh.iph->check = 0; 3972 skb->nh.iph->check = 0;
3934 skb->nh.iph->tot_len = htons(mss + ip_tcp_len + tcp_opt_len); 3973 skb->nh.iph->tot_len = htons(mss + hdr_len);
3935 if (tp->tg3_flags2 & TG3_FLG2_HW_TSO) { 3974 if (tp->tg3_flags2 & TG3_FLG2_HW_TSO) {
3936 skb->h.th->check = 0; 3975 skb->h.th->check = 0;
3937 base_flags &= ~TXD_FLAG_TCPUDP_CSUM; 3976 base_flags &= ~TXD_FLAG_TCPUDP_CSUM;
@@ -10192,8 +10231,14 @@ static int __devinit tg3_get_invariants(struct tg3 *tp)
10192 GET_ASIC_REV(tp->pci_chip_rev_id) == ASIC_REV_5787) { 10231 GET_ASIC_REV(tp->pci_chip_rev_id) == ASIC_REV_5787) {
10193 tp->tg3_flags2 |= TG3_FLG2_HW_TSO_2; 10232 tp->tg3_flags2 |= TG3_FLG2_HW_TSO_2;
10194 tp->tg3_flags2 |= TG3_FLG2_1SHOT_MSI; 10233 tp->tg3_flags2 |= TG3_FLG2_1SHOT_MSI;
10195 } else 10234 } else {
10196 tp->tg3_flags2 |= TG3_FLG2_HW_TSO_1; 10235 tp->tg3_flags2 |= TG3_FLG2_HW_TSO_1 |
10236 TG3_FLG2_HW_TSO_1_BUG;
10237 if (GET_ASIC_REV(tp->pci_chip_rev_id) ==
10238 ASIC_REV_5750 &&
10239 tp->pci_chip_rev_id >= CHIPREV_ID_5750_C2)
10240 tp->tg3_flags2 &= ~TG3_FLG2_HW_TSO_1_BUG;
10241 }
10197 } 10242 }
10198 10243
10199 if (GET_ASIC_REV(tp->pci_chip_rev_id) != ASIC_REV_5705 && 10244 if (GET_ASIC_REV(tp->pci_chip_rev_id) != ASIC_REV_5705 &&