aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorFugang Duan <b38611@freescale.com>2014-06-17 20:13:06 -0400
committerFugang Duan <b38611@freescale.com>2014-06-19 05:07:08 -0400
commitc74bb381e1a208096723073e1f3509846f6f8201 (patch)
tree9342cb83ae5f4cadf2865e23d1895ecef4ad276a
parent8e98d50d406301396fba80a7f0cb4d01a0ba79ec (diff)
net: fec: Don't clear IPV6 header checksum field when IP accelerator enable
The commit 96c50caa5148 (net: fec: Enable IP header hardware checksum) enable HW IP header checksum for IPV4 and IPV6, which causes IPV6 TCP/UDP cannot work. (The issue is reported by Russell King) For FEC IP header checksum function: Insert IP header checksum. This "IINS" bit is written by the user. If set, IP accelerator calculates the IP header checksum and overwrites the IINS corresponding header field with the calculated value. The checksum field must be cleared by user, otherwise the checksum always is 0xFFFF. So the previous patch clear IP header checksum field regardless of IP frame type. In fact, IP HW detect the packet as IPV6 type, even if the "IINS" bit is set, the IP accelerator is not triggered to calculates IPV6 header checksum because IPV6 frame format don't have checksum. So this results in the IPV6 frame being corrupted. The patch just add software detect the current packet type, if it is IPV6 frame, it don't clear IP header checksum field. Cc: Russell King <linux@arm.linux.org.uk> Reported-and-tested-by: Russell King <linux@arm.linux.org.uk> Signed-off-by: Fugang Duan <B38611@freescale.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--drivers/net/ethernet/freescale/fec_main.c8
1 files changed, 7 insertions, 1 deletions
diff --git a/drivers/net/ethernet/freescale/fec_main.c b/drivers/net/ethernet/freescale/fec_main.c
index e48bcbcd36e0..4a5a06eb3e03 100644
--- a/drivers/net/ethernet/freescale/fec_main.c
+++ b/drivers/net/ethernet/freescale/fec_main.c
@@ -310,6 +310,11 @@ static void *swap_buffer(void *bufaddr, int len)
310 return bufaddr; 310 return bufaddr;
311} 311}
312 312
313static inline bool is_ipv4_pkt(struct sk_buff *skb)
314{
315 return skb->protocol == htons(ETH_P_IP) && ip_hdr(skb)->version == 4;
316}
317
313static int 318static int
314fec_enet_clear_csum(struct sk_buff *skb, struct net_device *ndev) 319fec_enet_clear_csum(struct sk_buff *skb, struct net_device *ndev)
315{ 320{
@@ -320,7 +325,8 @@ fec_enet_clear_csum(struct sk_buff *skb, struct net_device *ndev)
320 if (unlikely(skb_cow_head(skb, 0))) 325 if (unlikely(skb_cow_head(skb, 0)))
321 return -1; 326 return -1;
322 327
323 ip_hdr(skb)->check = 0; 328 if (is_ipv4_pkt(skb))
329 ip_hdr(skb)->check = 0;
324 *(__sum16 *)(skb->head + skb->csum_start + skb->csum_offset) = 0; 330 *(__sum16 *)(skb->head + skb->csum_start + skb->csum_offset) = 0;
325 331
326 return 0; 332 return 0;