mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	i40e/i40evf: Enable support for SKB_GSO_UDP_TUNNEL_CSUM
The XL722 has support for providing the outer UDP tunnel checksum on transmits. Make use of this feature to support segmenting UDP tunnels with outer checksums enabled. Signed-off-by: Alexander Duyck <aduyck@mirantis.com> Tested-by: Andrew Bowers <andrewx.bowers@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
This commit is contained in:
		
							parent
							
								
									fad57330b6
								
							
						
					
					
						commit
						5453205cd0
					
				| @ -2272,6 +2272,7 @@ static int i40e_tso(struct i40e_ring *tx_ring, struct sk_buff *skb, | |||||||
| 	} ip; | 	} ip; | ||||||
| 	union { | 	union { | ||||||
| 		struct tcphdr *tcp; | 		struct tcphdr *tcp; | ||||||
|  | 		struct udphdr *udp; | ||||||
| 		unsigned char *hdr; | 		unsigned char *hdr; | ||||||
| 	} l4; | 	} l4; | ||||||
| 	u32 paylen, l4_offset; | 	u32 paylen, l4_offset; | ||||||
| @ -2298,7 +2299,18 @@ static int i40e_tso(struct i40e_ring *tx_ring, struct sk_buff *skb, | |||||||
| 		ip.v6->payload_len = 0; | 		ip.v6->payload_len = 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (skb_shinfo(skb)->gso_type & (SKB_GSO_UDP_TUNNEL | SKB_GSO_GRE)) { | 	if (skb_shinfo(skb)->gso_type & (SKB_GSO_UDP_TUNNEL | SKB_GSO_GRE | | ||||||
|  | 					 SKB_GSO_UDP_TUNNEL_CSUM)) { | ||||||
|  | 		if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM) { | ||||||
|  | 			/* determine offset of outer transport header */ | ||||||
|  | 			l4_offset = l4.hdr - skb->data; | ||||||
|  | 
 | ||||||
|  | 			/* remove payload length from outer checksum */ | ||||||
|  | 			paylen = (__force u16)l4.udp->check; | ||||||
|  | 			paylen += ntohs(1) * (u16)~(skb->len - l4_offset); | ||||||
|  | 			l4.udp->check = ~csum_fold((__force __wsum)paylen); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		/* reset pointers to inner headers */ | 		/* reset pointers to inner headers */ | ||||||
| 		ip.hdr = skb_inner_network_header(skb); | 		ip.hdr = skb_inner_network_header(skb); | ||||||
| 		l4.hdr = skb_inner_transport_header(skb); | 		l4.hdr = skb_inner_transport_header(skb); | ||||||
| @ -2460,6 +2472,11 @@ static int i40e_tx_enable_csum(struct sk_buff *skb, u32 *tx_flags, | |||||||
| 		tunnel |= ((ip.hdr - l4.hdr) / 2) << | 		tunnel |= ((ip.hdr - l4.hdr) / 2) << | ||||||
| 			  I40E_TXD_CTX_QW0_NATLEN_SHIFT; | 			  I40E_TXD_CTX_QW0_NATLEN_SHIFT; | ||||||
| 
 | 
 | ||||||
|  | 		/* indicate if we need to offload outer UDP header */ | ||||||
|  | 		if ((*tx_flags & I40E_TX_FLAGS_TSO) && | ||||||
|  | 		    (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM)) | ||||||
|  | 			tunnel |= I40E_TXD_CTX_QW0_L4T_CS_MASK; | ||||||
|  | 
 | ||||||
| 		/* record tunnel offload values */ | 		/* record tunnel offload values */ | ||||||
| 		*cd_tunneling |= tunnel; | 		*cd_tunneling |= tunnel; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1532,6 +1532,7 @@ static int i40e_tso(struct i40e_ring *tx_ring, struct sk_buff *skb, | |||||||
| 	} ip; | 	} ip; | ||||||
| 	union { | 	union { | ||||||
| 		struct tcphdr *tcp; | 		struct tcphdr *tcp; | ||||||
|  | 		struct udphdr *udp; | ||||||
| 		unsigned char *hdr; | 		unsigned char *hdr; | ||||||
| 	} l4; | 	} l4; | ||||||
| 	u32 paylen, l4_offset; | 	u32 paylen, l4_offset; | ||||||
| @ -1558,7 +1559,18 @@ static int i40e_tso(struct i40e_ring *tx_ring, struct sk_buff *skb, | |||||||
| 		ip.v6->payload_len = 0; | 		ip.v6->payload_len = 0; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if (skb_shinfo(skb)->gso_type & (SKB_GSO_UDP_TUNNEL | SKB_GSO_GRE)) { | 	if (skb_shinfo(skb)->gso_type & (SKB_GSO_UDP_TUNNEL | SKB_GSO_GRE | | ||||||
|  | 					 SKB_GSO_UDP_TUNNEL_CSUM)) { | ||||||
|  | 		if (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM) { | ||||||
|  | 			/* determine offset of outer transport header */ | ||||||
|  | 			l4_offset = l4.hdr - skb->data; | ||||||
|  | 
 | ||||||
|  | 			/* remove payload length from outer checksum */ | ||||||
|  | 			paylen = (__force u16)l4.udp->check; | ||||||
|  | 			paylen += ntohs(1) * (u16)~(skb->len - l4_offset); | ||||||
|  | 			l4.udp->check = ~csum_fold((__force __wsum)paylen); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
| 		/* reset pointers to inner headers */ | 		/* reset pointers to inner headers */ | ||||||
| 		ip.hdr = skb_inner_network_header(skb); | 		ip.hdr = skb_inner_network_header(skb); | ||||||
| 		l4.hdr = skb_inner_transport_header(skb); | 		l4.hdr = skb_inner_transport_header(skb); | ||||||
| @ -1678,6 +1690,11 @@ static int i40e_tx_enable_csum(struct sk_buff *skb, u32 *tx_flags, | |||||||
| 		tunnel |= ((ip.hdr - l4.hdr) / 2) << | 		tunnel |= ((ip.hdr - l4.hdr) / 2) << | ||||||
| 			  I40E_TXD_CTX_QW0_NATLEN_SHIFT; | 			  I40E_TXD_CTX_QW0_NATLEN_SHIFT; | ||||||
| 
 | 
 | ||||||
|  | 		/* indicate if we need to offload outer UDP header */ | ||||||
|  | 		if ((*tx_flags & I40E_TX_FLAGS_TSO) && | ||||||
|  | 		    (skb_shinfo(skb)->gso_type & SKB_GSO_UDP_TUNNEL_CSUM)) | ||||||
|  | 			tunnel |= I40E_TXD_CTX_QW0_L4T_CS_MASK; | ||||||
|  | 
 | ||||||
| 		/* record tunnel offload values */ | 		/* record tunnel offload values */ | ||||||
| 		*cd_tunneling |= tunnel; | 		*cd_tunneling |= tunnel; | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Alexander Duyck
						Alexander Duyck