]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
ipv6: sit: reload inner IPv6 header after GSO offloads
authorKyle Zeng <kylebot@openai.com>
Fri, 5 Jun 2026 07:34:48 +0000 (00:34 -0700)
committerJakub Kicinski <kuba@kernel.org>
Tue, 9 Jun 2026 02:03:56 +0000 (19:03 -0700)
ipip6_tunnel_xmit() caches the inner IPv6 header pointer at function
entry and continues using it after iptunnel_handle_offloads().

For GSO skbs, iptunnel_handle_offloads() calls skb_header_unclone().
When the skb header is cloned, skb_header_unclone() can call
pskb_expand_head(), which may move the skb head. The pskb_expand_head()
contract requires pointers into the skb header to be reloaded after the
call.

If the later skb_realloc_headroom() branch is not taken, SIT uses the
stale iph6 pointer to read the inner hop limit and DS field. That can
read from a freed skb head after the old head's remaining clone is
released.

Reload iph6 after the offload helper succeeds and before subsequent
reads from the inner IPv6 header. Keep the existing reload after
skb_realloc_headroom(), since that branch can also replace the skb.

Fixes: 14909664e4e1 ("sit: Setup and TX path for sit/UDP foo-over-udp encapsulation")
Signed-off-by: Kyle Zeng <kylebot@openai.com>
Reviewed-by: Eric Dumazet <edumazet@google.com>
Reported-by: syzbot+6eb9ca986d80f6f88cf9@syzkaller.appspotmail.com
Link: https://patch.msgid.link/20260605073448.6524-1-kylebot@openai.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/ipv6/sit.c

index 201347b4e12742451de6d6036281d1abaff0dd84..b41e231a669bec99cd15affcd85e66be328a281d 100644 (file)
@@ -961,6 +961,7 @@ static netdev_tx_t ipip6_tunnel_xmit(struct sk_buff *skb,
                ip_rt_put(rt);
                goto tx_error;
        }
+       iph6 = ipv6_hdr(skb);
 
        if (df) {
                mtu = dst4_mtu(&rt->dst) - t_hlen;