posix-timers: Cure si_sys_private race
authorThomas Gleixner <tglx@linutronix.de>
Tue, 1 Oct 2024 08:42:03 +0000 (10:42 +0200)
committerThomas Gleixner <tglx@linutronix.de>
Tue, 29 Oct 2024 10:43:18 +0000 (11:43 +0100)
The si_sys_private member of the siginfo which is embedded in the
preallocated sigqueue is used by the posix timer code to decide whether a
timer must be reprogrammed on signal delivery.

The handling of this is racy as a long standing comment in that code
documents. It is modified with the timer lock held, but without sighand
lock being held. The actual signal delivery code checks for it under
sighand lock without holding the timer lock.

Hand the new value to send_sigqueue() as argument and store it with sighand
lock held. This is an intermediate change to address this issue.

The arguments to this function will be cleanup in subsequent changes.

Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Reviewed-by: Frederic Weisbecker <frederic@kernel.org>
Acked-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Link: https://lore.kernel.org/all/20241001083835.434338954@linutronix.de
include/linux/sched/signal.h
kernel/signal.c
kernel/time/posix-timers.c

index c8ed09ac29ac580d15e9bad0175e0507349faee9..bd9f569231d98a432800bdb15ff3cef3712dd2b0 100644 (file)
@@ -340,7 +340,7 @@ extern int send_sig(int, struct task_struct *, int);
 extern int zap_other_threads(struct task_struct *p);
 extern struct sigqueue *sigqueue_alloc(void);
 extern void sigqueue_free(struct sigqueue *);
-extern int send_sigqueue(struct sigqueue *, struct pid *, enum pid_type);
+extern int send_sigqueue(struct sigqueue *, struct pid *, enum pid_type, int si_private);
 extern int do_sigaction(int, struct k_sigaction *, struct k_sigaction *);
 
 static inline void clear_notify_signal(void)
index f420c430b24a554b8bb02075a3f4df683e3be8a5..1563c83ff224f1cbc832082d3c44aa1753180d0f 100644 (file)
@@ -1919,7 +1919,7 @@ void sigqueue_free(struct sigqueue *q)
                __sigqueue_free(q);
 }
 
-int send_sigqueue(struct sigqueue *q, struct pid *pid, enum pid_type type)
+int send_sigqueue(struct sigqueue *q, struct pid *pid, enum pid_type type, int si_private)
 {
        int sig = q->info.si_signo;
        struct sigpending *pending;
@@ -1954,6 +1954,14 @@ int send_sigqueue(struct sigqueue *q, struct pid *pid, enum pid_type type)
        if (!likely(lock_task_sighand(t, &flags)))
                goto ret;
 
+       /*
+        * Update @q::info::si_sys_private for posix timer signals with
+        * sighand locked to prevent a race against dequeue_signal() which
+        * decides based on si_sys_private whether to invoke
+        * posixtimer_rearm() or not.
+        */
+       q->info.si_sys_private = si_private;
+
        ret = 1; /* the signal is ignored */
        result = TRACE_SIGNAL_IGNORED;
        if (!prepare_signal(sig, t, false))
index d461a32b7260820c915b468b7b0b53a6679f7523..05af074285fa477c1f33573e17ab5056827cd035 100644 (file)
@@ -299,21 +299,8 @@ int posix_timer_queue_signal(struct k_itimer *timr)
        if (timr->it_interval)
                si_private = ++timr->it_requeue_pending;
 
-       /*
-        * FIXME: if ->sigq is queued we can race with
-        * dequeue_signal()->posixtimer_rearm().
-        *
-        * If dequeue_signal() sees the "right" value of
-        * si_sys_private it calls posixtimer_rearm().
-        * We re-queue ->sigq and drop ->it_lock().
-        * posixtimer_rearm() locks the timer
-        * and re-schedules it while ->sigq is pending.
-        * Not really bad, but not that we want.
-        */
-       timr->sigq->info.si_sys_private = si_private;
-
        type = !(timr->it_sigev_notify & SIGEV_THREAD_ID) ? PIDTYPE_TGID : PIDTYPE_PID;
-       ret = send_sigqueue(timr->sigq, timr->it_pid, type);
+       ret = send_sigqueue(timr->sigq, timr->it_pid, type, si_private);
        /* If we failed to send the signal the timer stops. */
        return ret > 0;
 }