|  | // 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 |