From: Kuba Brecka Date: Thu, 22 Jan 2015 23:36:47 +0000 (+0000) Subject: Fix/workaround for OS X truncated stacktraces taken by external tools X-Git-Tag: llvmorg-3.7.0-rc1~14267 X-Git-Url: http://review.tizen.org/git/?a=commitdiff_plain;h=405de4ae106cc989f81eafa262a0f324f2732a84;p=platform%2Fupstream%2Fllvm.git 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 --- diff --git a/compiler-rt/lib/asan/asan_mac.cc b/compiler-rt/lib/asan/asan_mac.cc index 2e0b2d372c7a..056541287788 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 0fb50276186b..2399cfd0db13 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 450d9184e990..2a0b41f0bb99 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 000000000000..e9af5396e1c3 --- /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