Files
clang-p2996/compiler-rt/test/tsan/signal_thread2.cpp
Dmitry Vyukov 3056ccdbae tsan: fix deadlock/crash in signal handling
We set in_blocking_func around some blocking C functions so that we don't
delay signal infinitely (if in_blocking_func is set we deliver signals
synchronously).

However, pthread_join is blocking but also call munmap/free to free thread
resources. If we are inside the munmap/free interceptors called from
pthread_join and deliver a signal synchronously, it can lead to deadlocks
and crashes since we re-enter runtime and try to lock the same mutexes
or use the same per-thread data structures.

If we re-enter runtime via an interceptor when in_blocking_func is set,
temporary reset in_blocking_func around the interceptor and restore it back
when we return from the recursive interceptor.

Also move in_blocking_func from ThreadSignalContext to ThreadContext
so that we can CHECK that it's not set in SlotLocker ctor.

Fixes https://github.com/google/sanitizers/issues/1540

Reviewed By: melver

Differential Revision: https://reviews.llvm.org/D127845
2022-09-30 14:23:46 +02:00

67 lines
1.3 KiB
C++

// RUN: %clangxx_tsan %s -o %t && %run %t 2>&1 | FileCheck %s
// UNSUPPORTED: darwin
// Test case for https://github.com/google/sanitizers/issues/1540
#include <errno.h>
#include <pthread.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
volatile int X;
static void handler(int sig) {
(void)sig;
if (X != 0)
printf("bad");
}
static void *thr1(void *p) {
sleep(1);
return 0;
}
static void *thr(void *p) {
pthread_t th[10];
for (int i = 0; i < sizeof(th) / sizeof(th[0]); i++)
pthread_create(&th[i], 0, thr1, 0);
for (int i = 0; i < sizeof(th) / sizeof(th[0]); i++)
pthread_join(th[i], 0);
return 0;
}
int main() {
struct sigaction act = {};
act.sa_handler = &handler;
if (sigaction(SIGPROF, &act, 0)) {
perror("sigaction");
exit(1);
}
itimerval t;
t.it_value.tv_sec = 0;
t.it_value.tv_usec = 10;
t.it_interval = t.it_value;
if (setitimer(ITIMER_PROF, &t, 0)) {
perror("setitimer");
exit(1);
}
pthread_t th[100];
for (int i = 0; i < sizeof(th) / sizeof(th[0]); i++)
pthread_create(&th[i], 0, thr, 0);
for (int i = 0; i < sizeof(th) / sizeof(th[0]); i++)
pthread_join(th[i], 0);
fprintf(stderr, "DONE\n");
return 0;
}
// CHECK-NOT: WARNING: ThreadSanitizer:
// CHECK: DONE
// CHECK-NOT: WARNING: ThreadSanitizer: