// NeL - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "stdmisc.h"
#include "nel/misc/dynloadlib.h"
#include "nel/misc/path.h"
using namespace std;
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NLMISC
{
NL_LIB_HANDLE nlLoadLibrary(const std::string &libName)
{
NL_LIB_HANDLE res = 0;
#ifdef NL_OS_WINDOWS
res = LoadLibraryW(utf8ToWide(libName));
#elif defined(NL_OS_UNIX)
res = dlopen(libName.c_str(), RTLD_NOW);
#else
# error "You must code nlLoadLibrary() for your platform"
#endif
if(res == 0) nlwarning("Load library '%s' failed: %s", libName.c_str(), NLMISC::formatErrorMessage(NLMISC::getLastError()).c_str());
return res;
}
bool nlFreeLibrary(NL_LIB_HANDLE libHandle)
{
#ifdef NL_OS_WINDOWS
return FreeLibrary(libHandle) > 0;
#elif defined(NL_OS_UNIX)
return dlclose(libHandle) == 0;
#else
# error "You must code nlFreeLibrary() for your platform"
#endif
}
void *nlGetSymbolAddress(NL_LIB_HANDLE libHandle, const std::string &procName)
{
void *res = 0;
#ifdef NL_OS_WINDOWS
res = (void *)GetProcAddress(libHandle, procName.c_str());
#elif defined(NL_OS_UNIX)
res = dlsym(libHandle, procName.c_str());
#else
# error "You must code nlGetProcAddress() for your platform"
#endif
if(res == 0) nlwarning("Getting symbol address of '%s' failed: %s", procName.c_str(), NLMISC::formatErrorMessage(NLMISC::getLastError()).c_str());
return res;
}
// Again some OS specifics stuff
#ifdef NL_OS_WINDOWS
const string nlLibPrefix; // empty
const string nlLibExt(".dll");
#elif defined(NL_OS_MAC)
const string nlLibPrefix("lib");
const string nlLibExt(".dylib");
#elif defined(NL_OS_UNIX)
const string nlLibPrefix("lib");
const string nlLibExt(".so");
#else
# error "You must define the default dynamic lib extention"
#endif
std::vector CLibrary::_LibPaths;
CLibrary::CLibrary (const CLibrary &/* other */)
{
// Nothing to do has it is forbidden.
// Allowing copy require to manage reference count from CLibrary to the module resource.
nlassert(false);
}
CLibrary &CLibrary::operator =(const CLibrary &/* other */)
{
// Nothing to do has it is forbidden.
// Allowing assignment require to manage reference count from CLibrary to the module resource.
nlassert(false);
return *this;
}
std::string CLibrary::makeLibName(const std::string &baseName)
{
return nlLibPrefix+baseName+nlLibSuffix+nlLibExt;
}
std::string CLibrary::cleanLibName(const std::string &decoratedName)
{
// remove path and extension
string ret = CFile::getFilenameWithoutExtension(decoratedName);
if (!nlLibPrefix.empty())
{
// remove prefix
if (ret.find(nlLibPrefix) == 0)
ret = ret.substr(nlLibPrefix.size());
}
if (!nlLibSuffix.empty())
{
// remove suffix
if (ret.substr(ret.size()-nlLibSuffix.size()) == nlLibSuffix)
ret = ret.substr(0, ret.size() - nlLibSuffix.size());
}
return ret;
}
void CLibrary::addLibPaths(const std::vector &paths)
{
for (uint i=0; i(entryPoint));
// call the private initialization method.
_PureNelLibrary->_onLibraryLoaded(INelContext::getInstance());
}
}
return _LibHandle != NULL;
}
void CLibrary::freeLibrary()
{
if (_LibHandle)
{
nlassert(_Ownership);
if (_PureNelLibrary)
{
// call the private finalization method.
_PureNelLibrary->_onLibraryUnloaded();
}
nldebug("Freeing dynamic library '%s'", _LibFileName.c_str());
nlFreeLibrary(_LibHandle);
_PureNelLibrary = NULL;
_LibHandle = NULL;
_Ownership = false;
_LibFileName.clear();
}
}
void *CLibrary::getSymbolAddress(const std::string &symbolName)
{
nlassert(_LibHandle != NULL);
return nlGetSymbolAddress(_LibHandle, symbolName);
}
bool CLibrary::isLibraryLoaded()
{
return _LibHandle != NULL;
}
bool CLibrary::isLibraryPure()
{
return _LibHandle != NULL && _PureNelLibrary != NULL;
}
INelLibrary *CLibrary::getNelLibraryInterface()
{
if (!isLibraryPure())
return NULL;
return _PureNelLibrary;
}
INelLibrary::~INelLibrary()
{
// cleanup ram
if (_LibContext != NULL)
delete _LibContext;
}
void INelLibrary::_onLibraryLoaded(INelContext &nelContext)
{
++_LoadingCounter;
if (_LoadingCounter == 1)
{
// Linux relocates all symbols, so this is unnecessary.
#ifdef NL_OS_WINDOWS
// initialize NeL context
nlassert(!NLMISC::INelContext::isContextInitialised());
#endif // NL_OS_WINDOWS
_LibContext = new CLibraryContext(nelContext);
}
else
{
nlassert(NLMISC::INelContext::isContextInitialised());
}
onLibraryLoaded(_LoadingCounter==1);
}
void INelLibrary::_onLibraryUnloaded()
{
nlassert(_LoadingCounter > 0);
onLibraryUnloaded(_LoadingCounter == 1);
--_LoadingCounter;
}
uint32 INelLibrary::getLoadingCounter()
{
return _LoadingCounter;
}
} // namespace NLMISC