]> git.baikalelectronics.ru Git - kernel.git/commit
tcp: fix SO_RCVLOWAT related hangs under mem pressure
authorEric Dumazet <edumazet@google.com>
Fri, 12 Feb 2021 23:22:13 +0000 (15:22 -0800)
committerDavid S. Miller <davem@davemloft.net>
Sat, 13 Feb 2021 01:28:26 +0000 (17:28 -0800)
commit04e56677827a54eb036acc978504f0d2f89e1d91
tree7b99079b3e81b9a115eab659a64cee178cd8c309
parent906a1078db0ac918d8b8aaffbf8ddd036ab151ed
tcp: fix SO_RCVLOWAT related hangs under mem pressure

While commit 9e28c970ee8f ("tcp: fix SO_RCVLOWAT hangs with fat skbs")
fixed an issue vs too small sk_rcvbuf for given sk_rcvlowat constraint,
it missed to address issue caused by memory pressure.

1) If we are under memory pressure and socket receive queue is empty.
First incoming packet is allowed to be queued, after commit
76672cf0b8c2 ("tcp: allow one skb to be received per socket under memory pressure")

But we do not send EPOLLIN yet, in case tcp_data_ready() sees sk_rcvlowat
is bigger than skb length.

2) Then, when next packet comes, it is dropped, and we directly
call sk->sk_data_ready().

3) If application is using poll(), tcp_poll() will then use
tcp_stream_is_readable() and decide the socket receive queue is
not yet filled, so nothing will happen.

Even when sender retransmits packets, phases 2) & 3) repeat
and flow is effectively frozen, until memory pressure is off.

Fix is to consider tcp_under_memory_pressure() to take care
of global memory pressure or memcg pressure.

Fixes: 9e28c970ee8f ("tcp: fix SO_RCVLOWAT hangs with fat skbs")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Reported-by: Arjun Roy <arjunroy@google.com>
Suggested-by: Wei Wang <weiwan@google.com>
Reviewed-by: Wei Wang <weiwan@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/net/tcp.h