From 405de4ae106cc989f81eafa262a0f324f2732a84 Mon Sep 17 00:00:00 2001 From: Kuba Brecka Date: Thu, 22 Jan 2015 23:36:47 +0000 Subject: [PATCH] Fix/workaround for OS X truncated stacktraces taken by external tools This patch is a proposed solution for https://code.google.com/p/address-sanitizer/issues/detail?id=375: When the stacktraces are captured and printed by ASan itself, they are fine, but when the program has already printed the report (or is just printing it), capturing a stacktrace via other means is broken. "Other means" include OS X CrashReporter, debuggers or calling backtrace() within the program. For example calling backtrace() from a sanitizer_set_death_callback function prints a very truncated stacktrace. Reviewed at http://reviews.llvm.org/D7103 llvm-svn: 226878 --- compiler-rt/lib/asan/asan_mac.cc | 7 ---- compiler-rt/lib/asan/asan_report.cc | 2 + .../lib/sanitizer_common/sanitizer_internal_defs.h | 7 ++++ .../asan/TestCases/Darwin/crashlog-stacktraces.c | 43 ++++++++++++++++++++++ 4 files changed, 52 insertions(+), 7 deletions(-) create mode 100644 compiler-rt/test/asan/TestCases/Darwin/crashlog-stacktraces.c diff --git a/compiler-rt/lib/asan/asan_mac.cc b/compiler-rt/lib/asan/asan_mac.cc index 2e0b2d3..0565412 100644 --- a/compiler-rt/lib/asan/asan_mac.cc +++ b/compiler-rt/lib/asan/asan_mac.cc @@ -374,13 +374,6 @@ void dispatch_source_set_event_handler(dispatch_source_t ds, void(^work)(void)); work(); \ } -// Forces the compiler to generate a frame pointer in the function. -#define ENABLE_FRAME_POINTER \ - do { \ - volatile uptr enable_fp; \ - enable_fp = GET_CURRENT_FRAME(); \ - } while (0) - INTERCEPTOR(void, dispatch_async, dispatch_queue_t dq, void(^work)(void)) { ENABLE_FRAME_POINTER; diff --git a/compiler-rt/lib/asan/asan_report.cc b/compiler-rt/lib/asan/asan_report.cc index 0fb5027..2399cfd0 100644 --- a/compiler-rt/lib/asan/asan_report.cc +++ b/compiler-rt/lib/asan/asan_report.cc @@ -937,6 +937,8 @@ using namespace __asan; // NOLINT void __asan_report_error(uptr pc, uptr bp, uptr sp, uptr addr, int is_write, uptr access_size) { + ENABLE_FRAME_POINTER; + // Determine the error type. const char *bug_descr = "unknown-crash"; if (AddrIsInMem(addr)) { diff --git a/compiler-rt/lib/sanitizer_common/sanitizer_internal_defs.h b/compiler-rt/lib/sanitizer_common/sanitizer_internal_defs.h index 450d918..2a0b41f 100644 --- a/compiler-rt/lib/sanitizer_common/sanitizer_internal_defs.h +++ b/compiler-rt/lib/sanitizer_common/sanitizer_internal_defs.h @@ -325,4 +325,11 @@ extern "C" void* _ReturnAddress(void); } while (internal_iserror(res, &rverrno) && rverrno == EINTR); \ } +// Forces the compiler to generate a frame pointer in the function. +#define ENABLE_FRAME_POINTER \ + do { \ + volatile uptr enable_fp; \ + enable_fp = GET_CURRENT_FRAME(); \ + } while (0) + #endif // SANITIZER_DEFS_H diff --git a/compiler-rt/test/asan/TestCases/Darwin/crashlog-stacktraces.c b/compiler-rt/test/asan/TestCases/Darwin/crashlog-stacktraces.c new file mode 100644 index 0000000..e9af539 --- /dev/null +++ b/compiler-rt/test/asan/TestCases/Darwin/crashlog-stacktraces.c @@ -0,0 +1,43 @@ +// RUN: %clang_asan -O0 %s -o %t +// RUN: not %run %t 2>&1 | FileCheck %s + +#include +#include +#include +#include + +void death_function() { + fprintf(stderr, "DEATH CALLBACK\n"); + + void* callstack[128]; + int i, frames = backtrace(callstack, 128); + char** strs = backtrace_symbols(callstack, frames); + for (i = 0; i < frames; ++i) { + fprintf(stderr, "%s\n", strs[i]); + } + free(strs); + + fprintf(stderr, "END OF BACKTRACE\n"); +} + +int fault_function() { + char *x = (char*)malloc(10 * sizeof(char)); + free(x); + return x[5]; // BOOM +} + +int main() { + __sanitizer_set_death_callback(death_function); + fault_function(); + return 0; +} + +// CHECK: {{.*ERROR: AddressSanitizer: heap-use-after-free on address}} +// CHECK: {{READ of size 1 at 0x.* thread T0}} +// CHECK: {{ #0 0x.* in fault_function}} + +// CHECK: DEATH CALLBACK +// CHECK: death_function +// CHECK: fault_function +// CHECK: main +// CHECK: END OF BACKTRACE -- 2.7.4