forked from OSchip/llvm-project
76 lines
1.4 KiB
C++
76 lines
1.4 KiB
C++
// RUN: %clangxx_tsan %s -o %t && %run %t 2>&1 | FileCheck %s
|
|
|
|
#include <pthread.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <signal.h>
|
|
#include <sys/types.h>
|
|
#include <sys/time.h>
|
|
#include <unistd.h>
|
|
#include <errno.h>
|
|
|
|
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:
|