// RUN: %clangxx_tsan %s -o %t && %run %t 2>&1 | FileCheck %s #include #include #include #include #include #include #include #include volatile bool signal_delivered; static void handler(int sig) { if (sig == SIGALRM) signal_delivered = true; } static void* thr(void *p) { sigset_t sigset; sigemptyset(&sigset); sigaddset(&sigset, SIGALRM); int ret = pthread_sigmask(SIG_UNBLOCK, &sigset, NULL); if (ret) abort(); struct sigaction act = {}; act.sa_handler = &handler; if (sigaction(SIGALRM, &act, 0)) { perror("sigaction"); exit(1); } itimerval t; t.it_value.tv_sec = 0; t.it_value.tv_usec = 10000; t.it_interval = t.it_value; if (setitimer(ITIMER_REAL, &t, 0)) { perror("setitimer"); exit(1); } while (!signal_delivered) { usleep(1000); } t.it_value.tv_usec = 0; if (setitimer(ITIMER_REAL, &t, 0)) { perror("setitimer"); exit(1); } fprintf(stderr, "SIGNAL DELIVERED\n"); return 0; } int main() { sigset_t sigset; sigemptyset(&sigset); sigaddset(&sigset, SIGALRM); int ret = pthread_sigmask(SIG_BLOCK, &sigset, NULL); if (ret) abort(); pthread_t th; pthread_create(&th, 0, thr, 0); pthread_join(th, 0); fprintf(stderr, "DONE\n"); return 0; } // CHECK-NOT: WARNING: ThreadSanitizer: // CHECK: SIGNAL DELIVERED // CHECK: DONE // CHECK-NOT: WARNING: ThreadSanitizer: