| 1 | // RUN: %clang_tsan -O1 %s -o %t && %run %t 2>&1 | FileCheck %s |
| 2 | |
| 3 | // Test case for longjumping out of signal handler: |
| 4 | // https://github.com/google/sanitizers/issues/482 |
| 5 | |
| 6 | // This test fails on powerpc64 BE (VMA=44), a segmentation fault |
| 7 | // error happens at the second assignment |
| 8 | // "((volatile int *volatile)mem)[1] = 1". |
| 9 | // XFAIL: target=powerpc64-unknown-linux-gnu{{.*}} |
| 10 | |
| 11 | #include <setjmp.h> |
| 12 | #include <signal.h> |
| 13 | #include <stdlib.h> |
| 14 | #include <stdio.h> |
| 15 | #include <sys/mman.h> |
| 16 | |
| 17 | #ifdef __APPLE__ |
| 18 | #define SIGNAL_TO_HANDLE SIGBUS |
| 19 | #else |
| 20 | #define SIGNAL_TO_HANDLE SIGSEGV |
| 21 | #endif |
| 22 | |
| 23 | sigjmp_buf fault_jmp; |
| 24 | volatile int fault_expected; |
| 25 | |
| 26 | void sigfault_handler(int sig) { |
| 27 | if (!fault_expected) |
| 28 | abort(); |
| 29 | |
| 30 | /* just return from sighandler to proper place */ |
| 31 | fault_expected = 0; |
| 32 | siglongjmp(env: fault_jmp, val: 1); |
| 33 | } |
| 34 | |
| 35 | #define MUST_FAULT(code) do { \ |
| 36 | fault_expected = 1; \ |
| 37 | if (!sigsetjmp(fault_jmp, 1)) { \ |
| 38 | code; /* should pagefault -> sihandler does longjmp */ \ |
| 39 | fprintf(stderr, "%s not faulted\n", #code); \ |
| 40 | abort(); \ |
| 41 | } else { \ |
| 42 | fprintf(stderr, "%s faulted ok\n", #code); \ |
| 43 | } \ |
| 44 | } while (0) |
| 45 | |
| 46 | int main() { |
| 47 | struct sigaction act; |
| 48 | act.sa_handler = sigfault_handler; |
| 49 | act.sa_flags = 0; |
| 50 | if (sigemptyset(set: &act.sa_mask)) { |
| 51 | perror(s: "sigemptyset" ); |
| 52 | exit(status: 1); |
| 53 | } |
| 54 | |
| 55 | if (sigaction(SIGNAL_TO_HANDLE, act: &act, NULL)) { |
| 56 | perror(s: "sigaction" ); |
| 57 | exit(status: 1); |
| 58 | } |
| 59 | |
| 60 | void *mem = mmap(addr: 0, len: 4096, PROT_NONE, MAP_PRIVATE | MAP_ANON, |
| 61 | fd: -1, offset: 0); |
| 62 | |
| 63 | MUST_FAULT(((volatile int *volatile)mem)[0] = 0); |
| 64 | MUST_FAULT(((volatile int *volatile)mem)[1] = 1); |
| 65 | MUST_FAULT(((volatile int *volatile)mem)[3] = 1); |
| 66 | |
| 67 | // Ensure that tsan does not think that we are |
| 68 | // in a signal handler. |
| 69 | void *volatile p = malloc(size: 10); |
| 70 | ((volatile int*)p)[1] = 1; |
| 71 | free(ptr: (void*)p); |
| 72 | |
| 73 | munmap(addr: p, len: 4096); |
| 74 | |
| 75 | fprintf(stderr, format: "DONE\n" ); |
| 76 | return 0; |
| 77 | } |
| 78 | |
| 79 | // CHECK-NOT: WARNING: ThreadSanitizer |
| 80 | // CHECK: DONE |
| 81 | |