blob: f3287121b2e267e0bcdf130049e0013d722f5373 [file] [log] [blame]
#include <c10/util/Exception.h>
#include <c10/util/Unicode.h>
#include <ATen/DynamicLibrary.h>
#include <ATen/Utils.h>
#ifndef _WIN32
#include <dlfcn.h>
#include <libgen.h>
#else
#include <c10/util/win32-headers.h>
#endif
namespace at {
#ifndef C10_MOBILE
#ifndef _WIN32
// Unix
static void* checkDL(void* x) {
if (!x) {
TORCH_CHECK_WITH(DynamicLibraryError, false, "Error in dlopen or dlsym: ", dlerror());
}
return x;
}
DynamicLibrary::DynamicLibrary(const char* name, const char* alt_name, bool leak_handle_): leak_handle(leak_handle_) {
// NOLINTNEXTLINE(hicpp-signed-bitwise)
handle = dlopen(name, RTLD_LOCAL | RTLD_NOW);
if (!handle) {
if (alt_name) {
handle = dlopen(alt_name, RTLD_LOCAL | RTLD_NOW);
if (!handle) {
TORCH_CHECK_WITH(DynamicLibraryError, false, "Error in dlopen for library ", name, "and ", alt_name);
}
} else {
TORCH_CHECK_WITH(DynamicLibraryError, false, "Error in dlopen: ", dlerror());
}
}
}
void* DynamicLibrary::sym(const char* name) {
AT_ASSERT(handle);
return checkDL(dlsym(handle, name));
}
DynamicLibrary::~DynamicLibrary() {
if (!handle || leak_handle) {
return;
}
dlclose(handle);
}
#else
// Windows
DynamicLibrary::DynamicLibrary(const char* name, const char* alt_name, bool leak_handle_): leak_handle(leak_handle_) {
// NOLINTNEXTLINE(hicpp-signed-bitwise)
HMODULE theModule;
bool reload = true;
auto wname = c10::u8u16(name);
// Check if LOAD_LIBRARY_SEARCH_DEFAULT_DIRS is supported
if (GetProcAddress(GetModuleHandleW(L"KERNEL32.DLL"), "AddDllDirectory") != NULL) {
theModule = LoadLibraryExW(
wname.c_str(),
NULL,
LOAD_LIBRARY_SEARCH_DEFAULT_DIRS);
if (theModule != NULL || (GetLastError() != ERROR_MOD_NOT_FOUND)) {
reload = false;
}
}
if (reload) {
theModule = LoadLibraryW(wname.c_str());
}
if (theModule) {
handle = theModule;
} else {
char buf[256];
DWORD dw = GetLastError();
FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
NULL, dw, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
buf, (sizeof(buf) / sizeof(char)), NULL);
TORCH_CHECK_WITH(DynamicLibraryError, false, "error in LoadLibrary for ", name, ". WinError ", dw, ": ", buf);
}
}
void* DynamicLibrary::sym(const char* name) {
AT_ASSERT(handle);
FARPROC procAddress = GetProcAddress((HMODULE)handle, name);
if (!procAddress) {
TORCH_CHECK_WITH(DynamicLibraryError, false, "error in GetProcAddress");
}
return (void*)procAddress;
}
DynamicLibrary::~DynamicLibrary() {
if (!handle || leak_handle) {
return;
}
FreeLibrary((HMODULE)handle);
}
#endif
#endif
} // namespace at