forked from mirrors/linux
		
	esp: Fix possible buffer overflow in ESP transformation
The maximum message size that can be send is bigger than the maximum site that skb_page_frag_refill can allocate. So it is possible to write beyond the allocated buffer. Fix this by doing a fallback to COW in that case. v2: Avoid get get_order() costs as suggested by Linus Torvalds. Fixes:cac2661c53("esp4: Avoid skb_cow_data whenever possible") Fixes:03e2a30f6a("esp6: Avoid skb_cow_data whenever possible") Reported-by: valis <sec@valis.email> Signed-off-by: Steffen Klassert <steffen.klassert@secunet.com>
This commit is contained in:
		
							parent
							
								
									a3d9001b4e
								
							
						
					
					
						commit
						ebe48d368e
					
				
					 3 changed files with 12 additions and 0 deletions
				
			
		|  | @ -4,6 +4,8 @@ | ||||||
| 
 | 
 | ||||||
| #include <linux/skbuff.h> | #include <linux/skbuff.h> | ||||||
| 
 | 
 | ||||||
|  | #define ESP_SKB_FRAG_MAXSIZE (PAGE_SIZE << SKB_FRAG_PAGE_ORDER) | ||||||
|  | 
 | ||||||
| struct ip_esp_hdr; | struct ip_esp_hdr; | ||||||
| 
 | 
 | ||||||
| static inline struct ip_esp_hdr *ip_esp_hdr(const struct sk_buff *skb) | static inline struct ip_esp_hdr *ip_esp_hdr(const struct sk_buff *skb) | ||||||
|  |  | ||||||
|  | @ -446,6 +446,7 @@ int esp_output_head(struct xfrm_state *x, struct sk_buff *skb, struct esp_info * | ||||||
| 	struct page *page; | 	struct page *page; | ||||||
| 	struct sk_buff *trailer; | 	struct sk_buff *trailer; | ||||||
| 	int tailen = esp->tailen; | 	int tailen = esp->tailen; | ||||||
|  | 	unsigned int allocsz; | ||||||
| 
 | 
 | ||||||
| 	/* this is non-NULL only with TCP/UDP Encapsulation */ | 	/* this is non-NULL only with TCP/UDP Encapsulation */ | ||||||
| 	if (x->encap) { | 	if (x->encap) { | ||||||
|  | @ -455,6 +456,10 @@ int esp_output_head(struct xfrm_state *x, struct sk_buff *skb, struct esp_info * | ||||||
| 			return err; | 			return err; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	allocsz = ALIGN(skb->data_len + tailen, L1_CACHE_BYTES); | ||||||
|  | 	if (allocsz > ESP_SKB_FRAG_MAXSIZE) | ||||||
|  | 		goto cow; | ||||||
|  | 
 | ||||||
| 	if (!skb_cloned(skb)) { | 	if (!skb_cloned(skb)) { | ||||||
| 		if (tailen <= skb_tailroom(skb)) { | 		if (tailen <= skb_tailroom(skb)) { | ||||||
| 			nfrags = 1; | 			nfrags = 1; | ||||||
|  |  | ||||||
|  | @ -482,6 +482,7 @@ int esp6_output_head(struct xfrm_state *x, struct sk_buff *skb, struct esp_info | ||||||
| 	struct page *page; | 	struct page *page; | ||||||
| 	struct sk_buff *trailer; | 	struct sk_buff *trailer; | ||||||
| 	int tailen = esp->tailen; | 	int tailen = esp->tailen; | ||||||
|  | 	unsigned int allocsz; | ||||||
| 
 | 
 | ||||||
| 	if (x->encap) { | 	if (x->encap) { | ||||||
| 		int err = esp6_output_encap(x, skb, esp); | 		int err = esp6_output_encap(x, skb, esp); | ||||||
|  | @ -490,6 +491,10 @@ int esp6_output_head(struct xfrm_state *x, struct sk_buff *skb, struct esp_info | ||||||
| 			return err; | 			return err; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	allocsz = ALIGN(skb->data_len + tailen, L1_CACHE_BYTES); | ||||||
|  | 	if (allocsz > ESP_SKB_FRAG_MAXSIZE) | ||||||
|  | 		goto cow; | ||||||
|  | 
 | ||||||
| 	if (!skb_cloned(skb)) { | 	if (!skb_cloned(skb)) { | ||||||
| 		if (tailen <= skb_tailroom(skb)) { | 		if (tailen <= skb_tailroom(skb)) { | ||||||
| 			nfrags = 1; | 			nfrags = 1; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Steffen Klassert
						Steffen Klassert