| 1 | // Regression test for |
| 2 | // https://code.google.com/p/address-sanitizer/issues/detail?id=180 |
| 3 | // Fails with debug checks: https://bugs.llvm.org/show_bug.cgi?id=46860 |
| 4 | // XFAIL: !compiler-rt-optimized && tsan |
| 5 | |
| 6 | // FIXME: Implement. |
| 7 | // XFAIL: hwasan |
| 8 | |
| 9 | // RUN: %clangxx -O0 %s -o %t |
| 10 | |
| 11 | // RUN: %env_tool_opts=handle_segv=0 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK0 |
| 12 | // RUN: %env_tool_opts=handle_segv=1 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK1 |
| 13 | // RUN: %env_tool_opts=handle_segv=2 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK2 |
| 14 | |
| 15 | // RUN: %env_tool_opts=handle_segv=0:allow_user_segv_handler=0 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK0 |
| 16 | // RUN: %env_tool_opts=handle_segv=1:allow_user_segv_handler=0 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK2 |
| 17 | // RUN: %env_tool_opts=handle_segv=2:allow_user_segv_handler=0 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK2 |
| 18 | |
| 19 | // RUN: %env_tool_opts=handle_segv=0:allow_user_segv_handler=1 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK0 |
| 20 | // RUN: %env_tool_opts=handle_segv=1:allow_user_segv_handler=1 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK1 |
| 21 | // RUN: %env_tool_opts=handle_segv=2:allow_user_segv_handler=1 not %run %t 2>&1 | FileCheck %s --check-prefix=CHECK2 |
| 22 | |
| 23 | // Flaky errors in debuggerd with "waitpid returned unexpected pid (0)" in logcat. |
| 24 | // UNSUPPORTED: android && i386-target-arch |
| 25 | |
| 26 | #include <signal.h> |
| 27 | #include <stdio.h> |
| 28 | #include <stdlib.h> |
| 29 | |
| 30 | struct sigaction original_sigaction_sigbus; |
| 31 | struct sigaction original_sigaction_sigsegv; |
| 32 | |
| 33 | void User_OnSIGSEGV(int signum, siginfo_t *siginfo, void *context) { |
| 34 | fprintf(stderr, format: "User sigaction called\n" ); |
| 35 | struct sigaction original_sigaction = {}; |
| 36 | if (signum == SIGBUS) |
| 37 | original_sigaction = original_sigaction_sigbus; |
| 38 | else if (signum == SIGSEGV) |
| 39 | original_sigaction = original_sigaction_sigsegv; |
| 40 | else { |
| 41 | printf(format: "Invalid signum" ); |
| 42 | exit(status: 1); |
| 43 | } |
| 44 | if (original_sigaction.sa_flags | SA_SIGINFO) { |
| 45 | if (original_sigaction.sa_sigaction) |
| 46 | original_sigaction.sa_sigaction(signum, siginfo, context); |
| 47 | } else { |
| 48 | if (original_sigaction.sa_handler) |
| 49 | original_sigaction.sa_handler(signum); |
| 50 | } |
| 51 | exit(status: 1); |
| 52 | } |
| 53 | |
| 54 | int DoSEGV() { |
| 55 | volatile int *x = 0; |
| 56 | return *x; |
| 57 | } |
| 58 | |
| 59 | bool InstallHandler(int signum, struct sigaction *original_sigaction) { |
| 60 | struct sigaction user_sigaction = {}; |
| 61 | user_sigaction.sa_sigaction = User_OnSIGSEGV; |
| 62 | user_sigaction.sa_flags = SA_SIGINFO; |
| 63 | if (sigaction(sig: signum, act: &user_sigaction, oact: original_sigaction)) { |
| 64 | perror(s: "sigaction" ); |
| 65 | return false; |
| 66 | } |
| 67 | return true; |
| 68 | } |
| 69 | |
| 70 | int main() { |
| 71 | // Let's install handlers for both SIGSEGV and SIGBUS, since pre-Yosemite |
| 72 | // 32-bit Darwin triggers SIGBUS instead. |
| 73 | if (InstallHandler(SIGSEGV, original_sigaction: &original_sigaction_sigsegv) && |
| 74 | InstallHandler(SIGBUS, original_sigaction: &original_sigaction_sigbus)) { |
| 75 | fprintf(stderr, format: "User sigaction installed\n" ); |
| 76 | } |
| 77 | return DoSEGV(); |
| 78 | } |
| 79 | |
| 80 | // CHECK0-NOT: Sanitizer:DEADLYSIGNAL |
| 81 | // CHECK0-NOT: Sanitizer: SEGV on unknown address |
| 82 | // CHECK0: User sigaction installed |
| 83 | // CHECK0-NEXT: User sigaction called |
| 84 | |
| 85 | // CHECK1: User sigaction installed |
| 86 | // CHECK1-NEXT: User sigaction called |
| 87 | // CHECK1-NEXT: Sanitizer:DEADLYSIGNAL |
| 88 | // CHECK1: Sanitizer: SEGV on unknown address |
| 89 | |
| 90 | // CHECK2-NOT: User sigaction called |
| 91 | // CHECK2: User sigaction installed |
| 92 | // CHECK2-NEXT: Sanitizer:DEADLYSIGNAL |
| 93 | // CHECK2: Sanitizer: SEGV on unknown address |
| 94 | |