2012-02-10 01:20:14 +08:00
|
|
|
//===-- asan_win.cc -------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file is a part of AddressSanitizer, an address sanity checker.
|
|
|
|
//
|
|
|
|
// Windows-specific details.
|
|
|
|
//===----------------------------------------------------------------------===//
|
2013-03-19 22:33:38 +08:00
|
|
|
|
|
|
|
#include "sanitizer_common/sanitizer_platform.h"
|
2013-03-19 22:54:17 +08:00
|
|
|
#if SANITIZER_WINDOWS
|
2012-02-10 01:20:14 +08:00
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
#include <dbghelp.h>
|
2012-02-22 22:07:06 +08:00
|
|
|
#include <stdlib.h>
|
2012-02-10 01:20:14 +08:00
|
|
|
|
|
|
|
#include "asan_interceptors.h"
|
|
|
|
#include "asan_internal.h"
|
|
|
|
#include "asan_thread.h"
|
2012-06-19 17:21:57 +08:00
|
|
|
#include "sanitizer_common/sanitizer_libc.h"
|
2013-01-14 15:51:39 +08:00
|
|
|
#include "sanitizer_common/sanitizer_mutex.h"
|
2012-02-10 01:20:14 +08:00
|
|
|
|
|
|
|
namespace __asan {
|
|
|
|
|
|
|
|
// ---------------------- Stacktraces, symbols, etc. ---------------- {{{1
|
2013-01-14 15:51:39 +08:00
|
|
|
static BlockingMutex dbghelp_lock(LINKER_INITIALIZED);
|
2012-02-10 01:20:14 +08:00
|
|
|
static bool dbghelp_initialized = false;
|
|
|
|
#pragma comment(lib, "dbghelp.lib")
|
|
|
|
|
2012-12-13 18:03:50 +08:00
|
|
|
void GetStackTrace(StackTrace *stack, uptr max_s, uptr pc, uptr bp, bool fast) {
|
|
|
|
(void)fast;
|
2012-08-28 22:14:30 +08:00
|
|
|
stack->max_size = max_s;
|
2012-02-10 01:20:14 +08:00
|
|
|
void *tmp[kStackTraceMax];
|
|
|
|
|
|
|
|
// FIXME: CaptureStackBackTrace might be too slow for us.
|
|
|
|
// FIXME: Compare with StackWalk64.
|
|
|
|
// FIXME: Look at LLVMUnhandledExceptionFilter in Signals.inc
|
2012-08-28 22:14:30 +08:00
|
|
|
uptr cs_ret = CaptureStackBackTrace(1, stack->max_size, tmp, 0);
|
|
|
|
uptr offset = 0;
|
2012-02-10 01:20:14 +08:00
|
|
|
// Skip the RTL frames by searching for the PC in the stacktrace.
|
|
|
|
// FIXME: this doesn't work well for the malloc/free stacks yet.
|
2012-06-01 00:06:05 +08:00
|
|
|
for (uptr i = 0; i < cs_ret; i++) {
|
2012-05-31 22:35:53 +08:00
|
|
|
if (pc != (uptr)tmp[i])
|
2012-02-10 01:20:14 +08:00
|
|
|
continue;
|
|
|
|
offset = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2012-08-28 22:14:30 +08:00
|
|
|
stack->size = cs_ret - offset;
|
|
|
|
for (uptr i = 0; i < stack->size; i++)
|
|
|
|
stack->trace[i] = (uptr)tmp[i + offset];
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------- TSD ---------------- {{{1
|
|
|
|
static bool tsd_key_inited = false;
|
|
|
|
|
2012-05-31 22:35:53 +08:00
|
|
|
static __declspec(thread) void *fake_tsd = 0;
|
2012-02-10 01:20:14 +08:00
|
|
|
|
|
|
|
void AsanTSDInit(void (*destructor)(void *tsd)) {
|
|
|
|
// FIXME: we're ignoring the destructor for now.
|
|
|
|
tsd_key_inited = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void *AsanTSDGet() {
|
|
|
|
CHECK(tsd_key_inited);
|
|
|
|
return fake_tsd;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsanTSDSet(void *tsd) {
|
|
|
|
CHECK(tsd_key_inited);
|
|
|
|
fake_tsd = tsd;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------- Various stuff ---------------- {{{1
|
2012-08-24 17:22:05 +08:00
|
|
|
void MaybeReexec() {
|
|
|
|
// No need to re-exec on Windows.
|
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
void *AsanDoesNotSupportStaticLinkage() {
|
2012-03-12 19:45:09 +08:00
|
|
|
#if defined(_DEBUG)
|
|
|
|
#error Please build the runtime with a non-debug CRT: /MD or /MT
|
2012-02-22 21:59:49 +08:00
|
|
|
#endif
|
2012-05-31 22:35:53 +08:00
|
|
|
return 0;
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
2012-04-05 18:54:52 +08:00
|
|
|
void SetAlternateSignalStack() {
|
|
|
|
// FIXME: Decide what to do on Windows.
|
|
|
|
}
|
|
|
|
|
|
|
|
void UnsetAlternateSignalStack() {
|
|
|
|
// FIXME: Decide what to do on Windows.
|
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
void InstallSignalHandlers() {
|
|
|
|
// FIXME: Decide what to do on Windows.
|
|
|
|
}
|
|
|
|
|
2012-07-23 22:07:58 +08:00
|
|
|
void AsanPlatformThreadInit() {
|
|
|
|
// Nothing here for now.
|
|
|
|
}
|
|
|
|
|
2013-01-17 23:45:28 +08:00
|
|
|
void ReadContextStack(void *context, uptr *stack, uptr *ssize) {
|
2012-11-23 17:46:34 +08:00
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
} // namespace __asan
|
|
|
|
|
2012-10-02 20:11:17 +08:00
|
|
|
// ---------------------- Interface ---------------- {{{1
|
|
|
|
using namespace __asan; // NOLINT
|
|
|
|
|
2012-10-02 20:35:42 +08:00
|
|
|
extern "C" {
|
|
|
|
SANITIZER_INTERFACE_ATTRIBUTE NOINLINE
|
2012-10-02 20:11:17 +08:00
|
|
|
bool __asan_symbolize(const void *addr, char *out_buffer, int buffer_size) {
|
2013-01-14 15:51:39 +08:00
|
|
|
BlockingMutexLock lock(&dbghelp_lock);
|
2012-10-02 20:11:17 +08:00
|
|
|
if (!dbghelp_initialized) {
|
|
|
|
SymSetOptions(SYMOPT_DEFERRED_LOADS |
|
|
|
|
SYMOPT_UNDNAME |
|
|
|
|
SYMOPT_LOAD_LINES);
|
|
|
|
CHECK(SymInitialize(GetCurrentProcess(), 0, TRUE));
|
|
|
|
// FIXME: We don't call SymCleanup() on exit yet - should we?
|
|
|
|
dbghelp_initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// See http://msdn.microsoft.com/en-us/library/ms680578(VS.85).aspx
|
|
|
|
char buffer[sizeof(SYMBOL_INFO) + MAX_SYM_NAME * sizeof(CHAR)];
|
|
|
|
PSYMBOL_INFO symbol = (PSYMBOL_INFO)buffer;
|
|
|
|
symbol->SizeOfStruct = sizeof(SYMBOL_INFO);
|
|
|
|
symbol->MaxNameLen = MAX_SYM_NAME;
|
|
|
|
DWORD64 offset = 0;
|
|
|
|
BOOL got_objname = SymFromAddr(GetCurrentProcess(),
|
|
|
|
(DWORD64)addr, &offset, symbol);
|
|
|
|
if (!got_objname)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
DWORD unused;
|
|
|
|
IMAGEHLP_LINE64 info;
|
|
|
|
info.SizeOfStruct = sizeof(IMAGEHLP_LINE64);
|
|
|
|
BOOL got_fileline = SymGetLineFromAddr64(GetCurrentProcess(),
|
|
|
|
(DWORD64)addr, &unused, &info);
|
|
|
|
int written = 0;
|
|
|
|
out_buffer[0] = '\0';
|
|
|
|
// FIXME: it might be useful to print out 'obj' or 'obj+offset' info too.
|
|
|
|
if (got_fileline) {
|
|
|
|
written += internal_snprintf(out_buffer + written, buffer_size - written,
|
|
|
|
" %s %s:%d", symbol->Name,
|
|
|
|
info.FileName, info.LineNumber);
|
|
|
|
} else {
|
|
|
|
written += internal_snprintf(out_buffer + written, buffer_size - written,
|
|
|
|
" %s+0x%p", symbol->Name, offset);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2012-10-02 20:35:42 +08:00
|
|
|
} // extern "C"
|
2012-10-02 20:11:17 +08:00
|
|
|
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
#endif // _WIN32
|