esp: Fix possible buffer overflow in ESP transformation
commit ebe48d368e97d007bfeb76fcb065d6cfc4c96645 upstream. 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> Signed-off-by: Tadeusz Struk <tadeusz.struk@linaro.org> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:

committed by
Greg Kroah-Hartman

parent
96340cdd55
commit
9248694dac
@@ -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)
|
||||||
|
@@ -2670,6 +2670,7 @@ extern int sysctl_optmem_max;
|
|||||||
extern __u32 sysctl_wmem_default;
|
extern __u32 sysctl_wmem_default;
|
||||||
extern __u32 sysctl_rmem_default;
|
extern __u32 sysctl_rmem_default;
|
||||||
|
|
||||||
|
#define SKB_FRAG_PAGE_ORDER get_order(32768)
|
||||||
DECLARE_STATIC_KEY_FALSE(net_high_order_alloc_disable_key);
|
DECLARE_STATIC_KEY_FALSE(net_high_order_alloc_disable_key);
|
||||||
|
|
||||||
static inline int sk_get_wmem0(const struct sock *sk, const struct proto *proto)
|
static inline int sk_get_wmem0(const struct sock *sk, const struct proto *proto)
|
||||||
|
@@ -448,6 +448,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) {
|
||||||
@@ -457,6 +458,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;
|
||||||
|
@@ -483,6 +483,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);
|
||||||
@@ -491,6 +492,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;
|
||||||
|
Reference in New Issue
Block a user