diff options
Diffstat (limited to 'net/core/gro.c')
| -rw-r--r-- | net/core/gro.c | 31 | 
1 files changed, 28 insertions, 3 deletions
| diff --git a/net/core/gro.c b/net/core/gro.c index c7901253a1a8..b3b43de1a650 100644 --- a/net/core/gro.c +++ b/net/core/gro.c @@ -3,6 +3,7 @@  #include <net/dst_metadata.h>  #include <net/busy_poll.h>  #include <trace/events/net.h> +#include <linux/skbuff_ref.h>  #define MAX_GRO_SKBS 8 @@ -230,6 +231,33 @@ done:  	return 0;  } +int skb_gro_receive_list(struct sk_buff *p, struct sk_buff *skb) +{ +	if (unlikely(p->len + skb->len >= 65536)) +		return -E2BIG; + +	if (NAPI_GRO_CB(p)->last == p) +		skb_shinfo(p)->frag_list = skb; +	else +		NAPI_GRO_CB(p)->last->next = skb; + +	skb_pull(skb, skb_gro_offset(skb)); + +	NAPI_GRO_CB(p)->last = skb; +	NAPI_GRO_CB(p)->count++; +	p->data_len += skb->len; + +	/* sk ownership - if any - completely transferred to the aggregated packet */ +	skb->destructor = NULL; +	skb->sk = NULL; +	p->truesize += skb->truesize; +	p->len += skb->len; + +	NAPI_GRO_CB(skb)->same_flow = 1; + +	return 0; +} +  static void napi_gro_complete(struct napi_struct *napi, struct sk_buff *skb)  { @@ -330,8 +358,6 @@ static void gro_list_prepare(const struct list_head *head,  	list_for_each_entry(p, head, list) {  		unsigned long diffs; -		NAPI_GRO_CB(p)->flush = 0; -  		if (hash != skb_get_hash_raw(p)) {  			NAPI_GRO_CB(p)->same_flow = 0;  			continue; @@ -471,7 +497,6 @@ found_ptype:  					sizeof(u32))); /* Avoid slow unaligned acc */  	*(u32 *)&NAPI_GRO_CB(skb)->zeroed = 0;  	NAPI_GRO_CB(skb)->flush = skb_has_frag_list(skb); -	NAPI_GRO_CB(skb)->is_atomic = 1;  	NAPI_GRO_CB(skb)->count = 1;  	if (unlikely(skb_is_gso(skb))) {  		NAPI_GRO_CB(skb)->count = skb_shinfo(skb)->gso_segs; | 
