2010-05-06 00:08:41 +00:00
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
// 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 <http://www.gnu.org/licenses/>.
# ifndef APP_CONTEXT_H
# define APP_CONTEXT_H
# include "nel/misc/types_nl.h"
# include <map>
namespace NLMISC
{
class CLog ;
class CMemDisplayer ;
class CMsgBoxDisplayer ;
/** Interface definition for nel context.
* Any application wide data can be accessed thru this interface .
*
* The NeL context is a mean to allow dynamic library loading in NeL .
* In order to make all NeL application safe , it is mandatory to declare
* a NeL context at startup of any application ( first instruction of the
* main ( ) or WinMain ( ) is good practice ) .
* Note that for NLNET : : IService oriented application , service framwork
* already provide the application context .
*
* \ author Boris ' SoniX ' Boucher
* \ date 2005
*/
class INelContext
{
static INelContext * * _getInstance ( ) ;
public :
/// Access to the context singleton
static INelContext & getInstance ( ) ;
static bool isContextInitialised ( ) ;
virtual ~ INelContext ( ) ;
//@name Singleton registry
//@{
/** Return the pointer associated to a given singleton name
* If the name is not present , return NULL .
*/
virtual void * getSingletonPointer ( const std : : string & singletonName ) = 0 ;
/** Register a singleton pointer.
*/
virtual void setSingletonPointer ( const std : : string & singletonName , void * ptr ) = 0 ;
/** Release a singleton pointer */
virtual void releaseSingletonPointer ( const std : : string & singletonName , void * ptr ) = 0 ;
//@}
//@name Global debugging object
//@{
virtual CLog * getErrorLog ( ) = 0 ;
virtual void setErrorLog ( CLog * errorLog ) = 0 ;
virtual CLog * getWarningLog ( ) = 0 ;
virtual void setWarningLog ( CLog * warningLog ) = 0 ;
virtual CLog * getInfoLog ( ) = 0 ;
virtual void setInfoLog ( CLog * infoLog ) = 0 ;
virtual CLog * getDebugLog ( ) = 0 ;
virtual void setDebugLog ( CLog * debugLog ) = 0 ;
virtual CLog * getAssertLog ( ) = 0 ;
virtual void setAssertLog ( CLog * assertLog ) = 0 ;
virtual CMemDisplayer * getDefaultMemDisplayer ( ) = 0 ;
virtual void setDefaultMemDisplayer ( CMemDisplayer * memDisplayer ) = 0 ;
virtual CMsgBoxDisplayer * getDefaultMsgBoxDisplayer ( ) = 0 ;
virtual void setDefaultMsgBoxDisplayer ( CMsgBoxDisplayer * msgBoxDisplayer ) = 0 ;
virtual bool getDebugNeedAssert ( ) = 0 ;
virtual void setDebugNeedAssert ( bool needAssert ) = 0 ;
virtual bool getNoAssert ( ) = 0 ;
virtual void setNoAssert ( bool noAssert ) = 0 ;
virtual bool getAlreadyCreateSharedAmongThreads ( ) = 0 ;
virtual void setAlreadyCreateSharedAmongThreads ( bool b ) = 0 ;
//@}
protected :
/// Called by derived class to finalize initialisation of context
void contextReady ( ) ;
static INelContext * _NelContext ;
} ;
/** This class implement the context interface for the application module
* That means that this class will really hold the data .
* \ author Boris ' SoniX ' Boucher
* \ date 2005
*/
class CApplicationContext : public INelContext
{
public :
CApplicationContext ( ) ;
virtual void * getSingletonPointer ( const std : : string & singletonName ) ;
virtual void setSingletonPointer ( const std : : string & singletonName , void * ptr ) ;
virtual void releaseSingletonPointer ( const std : : string & singletonName , void * ptr ) ;
virtual CLog * getErrorLog ( ) ;
virtual void setErrorLog ( CLog * errorLog ) ;
virtual CLog * getWarningLog ( ) ;
virtual void setWarningLog ( CLog * warningLog ) ;
virtual CLog * getInfoLog ( ) ;
virtual void setInfoLog ( CLog * infoLog ) ;
virtual CLog * getDebugLog ( ) ;
virtual void setDebugLog ( CLog * debugLog ) ;
virtual CLog * getAssertLog ( ) ;
virtual void setAssertLog ( CLog * assertLog ) ;
virtual CMemDisplayer * getDefaultMemDisplayer ( ) ;
virtual void setDefaultMemDisplayer ( CMemDisplayer * memDisplayer ) ;
virtual CMsgBoxDisplayer * getDefaultMsgBoxDisplayer ( ) ;
virtual void setDefaultMsgBoxDisplayer ( CMsgBoxDisplayer * msgBoxDisplayer ) ;
virtual bool getDebugNeedAssert ( ) ;
virtual void setDebugNeedAssert ( bool needAssert ) ;
virtual bool getNoAssert ( ) ;
virtual void setNoAssert ( bool noAssert ) ;
virtual bool getAlreadyCreateSharedAmongThreads ( ) ;
virtual void setAlreadyCreateSharedAmongThreads ( bool b ) ;
private :
/// Singleton registry
typedef std : : map < std : : string , void * > TSingletonRegistry ;
TSingletonRegistry _SingletonRegistry ;
CLog * ErrorLog ;
CLog * WarningLog ;
CLog * InfoLog ;
CLog * DebugLog ;
CLog * AssertLog ;
CMemDisplayer * DefaultMemDisplayer ;
CMsgBoxDisplayer * DefaultMsgBoxDisplayer ;
bool DebugNeedAssert ;
bool NoAssert ;
bool AlreadyCreateSharedAmongThreads ;
} ;
/** This class implements the context interface for the a library module.
* All it contains is forward call to the application context instance .
* \ author Boris ' SoniX ' Boucher
* \ date 2005
*/
class CLibraryContext : public INelContext
{
public :
CLibraryContext ( INelContext & applicationContext ) ;
virtual void * getSingletonPointer ( const std : : string & singletonName ) ;
virtual void setSingletonPointer ( const std : : string & singletonName , void * ptr ) ;
virtual void releaseSingletonPointer ( const std : : string & singletonName , void * ptr ) ;
virtual CLog * getErrorLog ( ) ;
virtual void setErrorLog ( CLog * errorLog ) ;
virtual CLog * getWarningLog ( ) ;
virtual void setWarningLog ( CLog * warningLog ) ;
virtual CLog * getInfoLog ( ) ;
virtual void setInfoLog ( CLog * infoLog ) ;
virtual CLog * getDebugLog ( ) ;
virtual void setDebugLog ( CLog * debugLog ) ;
virtual CLog * getAssertLog ( ) ;
virtual void setAssertLog ( CLog * assertLog ) ;
virtual CMemDisplayer * getDefaultMemDisplayer ( ) ;
virtual void setDefaultMemDisplayer ( CMemDisplayer * memDisplayer ) ;
virtual CMsgBoxDisplayer * getDefaultMsgBoxDisplayer ( ) ;
virtual void setDefaultMsgBoxDisplayer ( CMsgBoxDisplayer * msgBoxDisplayer ) ;
virtual bool getDebugNeedAssert ( ) ;
virtual void setDebugNeedAssert ( bool needAssert ) ;
virtual bool getNoAssert ( ) ;
virtual void setNoAssert ( bool noAssert ) ;
virtual bool getAlreadyCreateSharedAmongThreads ( ) ;
virtual void setAlreadyCreateSharedAmongThreads ( bool b ) ;
private :
/// Pointer to the application context.
2010-12-20 15:25:23 +00:00
INelContext * _ApplicationContext ;
2010-05-06 00:08:41 +00:00
} ;
//@name Singleton utility
//@{
/** Some utility macro to build singleton compatible with
* the dynamic loading of library
* This macro must be put inside the singleton class
* definition .
* Warning : this macro change the current access right , it end up with
* private access right .
*/
# define NLMISC_SAFE_SINGLETON_DECL(className) \
private : \
/* declare private constructors*/ \
/*className () {}*/ \
className ( const className & ) { } \
/* the local static pointer to the singleton instance */ \
static className * _Instance ; \
public : \
static className & getInstance ( ) \
{ \
if ( _Instance = = NULL ) \
{ \
/* the nel context MUST be initialised */ \
nlassertex ( NLMISC : : INelContext : : isContextInitialised ( ) , ( " You are trying to access a safe singleton without having initialized a NeL context. The simplest correction is to add 'NLMISC::CApplicationContext myApplicationContext;' at the very beginning of your application. " ) ) ; \
void * ptr = NLMISC : : INelContext : : getInstance ( ) . getSingletonPointer ( # className ) ; \
if ( ptr = = NULL ) \
{ \
/* allocate the singleton and register it */ \
_Instance = new className ; \
NLMISC : : INelContext : : getInstance ( ) . setSingletonPointer ( # className , _Instance ) ; \
} \
else \
{ \
_Instance = reinterpret_cast < className * > ( ptr ) ; \
} \
} \
return * _Instance ; \
} \
private :
/** The same as above, but generate a getInstance method that
* return a pointer instead of a reference
*/
# define NLMISC_SAFE_SINGLETON_DECL_PTR(className) \
private : \
/* declare private constructors*/ \
/*className () {}*/ \
className ( const className & ) { } \
/* the local static pointer to the singleton instance */ \
static className * _Instance ; \
public : \
static className * getInstance ( ) \
{ \
if ( _Instance = = NULL ) \
{ \
/* the nel context MUST be initialised */ \
nlassertex ( NLMISC : : INelContext : : isContextInitialised ( ) , ( " You are trying to access a safe singleton without having initialized a NeL context. The simplest correction is to add 'NLMISC::CApplicationContext myApplicationContext;' at the very beginning of your application. " ) ) ; \
void * ptr = NLMISC : : INelContext : : getInstance ( ) . getSingletonPointer ( # className ) ; \
if ( ptr = = NULL ) \
{ \
/* allocate the singleton and register it */ \
_Instance = new className ; \
NLMISC : : INelContext : : getInstance ( ) . setSingletonPointer ( # className , _Instance ) ; \
} \
else \
{ \
_Instance = reinterpret_cast < className * > ( ptr ) ; \
} \
} \
return _Instance ; \
} \
private :
/** This macro is the complement of the previous one.
* It must be put in a cpp file to implement the static
* property of the singleton .
*/
# define NLMISC_SAFE_SINGLETON_IMPL(className) className *className::_Instance = NULL;
/// Function type for library entry point
typedef void ( * TInitLibraryFunc ) ( INelContext & applicationContext ) ;
/** An helper macro to build the dll entry point easily.
*/
# define NLMISC_LIBRARY_ENTRY \
void libraryEntryImp ( NLMISC : : INelContext & applicationContext ) \
{ \
nlassert ( ! NLMISC : : INelContext : : isContextInitialised ( ) | | & applicationContext = = & ( NLMISC : : INelContext : : getInstance ( ) ) ) ; \
\
if ( ! NLMISC : : INelContext : : isContextInitialised ( ) ) \
{ \
new CLibraryContext ( applicationContext ) ; \
} \
} \
extern " C " \
{ \
NLMISC_LIB_EXPORT TInitLibraryFunc libraryEntry = libraryEntryImp ; \
} \
class CLibrary ;
/// helper function to init newly loaded nel library
void initNelLibrary ( CLibrary & lib ) ;
} // namespace NLMISC
# endif //APP_CONTEXT_H