diff options
Diffstat (limited to 'drivers/net/tg3.c')
-rw-r--r-- | drivers/net/tg3.c | 53 |
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 | ||
3884 | static 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 | */ | ||
3889 | static 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 | |||
3910 | tg3_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 && |