diff options
author | Eric Dumazet <edumazet@google.com> | 2015-10-13 20:12:54 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2015-10-13 21:26:34 -0400 |
commit | 4bdc3d66147b3a623b32216a45431d0cff005f50 (patch) | |
tree | 2e3f5a76b0e11c05cd1e9e88f5ab0c5cc1efe344 /net/ipv4/tcp_ipv4.c | |
parent | bbb300eb976b613a8e4e666d3af39f5ab1031d22 (diff) |
tcp/dccp: fix behavior of stale SYN_RECV request sockets
When a TCP/DCCP listener is closed, its pending SYN_RECV request sockets
become stale, meaning 3WHS can not complete.
But current behavior is wrong :
incoming packets finding such stale sockets are dropped.
We need instead to cleanup the request socket and perform another
lookup :
- Incoming ACK will give a RST answer,
- SYN rtx might find another listener if available.
- We expedite cleanup of request sockets and old listener socket.
Fixes: 079096f103fa ("tcp/dccp: install syn_recv requests into ehash table")
Signed-off-by: Eric Dumazet <edumazet@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4/tcp_ipv4.c')
-rw-r--r-- | net/ipv4/tcp_ipv4.c | 7 |
1 files changed, 6 insertions, 1 deletions
diff --git a/net/ipv4/tcp_ipv4.c b/net/ipv4/tcp_ipv4.c index ddb198392c7f..1ff0923df715 100644 --- a/net/ipv4/tcp_ipv4.c +++ b/net/ipv4/tcp_ipv4.c | |||
@@ -1572,6 +1572,7 @@ int tcp_v4_rcv(struct sk_buff *skb) | |||
1572 | TCP_SKB_CB(skb)->ip_dsfield = ipv4_get_dsfield(iph); | 1572 | TCP_SKB_CB(skb)->ip_dsfield = ipv4_get_dsfield(iph); |
1573 | TCP_SKB_CB(skb)->sacked = 0; | 1573 | TCP_SKB_CB(skb)->sacked = 0; |
1574 | 1574 | ||
1575 | lookup: | ||
1575 | sk = __inet_lookup_skb(&tcp_hashinfo, skb, th->source, th->dest); | 1576 | sk = __inet_lookup_skb(&tcp_hashinfo, skb, th->source, th->dest); |
1576 | if (!sk) | 1577 | if (!sk) |
1577 | goto no_tcp_socket; | 1578 | goto no_tcp_socket; |
@@ -1587,8 +1588,12 @@ process: | |||
1587 | sk = req->rsk_listener; | 1588 | sk = req->rsk_listener; |
1588 | if (tcp_v4_inbound_md5_hash(sk, skb)) | 1589 | if (tcp_v4_inbound_md5_hash(sk, skb)) |
1589 | goto discard_and_relse; | 1590 | goto discard_and_relse; |
1590 | if (sk->sk_state == TCP_LISTEN) | 1591 | if (likely(sk->sk_state == TCP_LISTEN)) { |
1591 | nsk = tcp_check_req(sk, skb, req, false); | 1592 | nsk = tcp_check_req(sk, skb, req, false); |
1593 | } else { | ||
1594 | inet_csk_reqsk_queue_drop(sk, req); | ||
1595 | goto lookup; | ||
1596 | } | ||
1592 | if (!nsk) { | 1597 | if (!nsk) { |
1593 | reqsk_put(req); | 1598 | reqsk_put(req); |
1594 | goto discard_it; | 1599 | goto discard_it; |