fix async-cancel-safety of pthread_cancel
authorRich Felker <dalias@aerifal.cx>
Mon, 6 Jul 2020 22:46:57 +0000 (18:46 -0400)
committerRich Felker <dalias@aerifal.cx>
Mon, 6 Jul 2020 22:46:57 +0000 (18:46 -0400)
the previous commit addressing async-signal-safety issues around
pthread_kill did not fully fix pthread_cancel, which is also required
(albeit rather irrationally) to be async-cancel-safe.

without blocking implementation-internal signals, it's possible that,
when async cancellation is enabled, a cancel signal sent by another
thread interrupts pthread_kill while the killlock for a targeted
thread is held. as a result, the calling thread will terminate due to
cancellation without ever unlocking the targeted thread's killlock,
and thus the targeted thread will be unable to exit.

src/thread/pthread_kill.c

index 446254b..79ddb20 100644 (file)
@@ -5,7 +5,10 @@ int pthread_kill(pthread_t t, int sig)
 {
        int r;
        sigset_t set;
-       __block_app_sigs(&set);
+       /* Block not just app signals, but internal ones too, since
+        * pthread_kill is used to implement pthread_cancel, which
+        * must be async-cancel-safe. */
+       __block_all_sigs(&set);
        LOCK(t->killlock);
        r = t->tid ? -__syscall(SYS_tkill, t->tid, sig)
                : (sig+0U >= _NSIG ? EINVAL : 0);