ipv4: annotate data races arount inet->min_ttl
authorEric Dumazet <edumazet@google.com>
Mon, 25 Oct 2021 16:48:23 +0000 (09:48 -0700)
committerJakub Kicinski <kuba@kernel.org>
Tue, 26 Oct 2021 01:02:13 +0000 (18:02 -0700)
No report yet from KCSAN, yet worth documenting the races.

Signed-off-by: Eric Dumazet <edumazet@google.com>
Acked-by: Soheil Hassas Yeganeh <soheil@google.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/ipv4/ip_sockglue.c
net/ipv4/tcp_ipv4.c

index b297bb28556ec5cf383068f67ee910af38591cc3..d5487c8580674a01df8c7d8ce88f97c9add846b6 100644 (file)
@@ -1352,7 +1352,10 @@ static int do_ip_setsockopt(struct sock *sk, int level, int optname,
                        goto e_inval;
                if (val < 0 || val > 255)
                        goto e_inval;
-               inet->min_ttl = val;
+               /* tcp_v4_err() and tcp_v4_rcv() might read min_ttl
+                * while we are changint it.
+                */
+               WRITE_ONCE(inet->min_ttl, val);
                break;
 
        default:
index 2bdc32c1afb65bb123a27444d9f6e4d01a188074..a9cbc8e6b796207f4880b2b32ff9289321080068 100644 (file)
@@ -508,7 +508,8 @@ int tcp_v4_err(struct sk_buff *skb, u32 info)
        if (sk->sk_state == TCP_CLOSE)
                goto out;
 
-       if (unlikely(iph->ttl < inet_sk(sk)->min_ttl)) {
+       /* min_ttl can be changed concurrently from do_ip_setsockopt() */
+       if (unlikely(iph->ttl < READ_ONCE(inet_sk(sk)->min_ttl))) {
                __NET_INC_STATS(net, LINUX_MIB_TCPMINTTLDROP);
                goto out;
        }
@@ -2068,7 +2069,9 @@ process:
                        return 0;
                }
        }
-       if (unlikely(iph->ttl < inet_sk(sk)->min_ttl)) {
+
+       /* min_ttl can be changed concurrently from do_ip_setsockopt() */
+       if (unlikely(iph->ttl < READ_ONCE(inet_sk(sk)->min_ttl))) {
                __NET_INC_STATS(net, LINUX_MIB_TCPMINTTLDROP);
                goto discard_and_relse;
        }