X-Git-Url: https://git.netwichtig.de/gitweb/?a=blobdiff_plain;f=src%2Fdynamic.cpp;h=3a6a151cb5d047db124b4caca75d1a8063c0a1bf;hb=571714e28b26cc59cbc8d27098a5ba981240ee2d;hp=9cc68fc5f3d36aa7a9169ca482e9eda102a16f33;hpb=9f1bd74b09810565502451047b06b6eca7e47f7f;p=user%2Fhenk%2Fcode%2Finspircd.git diff --git a/src/dynamic.cpp b/src/dynamic.cpp index 9cc68fc5f..3a6a151cb 100644 --- a/src/dynamic.cpp +++ b/src/dynamic.cpp @@ -1,79 +1,111 @@ -/* +------------------------------------+ - * | Inspire Internet Relay Chat Daemon | - * +------------------------------------+ +/* + * InspIRCd -- Internet Relay Chat Daemon * - * Inspire is copyright (C) 2002-2004 ChatSpike-Dev. - * E-mail: - * - * - * - * Written by Craig Edwards, Craig McLure, and others. - * This program is free but copyrighted software; see - * the file COPYING for details. + * Copyright (C) 2009 Daniel De Graaf + * Copyright (C) 2007 Oliver Lupton + * Copyright (C) 2007 Robin Burchell + * Copyright (C) 2007 Dennis Friis + * Copyright (C) 2003, 2006 Craig Edwards * - * --------------------------------------------------- + * This file is part of InspIRCd. InspIRCd is free software: you can + * redistribute it and/or modify it under the terms of the GNU General Public + * License as published by the Free Software Foundation, version 2. + * + * This program is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS + * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more + * details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . */ -#include "globals.h" -#include + +#include "inspircd.h" #include "dynamic.h" -#include "inspstring.h" +#ifndef _WIN32 +#include +#else +#define dlopen(path, state) (void*)LoadLibraryA(path) +#define dlsym(handle, export) (void*)GetProcAddress((HMODULE)handle, export) +#define dlclose(handle) FreeLibrary((HMODULE)handle) +#endif DLLManager::DLLManager(const char *fname) { - // Try to open the library now and get any error message. - - h=dlopen( fname, RTLD_NOW ); - err=dlerror(); + if (!strstr(fname,".so")) + { + err = "This doesn't look like a module file to me..."; + h = NULL; + return; + } + + h = dlopen(fname, RTLD_NOW|RTLD_LOCAL); + if (!h) + { +#ifdef _WIN32 + RetrieveLastError(); +#else + err = dlerror(); +#endif + } } DLLManager::~DLLManager() { - // close the library if it isn't null - if (h!=0) - dlclose(h); + /* close the library */ + if (h) + dlclose(h); } +union init_t { + void* vptr; + Module* (*fptr)(); +}; -bool DLLManager::GetSymbol(void **v, const char *sym_name) +Module* DLLManager::CallInit() { - // try extract a symbol from the library - // get any error message is there is any - - if( h!=0 ) + if (!h) + return NULL; + + init_t initfn; + initfn.vptr = dlsym(h, MODULE_INIT_STR); + if (!initfn.vptr) { - *v = dlsym( h, sym_name ); - err=dlerror(); - if( err==0 ) - return true; - else - return false; +#ifdef _WIN32 + RetrieveLastError(); +#else + err = dlerror(); +#endif + return NULL; } - else - { - return false; - } - -} - -DLLFactoryBase::DLLFactoryBase(const char *fname, const char *factory) : DLLManager(fname) -{ - // try get the factory function if there is no error yet - - factory_func=0; - - if( LastError()==0 ) - { - GetSymbol( (void **)&factory_func, factory ? factory : "init_module" ); - } - + return (*initfn.fptr)(); } - -DLLFactoryBase::~DLLFactoryBase() +std::string DLLManager::GetVersion() { -} + if (!h) + return ""; + const char* srcver = (char*)dlsym(h, "inspircd_src_version"); + if (srcver) + return srcver; + return "Unversioned module"; +} +#ifdef _WIN32 +void DLLManager::RetrieveLastError() +{ + char errmsg[500]; + DWORD dwErrorCode = GetLastError(); + if (FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, dwErrorCode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPSTR)errmsg, _countof(errmsg), NULL) == 0) + sprintf_s(errmsg, _countof(errmsg), "Error code: %u", dwErrorCode); + SetLastError(ERROR_SUCCESS); + err = errmsg; + std::string::size_type p; + while ((p = err.find_last_of("\r\n")) != std::string::npos) + err.erase(p, 1); +} +#endif