blob: 0f2c0ce69844d4d30f5c7e39dc0c79fff12981b8 [file] [log] [blame]
// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
#include "vm/globals.h"
#if defined(HOST_OS_WINDOWS)
#include "vm/lockers.h"
#include "vm/native_symbol.h"
#include "vm/os.h"
#include "vm/os_thread.h"
#include <dbghelp.h> // NOLINT
namespace dart {
static bool running_ = false;
static Mutex* lock_ = NULL;
void NativeSymbolResolver::Init() {
ASSERT(running_ == false);
if (lock_ == NULL) {
lock_ = new Mutex();
}
running_ = true;
SymSetOptions(SYMOPT_UNDNAME | SYMOPT_DEFERRED_LOADS);
HANDLE hProcess = GetCurrentProcess();
if (!SymInitialize(hProcess, NULL, TRUE)) {
DWORD error = GetLastError();
OS::PrintErr("Failed to init NativeSymbolResolver (SymInitialize %" Pu32
")\n",
error);
return;
}
}
void NativeSymbolResolver::Cleanup() {
MutexLocker lock(lock_);
if (!running_) {
return;
}
running_ = false;
HANDLE hProcess = GetCurrentProcess();
if (!SymCleanup(hProcess)) {
DWORD error = GetLastError();
OS::PrintErr("Failed to shutdown NativeSymbolResolver (SymCleanup %" Pu32
")\n",
error);
}
}
char* NativeSymbolResolver::LookupSymbolName(uword pc, uword* start) {
static const intptr_t kMaxNameLength = 2048;
static const intptr_t kSymbolInfoSize = sizeof(SYMBOL_INFO); // NOLINT.
static char buffer[kSymbolInfoSize + kMaxNameLength];
static char name_buffer[kMaxNameLength];
MutexLocker lock(lock_);
if (!running_) {
return NULL;
}
if (start != NULL) {
*start = NULL;
}
memset(&buffer[0], 0, sizeof(buffer));
HANDLE hProcess = GetCurrentProcess();
DWORD64 address = static_cast<DWORD64>(pc);
PSYMBOL_INFO pSymbol = reinterpret_cast<PSYMBOL_INFO>(&buffer[0]);
pSymbol->SizeOfStruct = kSymbolInfoSize;
pSymbol->MaxNameLen = kMaxNameLength;
DWORD64 displacement;
BOOL r = SymFromAddr(hProcess, address, &displacement, pSymbol);
if (r == FALSE) {
return NULL;
}
if (start != NULL) {
*start = pc - displacement;
}
return Utils::StrDup(pSymbol->Name);
}
void NativeSymbolResolver::FreeSymbolName(char* name) {
free(name);
}
bool NativeSymbolResolver::LookupSharedObject(uword pc,
uword* dso_base,
char** dso_name) {
return false;
}
void NativeSymbolResolver::AddSymbols(const char* dso_name,
void* buffer,
size_t size) {
OS::PrintErr("warning: Dart_AddSymbols has no effect on Windows\n");
}
} // namespace dart
#endif // defined(HOST_OS_WINDOWS)