// Copyright 2016 The SwiftShader Authors. All Rights Reserved. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef SharedLibrary_hpp #define SharedLibrary_hpp #if defined(_WIN32) #include <Windows.h> #else #include <dlfcn.h> #endif #include <string> void *getLibraryHandle(const char *path); void *loadLibrary(const char *path); void freeLibrary(void *library); void *getProcAddress(void *library, const char *name); template<int n> void *loadLibrary(const std::string &libraryDirectory, const char *(&names)[n], const char *mustContainSymbol = nullptr) { for(const char *libraryName : names) { std::string libraryPath = libraryDirectory + libraryName; void *library = getLibraryHandle(libraryPath.c_str()); if(library) { if(!mustContainSymbol || getProcAddress(library, mustContainSymbol)) { return library; } freeLibrary(library); } } for(const char *libraryName : names) { std::string libraryPath = libraryDirectory + libraryName; void *library = loadLibrary(libraryPath.c_str()); if(library) { if(!mustContainSymbol || getProcAddress(library, mustContainSymbol)) { return library; } freeLibrary(library); } } return nullptr; } #if defined(_WIN32) inline void *loadLibrary(const char *path) { return (void*)LoadLibrary(path); } inline void *getLibraryHandle(const char *path) { HMODULE module = NULL; GetModuleHandleEx(0, path, &module); return (void*)module; } inline void freeLibrary(void *library) { FreeLibrary((HMODULE)library); } inline void *getProcAddress(void *library, const char *name) { return (void*)GetProcAddress((HMODULE)library, name); } inline std::string getModuleDirectory() { static int dummy_symbol = 0; HMODULE module = NULL; GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS, (LPCTSTR)&dummy_symbol, &module); char filename[1024]; if(module && (GetModuleFileName(module, filename, sizeof(filename)) != 0)) { std::string directory(filename); return directory.substr(0, directory.find_last_of("\\/") + 1).c_str(); } else { return ""; } } #else inline void *loadLibrary(const char *path) { return dlopen(path, RTLD_LAZY | RTLD_LOCAL); } inline void *getLibraryHandle(const char *path) { #ifdef __ANDROID__ // bionic doesn't support RTLD_NOLOAD before L return dlopen(path, RTLD_NOW | RTLD_LOCAL); #else void *resident = dlopen(path, RTLD_LAZY | RTLD_NOLOAD | RTLD_LOCAL); if(resident) { return dlopen(path, RTLD_LAZY | RTLD_LOCAL); // Increment reference count } return nullptr; #endif } inline void freeLibrary(void *library) { if(library) { dlclose(library); } } inline void *getProcAddress(void *library, const char *name) { void *symbol = dlsym(library, name); if(!symbol) { const char *reason = dlerror(); // Silence the error (void)reason; } return symbol; } inline std::string getModuleDirectory() { static int dummy_symbol = 0; Dl_info dl_info; if(dladdr(&dummy_symbol, &dl_info) != 0) { std::string directory(dl_info.dli_fname); return directory.substr(0, directory.find_last_of("\\/") + 1).c_str(); } else { return ""; } } #endif #endif // SharedLibrary_hpp