2015-06-25 04:13:44 +08:00
|
|
|
//===-- sanitizer_unwind_linux_libcdep.cc ---------------------------------===//
|
2014-09-01 20:43:03 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2014-09-01 20:43:03 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains the unwind.h-based (aka "slow") stack unwinding routines
|
[Sanitizers] Basic sanitizer Solaris support (PR 33274)
Summary:
This is the first mostly working version of the Sanitizer port to 32-bit Solaris/x86.
It is currently based on Solaris 11.4 Beta.
This part was initially developed inside libsanitizer in the GCC tree and should apply to
both. Subsequent parts will address changes to clang, the compiler-rt build system
and testsuite.
I'm not yet sure what the right patch granularity is: if it's profitable to split the patch
up, I'd like to get guidance on how to do so.
Most of the changes are probably straightforward with a few exceptions:
* The Solaris syscall interface isn't stable, undocumented and can change within an
OS release. The stable interface is the libc interface, which I'm using here, if possible
using the internal _-prefixed names.
* While the patch primarily target 32-bit x86, I've left a few sparc changes in. They
cannot currently be used with clang due to a backend limitation, but have worked
fine inside the gcc tree.
* Some functions (e.g. largefile versions of functions like open64) only exist in 32-bit
Solaris, so I've introduced a separate SANITIZER_SOLARIS32 to check for that.
The patch (with the subsequent ones to be submitted shortly) was tested
on i386-pc-solaris2.11. Only a few failures remain, some of them analyzed, some
still TBD:
AddressSanitizer-i386-sunos :: TestCases/Posix/concurrent_overflow.cc
AddressSanitizer-i386-sunos :: TestCases/init-order-atexit.cc
AddressSanitizer-i386-sunos :: TestCases/log-path_test.cc
AddressSanitizer-i386-sunos :: TestCases/malloc-no-intercept.c
AddressSanitizer-i386-sunos-dynamic :: TestCases/Posix/concurrent_overflow.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/Posix/start-deactivated.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/default_options.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/init-order-atexit.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/log-path_test.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/malloc-no-intercept.c
SanitizerCommon-Unit :: ./Sanitizer-i386-Test/MemoryMappingLayout.DumpListOfModules
SanitizerCommon-Unit :: ./Sanitizer-i386-Test/SanitizerCommon.PthreadDestructorIterations
Maybe this is good enough the get the ball rolling.
Reviewers: kcc, alekseyshl
Reviewed By: alekseyshl
Subscribers: srhines, jyknight, kubamracek, krytarowski, fedor.sergeev, llvm-commits, #sanitizers
Tags: #sanitizers
Differential Revision: https://reviews.llvm.org/D40898
llvm-svn: 320740
2017-12-15 04:14:29 +08:00
|
|
|
// available to the tools on Linux, Android, NetBSD, FreeBSD, and Solaris.
|
2014-09-01 20:43:03 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "sanitizer_platform.h"
|
[Sanitizers] Basic sanitizer Solaris support (PR 33274)
Summary:
This is the first mostly working version of the Sanitizer port to 32-bit Solaris/x86.
It is currently based on Solaris 11.4 Beta.
This part was initially developed inside libsanitizer in the GCC tree and should apply to
both. Subsequent parts will address changes to clang, the compiler-rt build system
and testsuite.
I'm not yet sure what the right patch granularity is: if it's profitable to split the patch
up, I'd like to get guidance on how to do so.
Most of the changes are probably straightforward with a few exceptions:
* The Solaris syscall interface isn't stable, undocumented and can change within an
OS release. The stable interface is the libc interface, which I'm using here, if possible
using the internal _-prefixed names.
* While the patch primarily target 32-bit x86, I've left a few sparc changes in. They
cannot currently be used with clang due to a backend limitation, but have worked
fine inside the gcc tree.
* Some functions (e.g. largefile versions of functions like open64) only exist in 32-bit
Solaris, so I've introduced a separate SANITIZER_SOLARIS32 to check for that.
The patch (with the subsequent ones to be submitted shortly) was tested
on i386-pc-solaris2.11. Only a few failures remain, some of them analyzed, some
still TBD:
AddressSanitizer-i386-sunos :: TestCases/Posix/concurrent_overflow.cc
AddressSanitizer-i386-sunos :: TestCases/init-order-atexit.cc
AddressSanitizer-i386-sunos :: TestCases/log-path_test.cc
AddressSanitizer-i386-sunos :: TestCases/malloc-no-intercept.c
AddressSanitizer-i386-sunos-dynamic :: TestCases/Posix/concurrent_overflow.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/Posix/start-deactivated.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/default_options.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/init-order-atexit.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/log-path_test.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/malloc-no-intercept.c
SanitizerCommon-Unit :: ./Sanitizer-i386-Test/MemoryMappingLayout.DumpListOfModules
SanitizerCommon-Unit :: ./Sanitizer-i386-Test/SanitizerCommon.PthreadDestructorIterations
Maybe this is good enough the get the ball rolling.
Reviewers: kcc, alekseyshl
Reviewed By: alekseyshl
Subscribers: srhines, jyknight, kubamracek, krytarowski, fedor.sergeev, llvm-commits, #sanitizers
Tags: #sanitizers
Differential Revision: https://reviews.llvm.org/D40898
llvm-svn: 320740
2017-12-15 04:14:29 +08:00
|
|
|
#if SANITIZER_FREEBSD || SANITIZER_LINUX || SANITIZER_NETBSD || \
|
|
|
|
SANITIZER_SOLARIS
|
2014-09-01 20:43:03 +08:00
|
|
|
#include "sanitizer_common.h"
|
|
|
|
#include "sanitizer_stacktrace.h"
|
|
|
|
|
|
|
|
#if SANITIZER_ANDROID
|
|
|
|
#include <dlfcn.h> // for dlopen()
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if SANITIZER_FREEBSD
|
|
|
|
#define _GNU_SOURCE // to declare _Unwind_Backtrace() from <unwind.h>
|
|
|
|
#endif
|
|
|
|
#include <unwind.h>
|
|
|
|
|
|
|
|
namespace __sanitizer {
|
|
|
|
|
2019-02-23 06:03:09 +08:00
|
|
|
//---------------------------- UnwindSlow --------------------------------------
|
2014-09-01 20:43:03 +08:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
uptr absolute_pc;
|
|
|
|
uptr stack_top;
|
|
|
|
uptr stack_size;
|
|
|
|
} backtrace_frame_t;
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
typedef void *(*acquire_my_map_info_list_func)();
|
|
|
|
typedef void (*release_my_map_info_list_func)(void *map);
|
|
|
|
typedef sptr (*unwind_backtrace_signal_arch_func)(
|
|
|
|
void *siginfo, void *sigcontext, void *map_info_list,
|
|
|
|
backtrace_frame_t *backtrace, uptr ignore_depth, uptr max_depth);
|
|
|
|
acquire_my_map_info_list_func acquire_my_map_info_list;
|
|
|
|
release_my_map_info_list_func release_my_map_info_list;
|
|
|
|
unwind_backtrace_signal_arch_func unwind_backtrace_signal_arch;
|
|
|
|
} // extern "C"
|
|
|
|
|
|
|
|
#if SANITIZER_ANDROID
|
|
|
|
void SanitizerInitializeUnwinder() {
|
2016-05-12 04:53:43 +08:00
|
|
|
if (AndroidGetApiLevel() >= ANDROID_LOLLIPOP_MR1) return;
|
|
|
|
|
|
|
|
// Pre-lollipop Android can not unwind through signal handler frames with
|
|
|
|
// libgcc unwinder, but it has a libcorkscrew.so library with the necessary
|
|
|
|
// workarounds.
|
2014-09-01 20:43:03 +08:00
|
|
|
void *p = dlopen("libcorkscrew.so", RTLD_LAZY);
|
|
|
|
if (!p) {
|
|
|
|
VReport(1,
|
|
|
|
"Failed to open libcorkscrew.so. You may see broken stack traces "
|
|
|
|
"in SEGV reports.");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
acquire_my_map_info_list =
|
|
|
|
(acquire_my_map_info_list_func)(uptr)dlsym(p, "acquire_my_map_info_list");
|
|
|
|
release_my_map_info_list =
|
|
|
|
(release_my_map_info_list_func)(uptr)dlsym(p, "release_my_map_info_list");
|
|
|
|
unwind_backtrace_signal_arch = (unwind_backtrace_signal_arch_func)(uptr)dlsym(
|
|
|
|
p, "unwind_backtrace_signal_arch");
|
|
|
|
if (!acquire_my_map_info_list || !release_my_map_info_list ||
|
|
|
|
!unwind_backtrace_signal_arch) {
|
|
|
|
VReport(1,
|
|
|
|
"Failed to find one of the required symbols in libcorkscrew.so. "
|
|
|
|
"You may see broken stack traces in SEGV reports.");
|
|
|
|
acquire_my_map_info_list = 0;
|
|
|
|
unwind_backtrace_signal_arch = 0;
|
|
|
|
release_my_map_info_list = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-08-05 21:36:49 +08:00
|
|
|
#if defined(__arm__) && !SANITIZER_NETBSD
|
|
|
|
// NetBSD uses dwarf EH
|
2014-09-01 20:43:03 +08:00
|
|
|
#define UNWIND_STOP _URC_END_OF_STACK
|
|
|
|
#define UNWIND_CONTINUE _URC_NO_REASON
|
|
|
|
#else
|
|
|
|
#define UNWIND_STOP _URC_NORMAL_STOP
|
|
|
|
#define UNWIND_CONTINUE _URC_NO_REASON
|
|
|
|
#endif
|
|
|
|
|
|
|
|
uptr Unwind_GetIP(struct _Unwind_Context *ctx) {
|
2015-06-24 05:39:57 +08:00
|
|
|
#if defined(__arm__) && !SANITIZER_MAC
|
2014-09-01 20:43:03 +08:00
|
|
|
uptr val;
|
|
|
|
_Unwind_VRS_Result res = _Unwind_VRS_Get(ctx, _UVRSC_CORE,
|
|
|
|
15 /* r15 = PC */, _UVRSD_UINT32, &val);
|
|
|
|
CHECK(res == _UVRSR_OK && "_Unwind_VRS_Get failed");
|
|
|
|
// Clear the Thumb bit.
|
|
|
|
return val & ~(uptr)1;
|
|
|
|
#else
|
2018-10-31 02:25:12 +08:00
|
|
|
return (uptr)_Unwind_GetIP(ctx);
|
2014-09-01 20:43:03 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
struct UnwindTraceArg {
|
2014-10-26 11:35:14 +08:00
|
|
|
BufferedStackTrace *stack;
|
2015-01-22 21:33:16 +08:00
|
|
|
u32 max_depth;
|
2014-09-01 20:43:03 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
_Unwind_Reason_Code Unwind_Trace(struct _Unwind_Context *ctx, void *param) {
|
|
|
|
UnwindTraceArg *arg = (UnwindTraceArg*)param;
|
|
|
|
CHECK_LT(arg->stack->size, arg->max_depth);
|
|
|
|
uptr pc = Unwind_GetIP(ctx);
|
2016-06-27 23:32:18 +08:00
|
|
|
const uptr kPageSize = GetPageSizeCached();
|
2016-06-28 20:53:20 +08:00
|
|
|
// Let's assume that any pointer in the 0th page (i.e. <0x1000 on i386 and
|
|
|
|
// x86_64) is invalid and stop unwinding here. If we're adding support for
|
|
|
|
// a platform where this isn't true, we need to reconsider this check.
|
2016-06-27 23:32:18 +08:00
|
|
|
if (pc < kPageSize) return UNWIND_STOP;
|
2014-10-26 11:35:14 +08:00
|
|
|
arg->stack->trace_buffer[arg->stack->size++] = pc;
|
2014-09-01 20:43:03 +08:00
|
|
|
if (arg->stack->size == arg->max_depth) return UNWIND_STOP;
|
|
|
|
return UNWIND_CONTINUE;
|
|
|
|
}
|
|
|
|
|
2019-02-23 06:03:09 +08:00
|
|
|
void BufferedStackTrace::UnwindSlow(uptr pc, u32 max_depth) {
|
2014-09-01 20:43:03 +08:00
|
|
|
CHECK_GE(max_depth, 2);
|
|
|
|
size = 0;
|
|
|
|
UnwindTraceArg arg = {this, Min(max_depth + 1, kStackTraceMax)};
|
|
|
|
_Unwind_Backtrace(Unwind_Trace, &arg);
|
|
|
|
// We need to pop a few frames so that pc is on top.
|
|
|
|
uptr to_pop = LocatePcInTrace(pc);
|
2014-11-14 23:13:23 +08:00
|
|
|
// trace_buffer[0] belongs to the current function so we always pop it,
|
|
|
|
// unless there is only 1 frame in the stack trace (1 frame is always better
|
|
|
|
// than 0!).
|
|
|
|
// 1-frame stacks don't normally happen, but this depends on the actual
|
|
|
|
// unwinder implementation (libgcc, libunwind, etc) which is outside of our
|
|
|
|
// control.
|
2014-11-07 20:03:07 +08:00
|
|
|
if (to_pop == 0 && size > 1)
|
2014-09-01 20:43:03 +08:00
|
|
|
to_pop = 1;
|
|
|
|
PopStackFrames(to_pop);
|
SanitizerCommon: fixes for unwinding & backtrace on SPARC
Summary:
This patch contains various fixes for the unwinding and backtrace machinery on the SPARC, which doesn't work correctly in various cases. It was tested with GCC on SPARC/Solaris and SPARC/Linux.
Patch by Eric Botcazou.
Reviewers: #sanitizers, vitalybuka
Reviewed By: #sanitizers, vitalybuka
Subscribers: jrtc27, delcypher, vitalybuka, ro, jyknight, kubamracek, fedor.sergeev, jdoerfert, llvm-commits, #sanitizers
Tags: #sanitizers, #llvm
Differential Revision: https://reviews.llvm.org/D58431
llvm-svn: 355965
2019-03-13 04:31:53 +08:00
|
|
|
#if defined(__GNUC__) && defined(__sparc__)
|
|
|
|
// __builtin_return_address returns the address of the call instruction
|
|
|
|
// on the SPARC and not the return address, so we need to compensate.
|
|
|
|
trace_buffer[0] = GetNextInstructionPc(pc);
|
|
|
|
#else
|
2014-10-26 11:35:14 +08:00
|
|
|
trace_buffer[0] = pc;
|
SanitizerCommon: fixes for unwinding & backtrace on SPARC
Summary:
This patch contains various fixes for the unwinding and backtrace machinery on the SPARC, which doesn't work correctly in various cases. It was tested with GCC on SPARC/Solaris and SPARC/Linux.
Patch by Eric Botcazou.
Reviewers: #sanitizers, vitalybuka
Reviewed By: #sanitizers, vitalybuka
Subscribers: jrtc27, delcypher, vitalybuka, ro, jyknight, kubamracek, fedor.sergeev, jdoerfert, llvm-commits, #sanitizers
Tags: #sanitizers, #llvm
Differential Revision: https://reviews.llvm.org/D58431
llvm-svn: 355965
2019-03-13 04:31:53 +08:00
|
|
|
#endif
|
2014-09-01 20:43:03 +08:00
|
|
|
}
|
|
|
|
|
2019-02-23 10:36:23 +08:00
|
|
|
void BufferedStackTrace::UnwindSlow(uptr pc, void *context, u32 max_depth) {
|
|
|
|
CHECK(context);
|
2014-09-01 20:43:03 +08:00
|
|
|
CHECK_GE(max_depth, 2);
|
|
|
|
if (!unwind_backtrace_signal_arch) {
|
2019-02-23 06:03:09 +08:00
|
|
|
UnwindSlow(pc, max_depth);
|
2014-09-01 20:43:03 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *map = acquire_my_map_info_list();
|
|
|
|
CHECK(map);
|
2018-05-07 13:56:36 +08:00
|
|
|
InternalMmapVector<backtrace_frame_t> frames(kStackTraceMax);
|
2014-09-01 20:43:03 +08:00
|
|
|
// siginfo argument appears to be unused.
|
|
|
|
sptr res = unwind_backtrace_signal_arch(/* siginfo */ 0, context, map,
|
|
|
|
frames.data(),
|
|
|
|
/* ignore_depth */ 0, max_depth);
|
|
|
|
release_my_map_info_list(map);
|
|
|
|
if (res < 0) return;
|
|
|
|
CHECK_LE((uptr)res, kStackTraceMax);
|
|
|
|
|
|
|
|
size = 0;
|
|
|
|
// +2 compensate for libcorkscrew unwinder returning addresses of call
|
|
|
|
// instructions instead of raw return addresses.
|
|
|
|
for (sptr i = 0; i < res; ++i)
|
2014-10-26 11:35:14 +08:00
|
|
|
trace_buffer[size++] = frames[i].absolute_pc + 2;
|
2014-09-01 20:43:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace __sanitizer
|
2014-09-01 21:07:50 +08:00
|
|
|
|
[Sanitizers] Basic sanitizer Solaris support (PR 33274)
Summary:
This is the first mostly working version of the Sanitizer port to 32-bit Solaris/x86.
It is currently based on Solaris 11.4 Beta.
This part was initially developed inside libsanitizer in the GCC tree and should apply to
both. Subsequent parts will address changes to clang, the compiler-rt build system
and testsuite.
I'm not yet sure what the right patch granularity is: if it's profitable to split the patch
up, I'd like to get guidance on how to do so.
Most of the changes are probably straightforward with a few exceptions:
* The Solaris syscall interface isn't stable, undocumented and can change within an
OS release. The stable interface is the libc interface, which I'm using here, if possible
using the internal _-prefixed names.
* While the patch primarily target 32-bit x86, I've left a few sparc changes in. They
cannot currently be used with clang due to a backend limitation, but have worked
fine inside the gcc tree.
* Some functions (e.g. largefile versions of functions like open64) only exist in 32-bit
Solaris, so I've introduced a separate SANITIZER_SOLARIS32 to check for that.
The patch (with the subsequent ones to be submitted shortly) was tested
on i386-pc-solaris2.11. Only a few failures remain, some of them analyzed, some
still TBD:
AddressSanitizer-i386-sunos :: TestCases/Posix/concurrent_overflow.cc
AddressSanitizer-i386-sunos :: TestCases/init-order-atexit.cc
AddressSanitizer-i386-sunos :: TestCases/log-path_test.cc
AddressSanitizer-i386-sunos :: TestCases/malloc-no-intercept.c
AddressSanitizer-i386-sunos-dynamic :: TestCases/Posix/concurrent_overflow.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/Posix/start-deactivated.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/default_options.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/init-order-atexit.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/log-path_test.cc
AddressSanitizer-i386-sunos-dynamic :: TestCases/malloc-no-intercept.c
SanitizerCommon-Unit :: ./Sanitizer-i386-Test/MemoryMappingLayout.DumpListOfModules
SanitizerCommon-Unit :: ./Sanitizer-i386-Test/SanitizerCommon.PthreadDestructorIterations
Maybe this is good enough the get the ball rolling.
Reviewers: kcc, alekseyshl
Reviewed By: alekseyshl
Subscribers: srhines, jyknight, kubamracek, krytarowski, fedor.sergeev, llvm-commits, #sanitizers
Tags: #sanitizers
Differential Revision: https://reviews.llvm.org/D40898
llvm-svn: 320740
2017-12-15 04:14:29 +08:00
|
|
|
#endif // SANITIZER_FREEBSD || SANITIZER_LINUX || SANITIZER_NETBSD ||
|
|
|
|
// SANITIZER_SOLARIS
|