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
67 lines
1.3 KiB
C++
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:
|