From patchwork Fri Oct 23 11:13:52 2020 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Ricardo Dias X-Patchwork-Id: 298642 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-9.8 required=3.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID, DKIM_VALID_AU, HEADER_FROM_DIFFERENT_DOMAINS, INCLUDES_PATCH, MAILING_LIST_MULTI, SIGNED_OFF_BY, SPF_HELO_NONE, SPF_PASS, URIBL_BLOCKED autolearn=ham autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id E1472C388F9 for ; Fri, 23 Oct 2020 11:14:04 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by mail.kernel.org (Postfix) with ESMTP id 69FEE2192A for ; Fri, 23 Oct 2020 11:14:04 +0000 (UTC) Authentication-Results: mail.kernel.org; dkim=pass (1024-bit key) header.d=memsql.com header.i=@memsql.com header.b="N6PdIeG6" Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S462818AbgJWLOE (ORCPT ); Fri, 23 Oct 2020 07:14:04 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:55556 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S462811AbgJWLOD (ORCPT ); Fri, 23 Oct 2020 07:14:03 -0400 Received: from mail-wm1-x336.google.com (mail-wm1-x336.google.com [IPv6:2a00:1450:4864:20::336]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 85A0EC0613CE for ; Fri, 23 Oct 2020 04:14:01 -0700 (PDT) Received: by mail-wm1-x336.google.com with SMTP id k21so889745wmi.1 for ; Fri, 23 Oct 2020 04:14:01 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=memsql.com; s=google; h=date:from:to:cc:subject:message-id:mime-version:content-disposition; bh=GF9B6SYLRzLidzBBztBZX6WXKZIvIfWHBORPObBW6Xg=; b=N6PdIeG6ldrqc6d0LiMR+wuU6XlqBbBv958Vkf8bbgXEA10mpsxSx5YqP0OnCmjtoN gaa+cfAwZsHCBd4q22ghwCKsjMNTWFl6EsuKlHafyCh4JBOpGFVssHoJ0iiRQE/RHLLB LvwaSubDIaALd8pfwP4q1x8P063eilFg3TAwg= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:message-id:mime-version :content-disposition; bh=GF9B6SYLRzLidzBBztBZX6WXKZIvIfWHBORPObBW6Xg=; b=rpR1xIvtDSGwPruVWV2ZVPqPNUWDEViEzWpZuTheYZfk6CvodB+7UvahOvcEnKd9Lb b3AA0MWeu6PCbt9iyvaeE2MsU8nbLVh0XLT1I4cdw8crnbtsVnXgljZW8JeI6O5gW5HK r77xaZBucMp80UlJgNSsI50QvVXIxp590hYGTVixPrEo32phmG6xlPfYWHZoM1I1kvmH r009eJnLIvZ8ZQ3ZyT0CVkm3lMfDLGnODN2IrR8b17GwMlSmK+SY6jPN294+bhhzRoqt cT7ALn5DxltEcS8WY5JfxLGBV5EP6Ak6Tb+ra5AxXmsWXgteTcD50ePKRf66t0vhieAc x/RQ== X-Gm-Message-State: AOAM531YTK3DBNu7ARrFAhaffV6u85jwiq3ESyC41DWq/p+y6qaJMaug eMo6Ivu7masxE0ycCugbkVTjqeUfUjm+mOBeOvo= X-Google-Smtp-Source: ABdhPJzFKRrv0FYAom527u+/LPePhF6kldFDrU2pQjDJZ93oTplTJEtCzAh4XGaOwSbIJ1zZ1bgIFg== X-Received: by 2002:a7b:c957:: with SMTP id i23mr743086wml.155.1603451640129; Fri, 23 Oct 2020 04:14:00 -0700 (PDT) Received: from rdias-suse-pc.lan (bl13-26-148.dsl.telepac.pt. [85.246.26.148]) by smtp.gmail.com with ESMTPSA id o129sm2452861wmb.25.2020.10.23.04.13.59 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Fri, 23 Oct 2020 04:13:59 -0700 (PDT) Date: Fri, 23 Oct 2020 12:13:52 +0100 From: Ricardo Dias To: davem@davemloft.net, kuba@kernel.org, kuznet@ms2.inr.ac.ru, yoshfuji@linux-ipv6.org, edumazet@google.com Cc: netdev@vger.kernel.org, linux-kernel@vger.kernel.org Subject: [PATCH] tcp: fix race condition when creating child sockets from syncookies Message-ID: <20201023111352.GA289522@rdias-suse-pc.lan> MIME-Version: 1.0 Content-Disposition: inline Precedence: bulk List-ID: X-Mailing-List: netdev@vger.kernel.org When the TCP stack is in SYN flood mode, the server child socket is created from the SYN cookie received in a TCP packet with the ACK flag set. The child socket is created when the server receives the first TCP packet with a valid SYN cookie from the client. Usually, this packet corresponds to the final step of the TCP 3-way handshake, the ACK packet. But is also possible to receive a valid SYN cookie from the first TCP data packet sent by the client, and thus create a child socket from that SYN cookie. Since a client socket is ready to send data as soon as it receives the SYN+ACK packet from the server, the client can send the ACK packet (sent by the TCP stack code), and the first data packet (sent by the userspace program) almost at the same time, and thus the server will equally receive the two TCP packets with valid SYN cookies almost at the same instant. When such event happens, the TCP stack code has a race condition that occurs between the momement a lookup is done to the established connections hashtable to check for the existence of a connection for the same client, and the moment that the child socket is added to the established connections hashtable. As a consequence, this race condition can lead to a situation where we add two child sockets to the established connections hashtable and deliver two sockets to the userspace program to the same client. This patch fixes the race condition by checking if an existing child socket exists for the same client when we are adding the second child socket to the established connections socket. If an existing child socket exists, we return that socket and use it to process the TCP packet received, and discard the second child socket to the same client. Signed-off-by: Ricardo Dias Reported-by: kernel test robot Reported-by: kernel test robot Reported-by: Dan Carpenter --- include/net/inet_hashtables.h | 1 + net/ipv4/inet_hashtables.c | 59 +++++++++++++++++++++++++++++++++++ net/ipv4/syncookies.c | 5 ++- net/ipv4/tcp_ipv4.c | 16 +++++++++- 4 files changed, 79 insertions(+), 2 deletions(-) diff --git a/include/net/inet_hashtables.h b/include/net/inet_hashtables.h index 92560974ea67..e1d7fd20468a 100644 --- a/include/net/inet_hashtables.h +++ b/include/net/inet_hashtables.h @@ -248,6 +248,7 @@ void inet_hashinfo2_init(struct inet_hashinfo *h, const char *name, int inet_hashinfo2_init_mod(struct inet_hashinfo *h); bool inet_ehash_insert(struct sock *sk, struct sock *osk); +struct sock *inet_ehash_insert_chk_dup(struct sock *sk); bool inet_ehash_nolisten(struct sock *sk, struct sock *osk); int __inet_hash(struct sock *sk, struct sock *osk); int inet_hash(struct sock *sk); diff --git a/net/ipv4/inet_hashtables.c b/net/ipv4/inet_hashtables.c index 239e54474b65..5dbe3aa291e6 100644 --- a/net/ipv4/inet_hashtables.c +++ b/net/ipv4/inet_hashtables.c @@ -539,6 +539,65 @@ bool inet_ehash_insert(struct sock *sk, struct sock *osk) return ret; } +/* Inserts a socket into ehash if no existing socket exists for the same + * quadruple (saddr, sport, daddr, dport). + * If there is an existing socket, returns that socket, otherwise returns NULL. + */ +struct sock *inet_ehash_insert_chk_dup(struct sock *sk) +{ + struct inet_hashinfo *hashinfo = sk->sk_prot->h.hashinfo; + struct hlist_nulls_head *list; + struct inet_ehash_bucket *head; + const struct hlist_nulls_node *node; + struct sock *esk; + spinlock_t *lock; /* protects hashinfo socket entry */ + struct net *net = sock_net(sk); + const int dif, sdif = sk->sk_bound_dev_if; + + INET_ADDR_COOKIE(acookie, sk->sk_daddr, sk->sk_rcv_saddr); + const __portpair ports = INET_COMBINED_PORTS(sk->sk_dport, sk->sk_num); + + WARN_ON_ONCE(!sk_unhashed(sk)); + + sk->sk_hash = sk_ehashfn(sk); + head = inet_ehash_bucket(hashinfo, sk->sk_hash); + list = &head->chain; + lock = inet_ehash_lockp(hashinfo, sk->sk_hash); + + spin_lock(lock); +begin: + sk_nulls_for_each_rcu(esk, node, list) { + if (esk->sk_hash != sk->sk_hash) + continue; + if (likely(INET_MATCH(esk, net, acookie, + sk->sk_daddr, sk->sk_rcv_saddr, ports, + dif, sdif))) { + if (unlikely(!refcount_inc_not_zero(&esk->sk_refcnt))) + goto out; + if (unlikely(!INET_MATCH(esk, net, acookie, + sk->sk_daddr, + sk->sk_rcv_saddr, ports, + dif, sdif))) { + sock_gen_put(esk); + goto begin; + } + goto found; + } + } +out: + esk = NULL; + __sk_nulls_add_node_rcu(sk, list); +found: + spin_unlock(lock); + if (esk) { + percpu_counter_inc(sk->sk_prot->orphan_count); + inet_sk_set_state(sk, TCP_CLOSE); + sock_set_flag(sk, SOCK_DEAD); + inet_csk_destroy_sock(sk); + } + return esk; +} + bool inet_ehash_nolisten(struct sock *sk, struct sock *osk) { bool ok = inet_ehash_insert(sk, osk); diff --git a/net/ipv4/syncookies.c b/net/ipv4/syncookies.c index e03756631541..c4bb895085f0 100644 --- a/net/ipv4/syncookies.c +++ b/net/ipv4/syncookies.c @@ -208,7 +208,7 @@ struct sock *tcp_get_cookie_sock(struct sock *sk, struct sk_buff *skb, child = icsk->icsk_af_ops->syn_recv_sock(sk, skb, req, dst, NULL, &own_req); - if (child) { + if (child && own_req) { refcount_set(&req->rsk_refcnt, 1); tcp_sk(child)->tsoffset = tsoff; sock_rps_save_rxhash(child, skb); @@ -223,6 +223,9 @@ struct sock *tcp_get_cookie_sock(struct sock *sk, struct sk_buff *skb, bh_unlock_sock(child); sock_put(child); + } else if (child && !own_req) { + __reqsk_free(req); + return child; } __reqsk_free(req); diff --git a/net/ipv4/tcp_ipv4.c b/net/ipv4/tcp_ipv4.c index 592c73962723..c705d335bd80 100644 --- a/net/ipv4/tcp_ipv4.c +++ b/net/ipv4/tcp_ipv4.c @@ -1501,6 +1501,8 @@ struct sock *tcp_v4_syn_recv_sock(const struct sock *sk, struct sk_buff *skb, int l3index; #endif struct ip_options_rcu *inet_opt; + struct sock *esk = NULL; + bool syncookie = false; if (sk_acceptq_is_full(sk)) goto exit_overflow; @@ -1535,6 +1537,7 @@ struct sock *tcp_v4_syn_recv_sock(const struct sock *sk, struct sk_buff *skb, goto put_and_exit; } else { /* syncookie case : see end of cookie_v4_check() */ + syncookie = true; } sk_setup_caps(newsk, dst); @@ -1565,7 +1568,18 @@ struct sock *tcp_v4_syn_recv_sock(const struct sock *sk, struct sk_buff *skb, if (__inet_inherit_port(sk, newsk) < 0) goto put_and_exit; - *own_req = inet_ehash_nolisten(newsk, req_to_sk(req_unhash)); + if (!syncookie) { + *own_req = inet_ehash_nolisten(newsk, req_to_sk(req_unhash)); + } else { + esk = inet_ehash_insert_chk_dup(newsk); + /* We're going to notify tcp_get_cookie_sock through own_req + * that an existing socket is going to be returned instead, + * since own_req is not used by the syncookies code. + */ + *own_req = !esk; + if (esk) + newsk = esk; + } if (likely(*own_req)) { tcp_move_syn(newtp, req); ireq->ireq_opt = NULL;