Commit ed4cccef authored by Antoine Tenart's avatar Antoine Tenart Committed by David S. Miller

gro: fix ownership transfer

If packets are GROed with fraglist they might be segmented later on and
continue their journey in the stack. In skb_segment_list those skbs can
be reused as-is. This is an issue as their destructor was removed in
skb_gro_receive_list but not the reference to their socket, and then
they can't be orphaned. Fix this by also removing the reference to the
socket.

For example this could be observed,

  kernel BUG at include/linux/skbuff.h:3131!  (skb_orphan)
  RIP: 0010:ip6_rcv_core+0x11bc/0x19a0
  Call Trace:
   ipv6_list_rcv+0x250/0x3f0
   __netif_receive_skb_list_core+0x49d/0x8f0
   netif_receive_skb_list_internal+0x634/0xd40
   napi_complete_done+0x1d2/0x7d0
   gro_cell_poll+0x118/0x1f0

A similar construction is found in skb_gro_receive, apply the same
change there.

Fixes: 5e10da53 ("skbuff: allow 'slow_gro' for skb carring sock reference")
Signed-off-by: default avatarAntoine Tenart <atenart@kernel.org>
Reviewed-by: default avatarWillem de Bruijn <willemb@google.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 3d010c80
...@@ -192,8 +192,9 @@ int skb_gro_receive(struct sk_buff *p, struct sk_buff *skb) ...@@ -192,8 +192,9 @@ int skb_gro_receive(struct sk_buff *p, struct sk_buff *skb)
} }
merge: merge:
/* sk owenrship - if any - completely transferred to the aggregated packet */ /* sk ownership - if any - completely transferred to the aggregated packet */
skb->destructor = NULL; skb->destructor = NULL;
skb->sk = NULL;
delta_truesize = skb->truesize; delta_truesize = skb->truesize;
if (offset > headlen) { if (offset > headlen) {
unsigned int eat = offset - headlen; unsigned int eat = offset - headlen;
......
...@@ -449,8 +449,9 @@ static int skb_gro_receive_list(struct sk_buff *p, struct sk_buff *skb) ...@@ -449,8 +449,9 @@ static int skb_gro_receive_list(struct sk_buff *p, struct sk_buff *skb)
NAPI_GRO_CB(p)->count++; NAPI_GRO_CB(p)->count++;
p->data_len += skb->len; p->data_len += skb->len;
/* sk owenrship - if any - completely transferred to the aggregated packet */ /* sk ownership - if any - completely transferred to the aggregated packet */
skb->destructor = NULL; skb->destructor = NULL;
skb->sk = NULL;
p->truesize += skb->truesize; p->truesize += skb->truesize;
p->len += skb->len; p->len += skb->len;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment