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.
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifdef _WIN32
|
|
|
|
#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 <new> // FIXME: temporarily needed for placement new in AsanLock.
|
|
|
|
|
|
|
|
#include "asan_interceptors.h"
|
|
|
|
#include "asan_internal.h"
|
|
|
|
#include "asan_lock.h"
|
|
|
|
#include "asan_procmaps.h"
|
|
|
|
#include "asan_thread.h"
|
|
|
|
|
2012-03-10 09:30:01 +08:00
|
|
|
// Should not add dependency on libstdc++,
|
|
|
|
// since most of the stuff here is inlinable.
|
|
|
|
#include <algorithm>
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
namespace __asan {
|
|
|
|
|
|
|
|
// ---------------------- Memory management ---------------- {{{1
|
2012-05-31 22:35:53 +08:00
|
|
|
void *AsanMmapFixedNoReserve(uptr fixed_addr, size_t size) {
|
2012-02-10 01:20:14 +08:00
|
|
|
return VirtualAlloc((LPVOID)fixed_addr, size,
|
|
|
|
MEM_RESERVE | MEM_COMMIT, PAGE_READWRITE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void *AsanMmapSomewhereOrDie(size_t size, const char *mem_type) {
|
2012-05-31 22:35:53 +08:00
|
|
|
void *rv = VirtualAlloc(0, size, MEM_RESERVE | MEM_COMMIT, PAGE_READWRITE);
|
|
|
|
if (rv == 0)
|
2012-02-10 01:20:14 +08:00
|
|
|
OutOfMemoryMessageAndDie(mem_type, size);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2012-05-31 22:35:53 +08:00
|
|
|
void *AsanMprotect(uptr fixed_addr, size_t size) {
|
2012-02-10 01:20:14 +08:00
|
|
|
return VirtualAlloc((LPVOID)fixed_addr, size,
|
|
|
|
MEM_RESERVE | MEM_COMMIT, PAGE_NOACCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsanUnmapOrDie(void *addr, size_t size) {
|
2012-02-24 23:28:43 +08:00
|
|
|
CHECK(VirtualFree(addr, size, MEM_DECOMMIT));
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------- IO ---------------- {{{1
|
|
|
|
size_t AsanWrite(int fd, const void *buf, size_t count) {
|
|
|
|
if (fd != 2)
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
|
2012-03-11 20:45:12 +08:00
|
|
|
HANDLE err = GetStdHandle(STD_ERROR_HANDLE);
|
2012-05-31 22:35:53 +08:00
|
|
|
if (err == 0)
|
2012-03-11 20:45:12 +08:00
|
|
|
return 0; // FIXME: this might not work on some apps.
|
|
|
|
DWORD ret;
|
2012-05-31 22:35:53 +08:00
|
|
|
if (!WriteFile(err, buf, count, &ret, 0))
|
2012-03-11 20:45:12 +08:00
|
|
|
return 0;
|
|
|
|
return ret;
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Looks like these functions are not needed and are linked in by the
|
|
|
|
// code unreachable on Windows. We should clean this up.
|
|
|
|
int AsanOpenReadonly(const char* filename) {
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t AsanRead(int fd, void *buf, size_t count) {
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
|
|
|
|
int AsanClose(int fd) {
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------- Stacktraces, symbols, etc. ---------------- {{{1
|
|
|
|
static AsanLock dbghelp_lock(LINKER_INITIALIZED);
|
|
|
|
static bool dbghelp_initialized = false;
|
|
|
|
#pragma comment(lib, "dbghelp.lib")
|
|
|
|
|
|
|
|
void AsanThread::SetThreadStackTopAndBottom() {
|
|
|
|
MEMORY_BASIC_INFORMATION mbi;
|
|
|
|
CHECK(VirtualQuery(&mbi /* on stack */,
|
|
|
|
&mbi, sizeof(mbi)) != 0);
|
|
|
|
// FIXME: is it possible for the stack to not be a single allocation?
|
|
|
|
// Are these values what ASan expects to get (reserved, not committed;
|
|
|
|
// including stack guard page) ?
|
2012-05-31 22:35:53 +08:00
|
|
|
stack_top_ = (uptr)mbi.BaseAddress + mbi.RegionSize;
|
|
|
|
stack_bottom_ = (uptr)mbi.AllocationBase;
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
2012-05-31 22:35:53 +08:00
|
|
|
void AsanStackTrace::GetStackTrace(size_t max_s, uptr pc, uptr bp) {
|
2012-02-10 01:20:14 +08:00
|
|
|
max_size = max_s;
|
|
|
|
void *tmp[kStackTraceMax];
|
|
|
|
|
|
|
|
// FIXME: CaptureStackBackTrace might be too slow for us.
|
|
|
|
// FIXME: Compare with StackWalk64.
|
|
|
|
// FIXME: Look at LLVMUnhandledExceptionFilter in Signals.inc
|
2012-05-31 22:35:53 +08:00
|
|
|
size_t cs_ret = CaptureStackBackTrace(1, max_size, tmp, 0),
|
2012-02-10 01:20:14 +08:00
|
|
|
offset = 0;
|
|
|
|
// Skip the RTL frames by searching for the PC in the stacktrace.
|
|
|
|
// FIXME: this doesn't work well for the malloc/free stacks yet.
|
|
|
|
for (size_t 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
size = cs_ret - offset;
|
|
|
|
for (size_t i = 0; i < size; i++)
|
2012-05-31 22:35:53 +08:00
|
|
|
trace[i] = (uptr)tmp[i + offset];
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
2012-05-21 22:25:36 +08:00
|
|
|
bool __asan_WinSymbolize(const void *addr, char *out_buffer, int buffer_size) {
|
2012-02-10 01:20:14 +08:00
|
|
|
ScopedLock lock(&dbghelp_lock);
|
|
|
|
if (!dbghelp_initialized) {
|
|
|
|
SymSetOptions(SYMOPT_DEFERRED_LOADS |
|
|
|
|
SYMOPT_UNDNAME |
|
|
|
|
SYMOPT_LOAD_LINES);
|
2012-05-31 22:35:53 +08:00
|
|
|
CHECK(SymInitialize(GetCurrentProcess(), 0, TRUE));
|
2012-02-10 01:20:14 +08:00
|
|
|
// 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 += SNPrintf(out_buffer + written, buffer_size - written,
|
|
|
|
" %s %s:%d", symbol->Name,
|
|
|
|
info.FileName, info.LineNumber);
|
|
|
|
} else {
|
|
|
|
written += SNPrintf(out_buffer + written, buffer_size - written,
|
|
|
|
" %s+0x%p", symbol->Name, offset);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------- AsanLock ---------------- {{{1
|
|
|
|
enum LockState {
|
|
|
|
LOCK_UNINITIALIZED = 0,
|
|
|
|
LOCK_READY = -1,
|
|
|
|
};
|
|
|
|
|
|
|
|
AsanLock::AsanLock(LinkerInitialized li) {
|
|
|
|
// FIXME: see comments in AsanLock::Lock() for the details.
|
|
|
|
CHECK(li == LINKER_INITIALIZED || owner_ == LOCK_UNINITIALIZED);
|
|
|
|
|
|
|
|
CHECK(sizeof(CRITICAL_SECTION) <= sizeof(opaque_storage_));
|
|
|
|
InitializeCriticalSection((LPCRITICAL_SECTION)opaque_storage_);
|
|
|
|
owner_ = LOCK_READY;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsanLock::Lock() {
|
|
|
|
if (owner_ == LOCK_UNINITIALIZED) {
|
|
|
|
// FIXME: hm, global AsanLock objects are not initialized?!?
|
|
|
|
// This might be a side effect of the clang+cl+link Frankenbuild...
|
|
|
|
new(this) AsanLock((LinkerInitialized)(LINKER_INITIALIZED + 1));
|
|
|
|
|
|
|
|
// FIXME: If it turns out the linker doesn't invoke our
|
|
|
|
// constructors, we should probably manually Lock/Unlock all the global
|
|
|
|
// locks while we're starting in one thread to avoid double-init races.
|
|
|
|
}
|
|
|
|
EnterCriticalSection((LPCRITICAL_SECTION)opaque_storage_);
|
|
|
|
CHECK(owner_ == LOCK_READY);
|
|
|
|
owner_ = GetThreadSelf();
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsanLock::Unlock() {
|
|
|
|
CHECK(owner_ == GetThreadSelf());
|
|
|
|
owner_ = LOCK_READY;
|
|
|
|
LeaveCriticalSection((LPCRITICAL_SECTION)opaque_storage_);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------- 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
|
|
|
|
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-02-14 01:09:40 +08:00
|
|
|
bool AsanShadowRangeIsAvailable() {
|
|
|
|
// FIXME: shall we do anything here on Windows?
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
int AtomicInc(int *a) {
|
|
|
|
return InterlockedExchangeAdd((LONG*)a, 1) + 1;
|
|
|
|
}
|
|
|
|
|
2012-05-31 23:02:07 +08:00
|
|
|
u16 AtomicExchange(u16 *a, u16 new_val) {
|
2012-04-06 01:16:32 +08:00
|
|
|
// InterlockedExchange16 seems unavailable on some MSVS installations.
|
|
|
|
// Everybody stand back, I pretend to know inline assembly!
|
|
|
|
// FIXME: I assume VC is smart enough to save/restore eax/ecx?
|
|
|
|
__asm {
|
|
|
|
mov eax, a
|
|
|
|
mov cx, new_val
|
2012-04-06 02:31:50 +08:00
|
|
|
xchg [eax], cx ; NOLINT
|
2012-04-06 01:16:32 +08:00
|
|
|
mov new_val, cx
|
|
|
|
}
|
|
|
|
return new_val;
|
2012-04-05 23:55:09 +08:00
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
const char* AsanGetEnv(const char* name) {
|
2012-03-14 00:12:03 +08:00
|
|
|
static char env_buffer[32767] = {};
|
|
|
|
|
|
|
|
// Note: this implementation stores the result in a static buffer so we only
|
|
|
|
// allow it to be called just once.
|
|
|
|
static bool called_once = false;
|
|
|
|
if (called_once)
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
called_once = true;
|
|
|
|
|
|
|
|
DWORD rv = GetEnvironmentVariableA(name, env_buffer, sizeof(env_buffer));
|
|
|
|
if (rv > 0 && rv < sizeof(env_buffer))
|
|
|
|
return env_buffer;
|
2012-05-31 22:35:53 +08:00
|
|
|
return 0;
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
2012-02-22 17:11:55 +08:00
|
|
|
void AsanDumpProcessMap() {
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
int GetPid() {
|
|
|
|
return GetProcessId(GetCurrentProcess());
|
|
|
|
}
|
|
|
|
|
2012-05-31 22:35:53 +08:00
|
|
|
uptr GetThreadSelf() {
|
2012-02-10 01:20:14 +08:00
|
|
|
return GetCurrentThreadId();
|
|
|
|
}
|
|
|
|
|
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.
|
|
|
|
}
|
|
|
|
|
|
|
|
void AsanDisableCoreDumper() {
|
|
|
|
UNIMPLEMENTED();
|
|
|
|
}
|
|
|
|
|
2012-02-14 05:24:29 +08:00
|
|
|
void SleepForSeconds(int seconds) {
|
|
|
|
Sleep(seconds * 1000);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Exit(int exitcode) {
|
|
|
|
_exit(exitcode);
|
2012-02-10 01:20:14 +08:00
|
|
|
}
|
|
|
|
|
2012-04-06 09:27:11 +08:00
|
|
|
void Abort() {
|
|
|
|
abort();
|
2012-05-21 22:25:36 +08:00
|
|
|
_exit(-1); // abort is not NORETURN on Windows.
|
2012-04-06 09:27:11 +08:00
|
|
|
}
|
|
|
|
|
2012-02-22 22:07:06 +08:00
|
|
|
int Atexit(void (*function)(void)) {
|
|
|
|
return atexit(function);
|
|
|
|
}
|
|
|
|
|
2012-05-31 22:35:53 +08:00
|
|
|
void SortArray(uptr *array, size_t size) {
|
2012-03-10 09:30:01 +08:00
|
|
|
std::sort(array, array + size);
|
|
|
|
}
|
|
|
|
|
2012-02-10 01:20:14 +08:00
|
|
|
} // namespace __asan
|
|
|
|
|
|
|
|
#endif // _WIN32
|