diff options
author | Sadie Powell <sadie@witchery.services> | 2020-02-02 17:07:34 +0000 |
---|---|---|
committer | Sadie Powell <sadie@witchery.services> | 2020-02-02 20:32:49 +0000 |
commit | aed712ba8e087232fcd9f71db4311687a7ce4398 (patch) | |
tree | fe06900c63293fabab60faf863bf97a390a0df20 /src/dynamic.cpp | |
parent | fda43fc0ff5ecf87d877cc341961c9da4affae76 (diff) |
Make loading modules considerably more robust and user friendly.
Diffstat (limited to 'src/dynamic.cpp')
-rw-r--r-- | src/dynamic.cpp | 77 |
1 files changed, 52 insertions, 25 deletions
diff --git a/src/dynamic.cpp b/src/dynamic.cpp index a6f758d33..a3ba43ff2 100644 --- a/src/dynamic.cpp +++ b/src/dynamic.cpp @@ -26,62 +26,89 @@ #include "inspircd.h" - #ifndef _WIN32 -#include <dlfcn.h> -#else -#define dlopen(path, state) (void*)LoadLibraryA(path) -#define dlsym(handle, export) (void*)GetProcAddress((HMODULE)handle, export) -#define dlclose(handle) FreeLibrary((HMODULE)handle) +# include <dlfcn.h> #endif -DLLManager::DLLManager(const char *fname) +/** The extension that dynamic libraries end with. */ +#define DLL_EXTENSION ".so" + +DLLManager::DLLManager(const std::string& name) + : lib(NULL) + , libname(name) { - if (!strstr(fname,".so")) + static size_t extlen = strlen(DLL_EXTENSION); + if (name.length() <= extlen || name.compare(name.length() - extlen, name.length(), DLL_EXTENSION)) { - err = "This doesn't look like a module file to me..."; - h = NULL; + err.assign(name + " is not a module (no " DLL_EXTENSION " extension)"); return; } - h = dlopen(fname, RTLD_NOW|RTLD_LOCAL); - if (!h) - { +#ifdef _WIN32 + lib = LoadLibraryA(name.c_str()); +#else + lib = dlopen(name.c_str(), RTLD_NOW|RTLD_LOCAL); +#endif + + if (!lib) RetrieveLastError(); - } } DLLManager::~DLLManager() { - /* close the library */ - if (h) - dlclose(h); + if (!lib) + return; + +#ifdef _WIN32 + FreeLibrary(lib) +#else + dlclose(lib); +#endif } Module* DLLManager::CallInit() { + const uint32_t* abi = GetSymbol<const uint32_t>(MODULE_STR_ABI); + if (!abi) + { + err.assign(libname + " is not a module (no ABI symbol)"); + return NULL; + } + else if (*abi != MODULE_ABI) + { + const char* version = GetVersion(); + err.assign(InspIRCd::Format("%s was built against %s which is too %s to use with %s", + libname.c_str(), version ? version : "an unknown version", + *abi < MODULE_ABI ? "old" : "new", INSPIRCD_VERSION)); + return NULL; + } + union { void* vptr; Module* (*fptr)(); }; - vptr = GetSymbol(MODULE_INIT_STR); + vptr = GetSymbol(MODULE_STR_INIT); if (!vptr) + { + err.assign(libname + " is not a module (no init symbol)"); return NULL; + } return (*fptr)(); } -void* DLLManager::GetSymbol(const char* name) +void* DLLManager::GetSymbol(const char* name) const { - return h ? dlsym(h, name) : NULL; -} + if (!lib) + return NULL; -std::string DLLManager::GetVersion() -{ - const char* srcver = static_cast<const char*>(GetSymbol("inspircd_src_version")); - return srcver ? srcver : ""; +#if defined _WIN32 + return GetProcAddress(lib, name); +#else + return dlsym(lib, name); +#endif } void DLLManager::RetrieveLastError() |