]> git.baikalelectronics.ru Git - kernel.git/commit
tcp: fix tcp_mtu_probe() vs highest_sack
authorEric Dumazet <edumazet@google.com>
Tue, 31 Oct 2017 06:08:20 +0000 (23:08 -0700)
committerDavid S. Miller <davem@davemloft.net>
Wed, 1 Nov 2017 12:18:34 +0000 (21:18 +0900)
commit25b64fa727396402017fe8c13ce32f0dd2e58d0e
tree5e6be10ad3727cadc7da624a4c10b7660f5bf74c
parent930a5c6901440cbc4d491d4fcb044c00f8512c31
tcp: fix tcp_mtu_probe() vs highest_sack

Based on SNMP values provided by Roman, Yuchung made the observation
that some crashes in tcp_sacktag_walk() might be caused by MTU probing.

Looking at tcp_mtu_probe(), I found that when a new skb was placed
in front of the write queue, we were not updating tcp highest sack.

If one skb is freed because all its content was copied to the new skb
(for MTU probing), then tp->highest_sack could point to a now freed skb.

Bad things would then happen, including infinite loops.

This patch renames tcp_highest_sack_combine() and uses it
from tcp_mtu_probe() to fix the bug.

Note that I also removed one test against tp->sacked_out,
since we want to replace tp->highest_sack regardless of whatever
condition, since keeping a stale pointer to freed skb is a recipe
for disaster.

Fixes: f4f2f47b0164 ("[TCP]: Convert highest_sack to sk_buff to allow direct access")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Reported-by: Alexei Starovoitov <alexei.starovoitov@gmail.com>
Reported-by: Roman Gushchin <guro@fb.com>
Reported-by: Oleksandr Natalenko <oleksandr@natalenko.name>
Acked-by: Alexei Starovoitov <ast@kernel.org>
Acked-by: Neal Cardwell <ncardwell@google.com>
Acked-by: Yuchung Cheng <ycheng@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/net/tcp.h
net/ipv4/tcp_output.c