Skip to content

Commit 285975d

Browse files
edumazetdavem330
authored andcommitted
net: annotate data-races around sk->sk_{rcv|snd}timeo
sk_getsockopt() runs without locks, we must add annotations to sk->sk_rcvtimeo and sk->sk_sndtimeo. In the future we might allow fetching these fields before we lock the socket in TCP fast path. Signed-off-by: Eric Dumazet <[email protected]> Signed-off-by: David S. Miller <[email protected]>
1 parent e6d12bd commit 285975d

File tree

2 files changed

+16
-12
lines changed

2 files changed

+16
-12
lines changed

net/core/sock.c

Lines changed: 14 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -429,6 +429,7 @@ static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen,
429429
{
430430
struct __kernel_sock_timeval tv;
431431
int err = sock_copy_user_timeval(&tv, optval, optlen, old_timeval);
432+
long val;
432433

433434
if (err)
434435
return err;
@@ -439,19 +440,20 @@ static int sock_set_timeout(long *timeo_p, sockptr_t optval, int optlen,
439440
if (tv.tv_sec < 0) {
440441
static int warned __read_mostly;
441442

442-
*timeo_p = 0;
443+
WRITE_ONCE(*timeo_p, 0);
443444
if (warned < 10 && net_ratelimit()) {
444445
warned++;
445446
pr_info("%s: `%s' (pid %d) tries to set negative timeout\n",
446447
__func__, current->comm, task_pid_nr(current));
447448
}
448449
return 0;
449450
}
450-
*timeo_p = MAX_SCHEDULE_TIMEOUT;
451-
if (tv.tv_sec == 0 && tv.tv_usec == 0)
452-
return 0;
453-
if (tv.tv_sec < (MAX_SCHEDULE_TIMEOUT / HZ - 1))
454-
*timeo_p = tv.tv_sec * HZ + DIV_ROUND_UP((unsigned long)tv.tv_usec, USEC_PER_SEC / HZ);
451+
val = MAX_SCHEDULE_TIMEOUT;
452+
if ((tv.tv_sec || tv.tv_usec) &&
453+
(tv.tv_sec < (MAX_SCHEDULE_TIMEOUT / HZ - 1)))
454+
val = tv.tv_sec * HZ + DIV_ROUND_UP((unsigned long)tv.tv_usec,
455+
USEC_PER_SEC / HZ);
456+
WRITE_ONCE(*timeo_p, val);
455457
return 0;
456458
}
457459

@@ -813,9 +815,9 @@ void sock_set_sndtimeo(struct sock *sk, s64 secs)
813815
{
814816
lock_sock(sk);
815817
if (secs && secs < MAX_SCHEDULE_TIMEOUT / HZ - 1)
816-
sk->sk_sndtimeo = secs * HZ;
818+
WRITE_ONCE(sk->sk_sndtimeo, secs * HZ);
817819
else
818-
sk->sk_sndtimeo = MAX_SCHEDULE_TIMEOUT;
820+
WRITE_ONCE(sk->sk_sndtimeo, MAX_SCHEDULE_TIMEOUT);
819821
release_sock(sk);
820822
}
821823
EXPORT_SYMBOL(sock_set_sndtimeo);
@@ -1721,12 +1723,14 @@ int sk_getsockopt(struct sock *sk, int level, int optname,
17211723

17221724
case SO_RCVTIMEO_OLD:
17231725
case SO_RCVTIMEO_NEW:
1724-
lv = sock_get_timeout(sk->sk_rcvtimeo, &v, SO_RCVTIMEO_OLD == optname);
1726+
lv = sock_get_timeout(READ_ONCE(sk->sk_rcvtimeo), &v,
1727+
SO_RCVTIMEO_OLD == optname);
17251728
break;
17261729

17271730
case SO_SNDTIMEO_OLD:
17281731
case SO_SNDTIMEO_NEW:
1729-
lv = sock_get_timeout(sk->sk_sndtimeo, &v, SO_SNDTIMEO_OLD == optname);
1732+
lv = sock_get_timeout(READ_ONCE(sk->sk_sndtimeo), &v,
1733+
SO_SNDTIMEO_OLD == optname);
17301734
break;
17311735

17321736
case SO_RCVLOWAT:

net/sched/em_meta.c

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -568,7 +568,7 @@ META_COLLECTOR(int_sk_rcvtimeo)
568568
*err = -1;
569569
return;
570570
}
571-
dst->value = sk->sk_rcvtimeo / HZ;
571+
dst->value = READ_ONCE(sk->sk_rcvtimeo) / HZ;
572572
}
573573

574574
META_COLLECTOR(int_sk_sndtimeo)
@@ -579,7 +579,7 @@ META_COLLECTOR(int_sk_sndtimeo)
579579
*err = -1;
580580
return;
581581
}
582-
dst->value = sk->sk_sndtimeo / HZ;
582+
dst->value = READ_ONCE(sk->sk_sndtimeo) / HZ;
583583
}
584584

585585
META_COLLECTOR(int_sk_sendmsg_off)

0 commit comments

Comments
 (0)