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/>.
# include "stdnet.h"
//
// Includes
//
# ifdef NL_OS_WINDOWS
// these defines is for IsDebuggerPresent(). it'll not compile on windows 95
// just comment this and the IsDebuggerPresent to compile on windows 95
# define _WIN32_WINDOWS 0x0410
# define WINVER 0x0400
# define NOMINMAX
# include <windows.h>
# include <direct.h>
# elif defined NL_OS_UNIX
# include <unistd.h>
# endif
# include <cstdlib>
# include <csignal>
# include <ctime>
# include "nel/misc/config_file.h"
# include "nel/misc/displayer.h"
# include "nel/misc/mutex.h"
# include "nel/misc/window_displayer.h"
# include "nel/misc/gtk_displayer.h"
# include "nel/misc/win_displayer.h"
# include "nel/misc/path.h"
# include "nel/misc/hierarchical_timer.h"
# include "nel/misc/report.h"
# include "nel/misc/system_info.h"
# include "nel/misc/timeout_assertion_thread.h"
# include "nel/net/naming_client.h"
# include "nel/net/service.h"
# include "nel/net/unified_network.h"
# include "nel/net/net_displayer.h"
# include "nel/net/email.h"
# include "nel/net/varpath.h"
# include "nel/net/admin.h"
# include "nel/net/module_manager.h"
# include "nel/net/transport_class.h"
# include "stdin_monitor_thread.h"
//
// Namespaces
//
using namespace std ;
using namespace NLMISC ;
namespace NLNET
{
//
// Constants
//
static const sint Signal [ ] = {
SIGABRT , SIGFPE , SIGILL , SIGINT , SIGSEGV , SIGTERM
} ;
static const char * SignalName [ ] =
{
" SIGABRT " , " SIGFPE " , " SIGILL " , " SIGINT " , " SIGSEGV " , " SIGTERM "
} ;
static const char * NegFiltersNames [ ] =
{
" NegFiltersDebug " ,
" NegFiltersInfo " ,
" NegFiltersWarning " ,
" NegFiltersAssert " ,
" NegFiltersError " ,
0
} ;
//
// Variables
//
TUnifiedCallbackItem EmptyCallbackArray [ 1 ] = { { " " , NULL } } ;
// class static member
IService * IService : : _Instance = NULL ;
static sint ExitSignalAsked = 0 ;
// services stat
CVariable < sint32 > UserSpeedLoop ( " nel " , " UserSpeedLoop " , " duration of the last user loop (in ms) " , 10 , false ) ;
CVariable < sint32 > NetSpeedLoop ( " nel " , " NetSpeedLoop " , " duration of the last network loop (in ms) " , 10 , false ) ;
/// The time passed in callback during the loop
CVariable < uint32 > L5CallbackTime ( " nel " , " L5CallbackTime " , " Time passed in the L5 callback function in the last loop (in ms) " , 0 , 100 ) ;
/// The number of L5 callback treated
CVariable < uint32 > L5CallbackCount ( " nel " , " L5CallbackCount " , " The number of layer 5 callback received in the last loop " , 0 , 100 ) ;
extern uint32 TotalCallbackCalled ;
extern uint32 TimeInCallback ;
uint32 LastTotalCallbackCalled = 0 ;
uint32 LastTimeInCallback = 0 ;
// this is the thread that initialized the signal redirection
// we'll ignore other thread signals
2010-05-13 20:23:34 +00:00
static size_t SignalisedThread ;
2010-05-06 00:08:41 +00:00
static CFileDisplayer fd ;
static CNetDisplayer commandDisplayer ( false ) ;
//static CLog commandLog;
static string CompilationDate ;
static uint32 LaunchingDate ;
static uint32 NbUserUpdate = 0 ;
string CompilationMode = nlMode ;
//static bool Bench = false;
CVariable < bool > Bench ( " nel " , " Bench " , " 1 if benching 0 if not " , 0 , true ) ;
// This produce an assertion in the thread if the update loop is too slow
static CTimeoutAssertionThread MyTAT ;
static void UpdateAssertionThreadTimeoutCB ( IVariable & var ) { uint32 timeOut ; fromString ( var . toString ( ) , timeOut ) ; MyTAT . timeout ( timeOut ) ; }
static CVariable < uint32 > UpdateAssertionThreadTimeout ( " nel " , " UpdateAssertionThreadTimeout " , " in millisecond, timeout before thread assertion " , 0 , 0 , true , UpdateAssertionThreadTimeoutCB ) ;
// Flag to enable/disable the flushing of the sending queues when the service is shut down
// Default: false (matches the former behavior)
// Set it to true in services that need to send data on exit (for instance in their release() method)
CVariable < bool > FlushSendingQueuesOnExit ( " nel " , " FlushSendingQueuesOnExit " ,
" Flag to enable/disable the flushing of the sending queues when the service is shut down " , false , 0 , true ) ;
// If FlushSendingQueuesOnExit is on, only the sending queues to these specified services will be flushed
// Format: service short names separated by ':'
// Default: "" (all will be flushed if FlushSendingQueuesOnExit is on, none if it is off)
CVariable < string > NamesOfOnlyServiceToFlushSending ( " nel " , " NamesOfOnlyServiceToFlushSending " ,
" If FlushSendingQueuesOnExit is on, only the sending queues to these specified services will be flushed (ex: \" WS:LS \" ; all will be flushed if empty string) " , " " , 0 , true ) ;
//
// Signals managing
//
// This function is called when a signal comes
static void sigHandler ( int Sig )
{
// redirect the signal for the next time
signal ( Sig , sigHandler ) ;
// find the signal
for ( int i = 0 ; i < ( int ) ( sizeof ( Signal ) / sizeof ( Signal [ 0 ] ) ) ; i + + )
{
if ( Sig = = Signal [ i ] )
{
if ( getThreadId ( ) ! = SignalisedThread )
{
nldebug ( " SERVICE: Not the main thread (%u, %u) received the signal (%s, %d), ignore it " , getThreadId ( ) , SignalisedThread , SignalName [ i ] , Sig ) ;
return ;
}
else
{
nlinfo ( " SERVICE: Signal %s (%d) received " , SignalName [ i ] , Sig ) ;
switch ( Sig )
{
// Note: SIGKILL (9) and SIGSTOP (19) can't be trapped
case SIGINT :
if ( IService : : getInstance ( ) - > haveLongArg ( " nobreak " ) )
{
// ignore ctrl-c
nlinfo ( " SERVICE: Ignoring ctrl-c " ) ;
return ;
}
case SIGABRT :
case SIGILL :
case SIGSEGV :
case SIGTERM :
// you should not call a function and system function like printf in a SigHandle because
// signal-handler routines are usually called asynchronously when an interrupt occurs.
if ( ExitSignalAsked = = 0 )
{
nlinfo ( " SERVICE: Receive a signal that said that i must exit " ) ;
ExitSignalAsked = Sig ;
return ;
}
else
{
nlinfo ( " SERVICE: Signal already received, launch the brutal exit " ) ;
exit ( EXIT_FAILURE ) ;
}
break ;
}
}
}
}
nlwarning ( " SERVICE: Unknown signal received (%d) " , Sig ) ;
}
// Initialise the signal redirection
static void initSignal ( )
{
SignalisedThread = getThreadId ( ) ;
# ifdef NL_DEBUG
// in debug mode, we only trap the SIGINT signal (for ctrl-c handling)
//signal(Signal[3], sigHandler);
//nldebug("Signal : %s (%d) trapped", SignalName[3], Signal[3]);
# else
// in release, redirect all signals
// don't redirect now because too hard to debug...
// for (int i = 0; i < (int)(sizeof(Signal)/sizeof(Signal[0])); i++)
// {
// signal(Signal[i], sigHandler);
// nldebug("Signal %s (%d) trapped", SignalName[i], Signal[i]);
// }
//
if ( IService : : getInstance ( ) - > haveLongArg ( " nobreak " ) )
{
signal ( Signal [ 3 ] , sigHandler ) ;
}
# endif
}
void cbDirectoryChanged ( IVariable & var )
{
IService * instance = IService : : getInstance ( ) ;
// Convert to full path if required
// (warning: ConvertSavesFilesDirectoryToFullPath, read from the config file, won't be ready for the initial variable assigments done before the config file has been loaded)
string vp = var . toString ( ) ;
if ( ( var . getName ( ) ! = " SaveFilesDirectory " ) | | instance - > ConvertSavesFilesDirectoryToFullPath . get ( ) )
{
vp = CPath : : getFullPath ( vp ) ;
var . fromString ( vp ) ;
}
nlinfo ( " SERVICE: '%s' changed to '%s' " , var . getName ( ) . c_str ( ) , vp . c_str ( ) ) ;
// Update the running directory if needed
if ( var . getName ( ) = = " RunningDirectory " )
{
# ifdef NL_OS_WINDOWS
_chdir ( vp . c_str ( ) ) ;
# else
chdir ( vp . c_str ( ) ) ;
# endif
}
// Call the callback if provided
if ( instance - > _DirectoryChangedCBI )
instance - > _DirectoryChangedCBI - > onVariableChanged ( var ) ;
}
//
// Service built-in callbacks
//
void cbReceiveShardId ( CMessage & msgin , const string & serviceName , TServiceId serviceId )
{
uint32 shardId ;
msgin . serial ( shardId ) ;
if ( IService : : getInstance ( ) - > getDontUseNS ( ) )
{
// we don't use NS, so shard ID message don't concern us
return ;
}
if ( serviceName ! = " WS " )
{
nlwarning ( " SERVICE: received unauthorized R_SH_ID callback from service %s-%uh asking to set ShardId to %d " , serviceName . c_str ( ) , serviceId . get ( ) , shardId ) ;
return ;
}
nlinfo ( " SERVICE: ShardId is %u " , shardId ) ;
IService : : getInstance ( ) - > setShardId ( shardId ) ;
}
std : : string IService : : getServiceStatusString ( ) const
{
static string emptyString ;
return emptyString ;
}
//
void IService : : anticipateShardId ( uint32 shardId )
{
if ( ! ( ( _ShardId = = DEFAULT_SHARD_ID ) | | ( shardId = = _ShardId ) ) )
nlerror ( " IService::anticipateShardId() overwrites %u with %u " , _ShardId , shardId ) ;
_ShardId = shardId ;
}
//
void IService : : setShardId ( uint32 shardId )
{
if ( ! ( ( _ShardId = = DEFAULT_SHARD_ID ) | | ( shardId = = _ShardId ) ) )
nlwarning ( " SERVICE: The shardId from the WS (%u) is different from the anticipated shardId (%u) " , shardId , _ShardId ) ;
_ShardId = shardId ;
}
TUnifiedCallbackItem builtinServiceCallbacks [ ] =
{
{ " R_SH_ID " , cbReceiveShardId } ,
} ;
//
// Class implementation
//
// Ctor
IService : : IService ( ) :
WindowDisplayer ( 0 ) ,
WriteFilesDirectory ( " nel " , " WriteFilesDirectory " , " directory where to save generic shard information (packed_sheets for example) " , " . " , 0 , true , cbDirectoryChanged ) ,
SaveFilesDirectory ( " nel " , " SaveFilesDirectory " , " directory where to save specific shard information (shard time for example) " , " . " , 0 , true , cbDirectoryChanged ) ,
ConvertSavesFilesDirectoryToFullPath ( " nel " , " ConvertSaveFilesDirectoryToFullPath " , " If true (default), the provided SaveFilesDirectory will be converted to a full path (ex: saves -> /home/dir/saves) " , true , 0 , true ) ,
ListeningPort ( " nel " , " ListeningPort " , " listening port for this service " , 0 , 0 , true ) ,
_RecordingState ( CCallbackNetBase : : Off ) ,
_UpdateTimeout ( 100 ) ,
_SId ( 0 ) ,
_ExitStatus ( 0 ) ,
_Initialized ( false ) ,
ConfigDirectory ( " nel " , " ConfigDirectory " , " directory where config files are " , " . " , 0 , true , cbDirectoryChanged ) ,
LogDirectory ( " nel " , " LogDirectory " , " directory where the service is logging " , " . " , 0 , true , cbDirectoryChanged ) ,
RunningDirectory ( " nel " , " RunningDirectory " , " directory where the service is running on " , " . " , 0 , true , cbDirectoryChanged ) ,
Version ( " nel " , " Version " , " Version of the shard " , " " ) ,
_CallbackArray ( 0 ) ,
_CallbackArraySize ( 0 ) ,
_DontUseNS ( false ) ,
_DontUseAES ( false ) ,
_ResetMeasures ( false ) ,
_ShardId ( 0 ) ,
_ClosureClearanceStatus ( CCMustRequestClearance ) ,
_RequestClosureClearanceCallback ( NULL ) ,
_DirectoryChangedCBI ( NULL )
{
// Singleton
_Instance = this ;
// register in the safe singleton registry
INelContext : : getInstance ( ) . setSingletonPointer ( " IService " , this ) ;
}
IService : : ~ IService ( )
{
// unregister the singleton
INelContext : : getInstance ( ) . releaseSingletonPointer ( " IService " , this ) ;
}
bool IService : : haveArg ( char argName ) const
{
for ( uint32 i = 0 ; i < _Args . size ( ) ; i + + )
{
if ( _Args [ i ] . size ( ) > = 2 & & _Args [ i ] [ 0 ] = = ' - ' )
{
if ( _Args [ i ] [ 1 ] = = argName )
{
return true ;
}
}
}
return false ;
}
string IService : : getArg ( char argName ) const
{
for ( uint32 i = 0 ; i < _Args . size ( ) ; i + + )
{
if ( _Args [ i ] . size ( ) > = 2 & & _Args [ i ] [ 0 ] = = ' - ' )
{
if ( _Args [ i ] [ 1 ] = = argName )
{
/* Remove the first and last '"' :
- c " C: \ Documents and Settings \t oto.tmp "
will return :
C : \ Documents and Settings \ toto . tmp
*/
uint begin = 2 ;
if ( _Args [ i ] . size ( ) < 3 )
return " " ;
//throw Exception ("Parameter '-%c' is malformed, missing content", argName);
if ( _Args [ i ] [ begin ] = = ' " ' )
begin + + ;
// End
2010-05-13 20:23:34 +00:00
uint size = ( uint ) _Args [ i ] . size ( ) ;
2010-05-06 00:08:41 +00:00
if ( size & & _Args [ i ] [ size - 1 ] = = ' " ' )
size - - ;
size = ( uint ) ( std : : max ( ( int ) 0 , ( int ) size - ( int ) begin ) ) ;
return _Args [ i ] . substr ( begin , size ) ;
}
}
}
throw Exception ( " Parameter '-%c' is not found in command line " , argName ) ;
}
bool IService : : haveLongArg ( const char * argName ) const
{
for ( uint32 i = 0 ; i < _Args . size ( ) ; i + + )
{
if ( _Args [ i ] . left ( 2 ) = = " -- " & & _Args [ i ] . leftCrop ( 2 ) . splitTo ( ' = ' ) = = argName )
{
return true ;
}
}
return false ;
}
string IService : : getLongArg ( const char * argName ) const
{
for ( uint32 i = 0 ; i < _Args . size ( ) ; i + + )
{
if ( _Args [ i ] . left ( 2 ) = = " -- " & & _Args [ i ] . leftCrop ( 2 ) . splitTo ( ' = ' ) = = argName )
{
NLMISC : : CSString val = _Args [ i ] . splitFrom ( ' = ' ) ;
if ( ! val . empty ( ) )
{
return val . unquoteIfQuoted ( ) ;
}
if ( i + 1 < _Args . size ( ) & & _Args [ i + 1 ] . c_str ( ) [ 0 ] ! = ' - ' )
{
return _Args [ i + 1 ] . unquoteIfQuoted ( ) ;
}
return std : : string ( ) ;
}
}
return std : : string ( ) ;
}
void IService : : setArgs ( const char * args )
{
_Args . push_back ( " <ProgramName> " ) ;
string sargs ( args ) ;
string : : size_type pos1 = 0 , pos2 = 0 ;
do
{
// Look for the first non space character
pos1 = sargs . find_first_not_of ( " " , pos2 ) ;
if ( pos1 = = string : : npos ) break ;
// Look for the first space or "
pos2 = sargs . find_first_of ( " \" " , pos1 ) ;
if ( pos2 ! = string : : npos )
{
// " ?
if ( sargs [ pos2 ] = = ' " ' )
{
// Look for the final \"
pos2 = sargs . find_first_of ( " \" " , pos2 + 1 ) ;
if ( pos2 ! = string : : npos )
{
// Look for the first space
pos2 = sargs . find_first_of ( " " , pos2 + 1 ) ;
}
}
}
// Compute the size of the string to extract
string : : difference_type length = ( pos2 ! = string : : npos ) ? pos2 - pos1 : string : : npos ;
string tmp = sargs . substr ( pos1 , length ) ;
_Args . push_back ( tmp ) ;
}
while ( pos2 ! = string : : npos ) ;
}
void IService : : setArgs ( int argc , const char * * argv )
{
for ( sint i = 0 ; i < argc ; i + + )
{
_Args . push_back ( argv [ i ] ) ;
}
}
void cbLogFilter ( CConfigFile : : CVar & var )
{
CLog * log = NULL ;
if ( var . Name = = " NegFiltersDebug " )
{
log = DebugLog ;
}
else if ( var . Name = = " NegFiltersInfo " )
{
log = InfoLog ;
}
else if ( var . Name = = " NegFiltersWarning " )
{
log = WarningLog ;
}
else if ( var . Name = = " NegFiltersAssert " )
{
log = AssertLog ;
}
else if ( var . Name = = " NegFiltersError " )
{
log = ErrorLog ;
}
else
{
nlstop ;
}
nlinfo ( " SERVICE: Updating %s from config file " , var . Name . c_str ( ) ) ;
// remove all old filters from config file
CConfigFile : : CVar & oldvar = IService : : getInstance ( ) - > ConfigFile . getVar ( var . Name ) ;
for ( uint j = 0 ; j < oldvar . size ( ) ; j + + )
{
log - > removeFilter ( oldvar . asString ( j ) . c_str ( ) ) ;
}
// add all new filters from config file
for ( uint i = 0 ; i < var . size ( ) ; i + + )
{
log - > addNegativeFilter ( var . asString ( i ) . c_str ( ) ) ;
}
}
void cbExecuteCommands ( CConfigFile : : CVar & var )
{
for ( uint i = 0 ; i < var . size ( ) ; i + + )
{
ICommand : : execute ( var . asString ( i ) , IService : : getInstance ( ) - > CommandLog ) ;
}
}
//
// The main function of the service
//
sint IService : : main ( const char * serviceShortName , const char * serviceLongName , uint16 servicePort , const char * configDir , const char * logDir , const char * compilationDate )
{
bool userInitCalled = false ;
CConfigFile : : CVar * var = NULL ;
IThread * timeoutThread = NULL ;
// a short name service can't be a number
uint tmp ;
nlassert ( ! fromString ( serviceShortName , tmp ) ) ;
try
{
createDebug ( ) ;
// init the module manager
IModuleManager : : getInstance ( ) ;
//
// Init parameters
//
// at the very beginning, eventually wrote a file with the pid
if ( haveLongArg ( " writepid " ) )
{
// use legacy C primitives
FILE * fp = fopen ( " pid.state " , " wt " ) ;
if ( fp )
{
fprintf ( fp , " %u " , getpid ( ) ) ;
fclose ( fp ) ;
}
}
_ShortName = serviceShortName ;
CLog : : setProcessName ( _ShortName ) ;
// get the path where to run the service if any in the command line
if ( haveArg ( ' A ' ) )
RunningDirectory = CPath : : standardizePath ( getArg ( ' A ' ) ) ;
ConfigDirectory = CPath : : standardizePath ( configDir ) ;
LogDirectory = CPath : : standardizePath ( logDir ) ;
_LongName = serviceLongName ;
CompilationDate = compilationDate ;
LaunchingDate = CTime : : getSecondsSince1970 ( ) ;
ListeningPort = servicePort ;
setReportEmailFunction ( ( void * ) sendEmail ) ;
// setDefaultEmailParams ("gw.nevrax.com", "", "cado@nevrax.com");
//
// Load the config file
//
// get the config file dir if any in the command line
if ( haveArg ( ' C ' ) )
ConfigDirectory = CPath : : standardizePath ( getArg ( ' C ' ) ) ;
string cfn = ConfigDirectory . c_str ( ) + _LongName + " .cfg " ;
if ( ! CFile : : fileExists ( ConfigDirectory . c_str ( ) + _LongName + " .cfg " ) )
{
// check if the default exists
if ( ! CFile : : fileExists ( ConfigDirectory . c_str ( ) + _LongName + " _default.cfg " ) )
{
nlerror ( " SERVICE: Neither the config file '%s' nor the default one can be found, can't launch the service " , cfn . c_str ( ) ) ;
}
else
{
// create the basic .cfg that link the default one
FILE * fp = fopen ( cfn . c_str ( ) , " w " ) ;
if ( fp = = NULL )
{
nlerror ( " SERVICE: Can't create config file '%s' " , cfn . c_str ( ) ) ;
}
fprintf ( fp , " // link the default config file for %s \n " , _LongName . c_str ( ) ) ;
fprintf ( fp , " RootConfigFilename = \" %s_default.cfg \" ; \n " , _LongName . c_str ( ) ) ;
fclose ( fp ) ;
}
}
ConfigFile . load ( cfn ) ;
// setup variable with config file variable
IVariable : : init ( ConfigFile ) ;
if ( ConfigFile . exists ( " DefaultEmailSMTP " ) & & ConfigFile . exists ( " DefaultEmailTo " ) )
NLNET : : setDefaultEmailParams (
ConfigFile . getVar ( " DefaultEmailSMTP " ) . asString ( ) ,
ConfigFile . exists ( " DefaultEmailFrom " )
? ConfigFile . getVar ( " DefaultEmailFrom " ) . asString ( )
: " service@opennel.org " ,
ConfigFile . getVar ( " DefaultEmailTo " ) . asString ( ) ) ;
//
// Set the shard Id
//
if ( ( var = ConfigFile . getVarPtr ( " NoWSShardId " ) ) ! = NULL )
{
_ShardId = var - > asInt ( ) ;
}
else
{
// something high enough as default
_ShardId = DEFAULT_SHARD_ID ;
}
if ( haveArg ( ' Z ' ) )
{
string s = IService : : getInstance ( ) - > getArg ( ' Z ' ) ;
if ( s = = " u " )
{
// do not release the module manager
}
else
{
// release the module manager
IModuleManager : : getInstance ( ) . releaseInstance ( ) ;
}
return 0 ;
}
// we have to call this again because the config file can changed this variable but the cmd line is more prioritary
if ( haveArg ( ' A ' ) )
RunningDirectory = CPath : : standardizePath ( getArg ( ' A ' ) ) ;
//
// Init debug/log stuffs (must be first things otherwise we can't log if errors)
//
// get the log dir if any in the command line
if ( haveArg ( ' L ' ) )
LogDirectory = CPath : : standardizePath ( getArg ( ' L ' ) ) ;
changeLogDirectory ( LogDirectory ) ;
bool noLog = ( ConfigFile . exists ( " DontLog " ) ) & & ( ConfigFile . getVar ( " DontLog " ) . asInt ( ) = = 1 ) ;
noLog | = haveLongArg ( " nolog " ) ;
if ( ! noLog )
{
// we create the log with service name filename ("test_service_ALIAS.log" for example)
string logname = LogDirectory . toString ( ) + _LongName ;
if ( haveArg ( ' N ' ) )
logname + = " _ " + toLower ( getArg ( ' N ' ) ) ;
logname + = " .log " ;
fd . setParam ( logname , false ) ;
DebugLog - > addDisplayer ( & fd ) ;
InfoLog - > addDisplayer ( & fd ) ;
WarningLog - > addDisplayer ( & fd ) ;
AssertLog - > addDisplayer ( & fd ) ;
ErrorLog - > addDisplayer ( & fd ) ;
CommandLog . addDisplayer ( & fd , true ) ;
}
bool dontUseStdIn = ( ConfigFile . exists ( " DontUseStdIn " ) ) & & ( ConfigFile . getVar ( " DontUseStdIn " ) . asInt ( ) = = 1 ) ;
if ( ! dontUseStdIn )
{
IStdinMonitorSingleton : : getInstance ( ) - > init ( ) ;
}
//
// Init the hierarchical timer
//
CHTimer : : startBench ( false , true ) ;
CHTimer : : endBench ( ) ;
//
// Set the assert mode
//
if ( ConfigFile . exists ( " Assert " ) )
setAssert ( ConfigFile . getVar ( " Assert " ) . asInt ( ) = = 1 ) ;
//
// Set the update timeout if found in the cfg
//
if ( ( var = ConfigFile . getVarPtr ( " UpdateTimeout " ) ) ! = NULL )
{
_UpdateTimeout = var - > asInt ( ) ;
}
//
// Set the negative filter from the config file
//
for ( const char * * name = NegFiltersNames ; * name ; name + + )
{
if ( ( var = ConfigFile . getVarPtr ( * name ) ) ! = NULL )
{
ConfigFile . setCallback ( * name , cbLogFilter ) ;
cbLogFilter ( * var ) ;
}
}
ConfigFile . setCallback ( " Commands " , cbExecuteCommands ) ;
if ( ( var = ConfigFile . getVarPtr ( " Commands " ) ) ! = NULL )
{
cbExecuteCommands ( * var ) ;
}
//
// Command line start
//
commandStart ( ) ;
//
// Create the window if needed
//
if ( ( var = ConfigFile . getVarPtr ( " WindowStyle " ) ) ! = NULL )
{
string disp = var - > asString ( ) ;
# ifdef NL_USE_GTK
if ( disp = = " GTK " )
{
WindowDisplayer = new CGtkDisplayer ( " DEFAULT_WD " ) ;
}
# endif // NL_USE_GTK
# ifdef NL_OS_WINDOWS
if ( disp = = " WIN " )
{
WindowDisplayer = new CWinDisplayer ( " DEFAULT_WD " ) ;
}
# endif // NL_OS_WINDOWS
if ( WindowDisplayer = = NULL & & disp ! = " NONE " )
{
nlinfo ( " SERVICE: Unknown value for the WindowStyle (should be GTK, WIN or NONE), use no window displayer " ) ;
}
}
vector < pair < string , uint > > displayedVariables ;
//uint speedNetLabel, speedUsrLabel, rcvLabel, sndLabel, rcvQLabel, sndQLabel, scrollLabel;
if ( WindowDisplayer ! = NULL )
{
//
// Init window param if necessary
//
sint x = - 1 , y = - 1 , w = - 1 , h = - 1 , fs = 10 , history = - 1 ;
bool iconified = false , ww = false ;
string fn ;
if ( ( var = ConfigFile . getVarPtr ( " XWinParam " ) ) ! = NULL ) x = var - > asInt ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " YWinParam " ) ) ! = NULL ) y = var - > asInt ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " WWinParam " ) ) ! = NULL ) w = var - > asInt ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " HWinParam " ) ) ! = NULL ) h = var - > asInt ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " HistoryWinParam " ) ) ! = NULL ) history = var - > asInt ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " IWinParam " ) ) ! = NULL ) iconified = var - > asInt ( ) = = 1 ;
if ( ( var = ConfigFile . getVarPtr ( " FontSize " ) ) ! = NULL ) fs = var - > asInt ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " FontName " ) ) ! = NULL ) fn = var - > asString ( ) ;
if ( ( var = ConfigFile . getVarPtr ( " WordWrap " ) ) ! = NULL ) ww = var - > asInt ( ) = = 1 ;
if ( haveArg ( ' I ' ) ) iconified = true ;
WindowDisplayer - > create ( string ( " *INIT* " ) + _ShortName + " " + _LongName , iconified , x , y , w , h , history , fs , fn , ww , & CommandLog ) ;
DebugLog - > addDisplayer ( WindowDisplayer ) ;
InfoLog - > addDisplayer ( WindowDisplayer ) ;
WarningLog - > addDisplayer ( WindowDisplayer ) ;
ErrorLog - > addDisplayer ( WindowDisplayer ) ;
AssertLog - > addDisplayer ( WindowDisplayer ) ;
CommandLog . addDisplayer ( WindowDisplayer , true ) ;
// adding default displayed variables
displayedVariables . push_back ( make_pair ( string ( " NetLop|NetSpeedLoop " ) , WindowDisplayer - > createLabel ( " NetLop " ) ) ) ;
displayedVariables . push_back ( make_pair ( string ( " UsrLop|UserSpeedLoop " ) , WindowDisplayer - > createLabel ( " UsrLop " ) ) ) ;
displayedVariables . push_back ( make_pair ( string ( " |Scroller " ) , WindowDisplayer - > createLabel ( " NeL Rulez " ) ) ) ;
CConfigFile : : CVar * v = ConfigFile . getVarPtr ( " DisplayedVariables " ) ;
if ( v ! = NULL )
{
for ( uint i = 0 ; i < v - > size ( ) ; i + + )
{
displayedVariables . push_back ( make_pair ( v - > asString ( i ) , WindowDisplayer - > createLabel ( v - > asString ( i ) . c_str ( ) ) ) ) ;
}
}
}
nlinfo ( " SERVICE: Starting Service '%s' using NeL ( " __DATE__ " " __TIME__ " ) compiled %s " , _ShortName . c_str ( ) , CompilationDate . c_str ( ) ) ;
nlinfo ( " SERVICE: On OS: %s " , CSystemInfo : : getOS ( ) . c_str ( ) ) ;
setExitStatus ( EXIT_SUCCESS ) ;
//
// Redirect signal if needed (in release mode only)
//
# ifdef NL_OS_WINDOWS
# ifdef NL_NO_DEBUG
initSignal ( ) ;
# else
// don't install signal is the application is started in debug mode
if ( IsDebuggerPresent ( ) )
{
//nlinfo("Running with the debugger, don't redirect signals");
initSignal ( ) ;
}
else
{
//nlinfo("Running without the debugger, redirect SIGINT signal");
initSignal ( ) ;
}
# endif
# else // NL_OS_UNIX
initSignal ( ) ;
# endif
//
// Ignore SIGPIPE (broken pipe) on unix system
//
# ifdef NL_OS_UNIX
// Ignore the SIGPIPE signal
sigset_t SigList ;
bool IgnoredPipe = true ;
if ( sigemptyset ( & SigList ) = = - 1 )
{
perror ( " sigemptyset() " ) ;
IgnoredPipe = false ;
}
if ( sigaddset ( & SigList , SIGPIPE ) = = - 1 )
{
perror ( " sigaddset() " ) ;
IgnoredPipe = false ;
}
if ( sigprocmask ( SIG_BLOCK , & SigList , NULL ) = = - 1 )
{
perror ( " sigprocmask() " ) ;
IgnoredPipe = false ;
}
nldebug ( " SERVICE: SIGPIPE %s " , IgnoredPipe ? " Ignored " : " Not Ignored " ) ;
# endif // NL_OS_UNIX
//
// Initialize the network system
//
string localhost ;
try
{
// Initialize WSAStartup and network stuffs
CSock : : initNetwork ( ) ;
// Get the localhost name
localhost = CInetAddress : : localHost ( ) . hostName ( ) ;
}
catch ( NLNET : : ESocket & )
{
localhost = " <UnknownHost> " ;
}
// Set the localhost name and service name to the logger
CLog : : setProcessName ( localhost + " / " + _ShortName ) ;
nlinfo ( " SERVICE: Host: %s " , localhost . c_str ( ) ) ;
//
// Initialize server parameters
//
// set the listen port if there are a port arg in the command line
if ( haveArg ( ' P ' ) )
{
NLMISC : : fromString ( getArg ( ' P ' ) , ListeningPort ) ;
}
// set the aes aliasname if present in cfg file
CConfigFile : : CVar * varAliasName = ConfigFile . getVarPtr ( " AESAliasName " ) ;
if ( varAliasName ! = NULL )
{
_AliasName = varAliasName - > asString ( ) ;
nlinfo ( " SERVICE: Setting alias name to: '%s' " , _AliasName . c_str ( ) ) ;
}
// set the aes aliasname if is present in the command line
if ( haveArg ( ' N ' ) )
{
_AliasName = getArg ( ' N ' ) ;
nlinfo ( " SERVICE: Setting alias name to: '%s' " , _AliasName . c_str ( ) ) ;
}
// Load the recording state from the config file
if ( ( var = ConfigFile . getVarPtr ( " Rec " ) ) ! = NULL )
{
string srecstate = toUpper ( var - > asString ( ) ) ;
if ( srecstate = = " RECORD " )
{
_RecordingState = CCallbackNetBase : : Record ;
nlinfo ( " SERVICE: Service recording messages " ) ;
}
else if ( srecstate = = " REPLAY " )
{
_RecordingState = CCallbackNetBase : : Replay ;
nlinfo ( " SERVICE: Service replaying messages " ) ;
}
else
{
_RecordingState = CCallbackNetBase : : Off ;
}
}
else
{
// Not found
_RecordingState = CCallbackNetBase : : Off ;
}
// Load the default stream format
if ( ( var = ConfigFile . getVarPtr ( " StringMsgFormat " ) ) ! = NULL )
{
CMessage : : setDefaultStringMode ( var - > asInt ( ) = = 1 ) ;
}
else
{
// Not found => binary
CMessage : : setDefaultStringMode ( false ) ;
}
///
/// Layer5 Startup
///
// get the sid
if ( ( var = ConfigFile . getVarPtr ( " SId " ) ) ! = NULL )
{
sint32 sid = var - > asInt ( ) ;
if ( sid < = 0 | | sid > 255 )
{
nlwarning ( " SERVICE: Bad SId value in the config file, %d is not in [0;255] range " , sid ) ;
_SId . set ( 0 ) ;
}
else
{
_SId . set ( static_cast < uint16 > ( sid ) ) ;
}
}
else
{
// ok, SId not found, use dynamic sid
_SId . set ( 0 ) ;
}
// look if we don't want to use NS
if ( ( var = ConfigFile . getVarPtr ( " DontUseNS " ) ) ! = NULL )
{
// if we set the value in the config file, get it
_DontUseNS = ( var - > asInt ( ) = = 1 ) ;
}
else
{
// if not, we use ns only if service is not ns, ls, aes, as
_DontUseNS = false ;
}
if ( haveLongArg ( " nons " ) )
{
// command line override
_DontUseNS = true ;
}
//
// Register all network associations (must be before the CUnifiedNetwork::getInstance()->init)
//
if ( ( var = ConfigFile . getVarPtr ( " Networks " ) ) ! = NULL )
{
for ( uint8 i = 0 ; i < var - > size ( ) ; i + + )
CUnifiedNetwork : : getInstance ( ) - > addNetworkAssociation ( var - > asString ( i ) , i ) ;
}
if ( ( var = ConfigFile . getVarPtr ( " DefaultNetworks " ) ) ! = NULL )
{
for ( uint8 i = 0 ; i < var - > size ( ) ; i + + )
CUnifiedNetwork : : getInstance ( ) - > addDefaultNetwork ( var - > asString ( i ) ) ;
}
// normal setup for the common services
if ( ! _DontUseNS )
{
bool ok = false ;
while ( ! ok )
{
string LSAddr ;
if ( haveArg ( ' B ' ) )
{
// if the naming service address is set on the command line, get it (overwrite the cfg)
LSAddr = getArg ( ' B ' ) ;
}
else
{
// else read the naming service address from the config file
LSAddr = ConfigFile . getVar ( " NSHost " ) . asString ( ) ;
}
// if there's no port to the NS, use the default one 50000
if ( LSAddr . find ( " : " ) = = string : : npos )
LSAddr + = " :50000 " ;
CInetAddress loc ( LSAddr ) ;
try
{
// todo: check if app not closed by user, or you get stuck here
if ( CUnifiedNetwork : : getInstance ( ) - > init ( & loc , _RecordingState , _ShortName , ListeningPort , _SId ) )
{
ok = true ;
}
else
{
nlinfo ( " SERVICE: Exiting... " ) ;
beep ( 880 , 400 ) ;
beep ( 440 , 400 ) ;
beep ( 220 , 400 ) ;
// remove the stdin monitor thread
IStdinMonitorSingleton : : getInstance ( ) - > release ( ) ; // does nothing if not initialized
// release the module manager
IModuleManager : : getInstance ( ) . releaseInstance ( ) ;
return 10 ;
}
}
catch ( ESocketConnectionFailed & )
{
nlinfo ( " SERVICE: Could not connect to the Naming Service (%s). Retrying in a few seconds... " , loc . asString ( ) . c_str ( ) ) ;
nlSleep ( 5000 ) ;
}
}
}
else
{
CUnifiedNetwork : : getInstance ( ) - > init ( NULL , _RecordingState , _ShortName , ListeningPort , _SId ) ;
}
// get the hostname for later use
_HostName = CInetAddress : : localHost ( ) . hostName ( ) ;
// At this point, the _SId must be ok if we use the naming service.
// If it's 0, it means that we don't use NS and we left the other side server to find a sid for your connection
if ( ! _DontUseNS )
{
nlassert ( _SId . get ( ) ! = 0 ) ;
}
//
// Connect to the local AES and send identification
//
// look if we don't want to use NS
if ( ( var = ConfigFile . getVarPtr ( " DontUseAES " ) ) ! = NULL )
{
// if we set the value in the config file, get it
_DontUseAES = var - > asInt ( ) = = 1 ;
}
else
{
// if not, we use aes only if service is not aes or as
_DontUseAES = false ;
}
initAdmin ( _DontUseAES ) ;
while ( NLNET : : CUnifiedNetwork : : getInstance ( ) - > tryFlushAllQueues ( ) ! = 0 )
{
nlSleep ( 10 ) ;
}
//
// Add callback array
//
// add inner service callback array
NLNET : : CUnifiedNetwork : : getInstance ( ) - > addCallbackArray ( builtinServiceCallbacks , sizeof ( builtinServiceCallbacks ) / sizeof ( builtinServiceCallbacks [ 0 ] ) ) ;
// add callback set in the NLNET_SERVICE_MAIN macro
NLNET : : CUnifiedNetwork : : getInstance ( ) - > addCallbackArray ( _CallbackArray , _CallbackArraySize ) ;
//
// Now we have the service id, we can set the entites id generator
//
NLMISC : : CEntityId : : setServiceId ( TServiceId8 ( _SId ) . get ( ) ) ;
// Set the localhost name and service name and the sid
CLog : : setProcessName ( localhost + " / " + _ShortName + " - " + toString ( _SId . get ( ) ) ) ;
//
// Add default pathes
//
if ( ( var = ConfigFile . getVarPtr ( " IgnoredFiles " ) ) ! = NULL )
{
for ( uint i = 0 ; i < var - > size ( ) ; i + + )
{
CPath : : addIgnoredDoubleFile ( var - > asString ( i ) ) ;
}
}
if ( ( var = ConfigFile . getVarPtr ( " Paths " ) ) ! = NULL )
{
for ( uint i = 0 ; i < var - > size ( ) ; i + + )
{
CPath : : addSearchPath ( var - > asString ( i ) , true , false ) ;
}
}
if ( ( var = ConfigFile . getVarPtr ( " PathsNoRecurse " ) ) ! = NULL )
{
for ( uint i = 0 ; i < var - > size ( ) ; i + + )
{
CPath : : addSearchPath ( var - > asString ( i ) , false , false ) ;
}
}
// if we can, try to setup where to save files
if ( IService : : getInstance ( ) - > haveArg ( ' W ' ) )
{
// use the command line param if set (must be done after the config file has been loaded)
SaveFilesDirectory = IService : : getInstance ( ) - > getArg ( ' W ' ) ;
}
CTransportClass : : init ( ) ;
//
// Call the user service init
//
userInitCalled = true ; // the bool must be put *before* the call to init()
setCurrentStatus ( " Initializing " ) ;
init ( ) ;
clearCurrentStatus ( " Initializing " ) ;
//
// Connects to the present services
// WARNING: only after the user init() was called because the
// addService may call up service callbacks.
//
CUnifiedNetwork : : getInstance ( ) - > connect ( ) ;
//
// Say to the AES that the service is ready
//
if ( ! _DontUseAES )
{
// send the ready message (service init finished)
CMessage msgout ( " SR " ) ;
CUnifiedNetwork : : getInstance ( ) - > send ( " AES " , msgout ) ;
}
_Initialized = true ;
nlinfo ( " SERVICE: Service initialized, executing StartCommands " ) ;
//
// Call the user command from the config file if any
//
string cmdRoot ( " StartCommands " ) ;
vector < string > posts ;
// add an empty string (for the common part of start commands)
posts . push_back ( string ( ) ) ;
if ( IService : : getInstance ( ) - > haveArg ( ' S ' ) )
{
string s = IService : : getInstance ( ) - > getArg ( ' S ' ) ;
posts . push_back ( s ) ;
}
CConfigFile : : CVar * var ;
for ( uint i = 0 ; i < posts . size ( ) ; + + i )
{
string varName = cmdRoot + posts [ i ] ;
if ( ( var = IService : : getInstance ( ) - > ConfigFile . getVarPtr ( varName ) ) ! = NULL )
{
for ( uint i = 0 ; i < var - > size ( ) ; i + + )
{
ICommand : : execute ( var - > asString ( i ) , CommandLog ) ;
}
}
}
string str ;
CLog logDisplayVars ;
CLightMemDisplayer mdDisplayVars ;
logDisplayVars . addDisplayer ( & mdDisplayVars ) ;
//
// Activate the timeout assertion thread
//
timeoutThread = IThread : : create ( & MyTAT , 1024 * 4 ) ;
timeoutThread - > start ( ) ;
//
// Set service ready
//
nlinfo ( " SERVICE: Service ready " ) ;
if ( WindowDisplayer ! = NULL )
WindowDisplayer - > setTitleBar ( _ShortName + " " + _LongName + " " + Version . c_str ( ) ) ;
//
// Call the user service update each loop and check files and network activity
//
TTime checkCpuProcTime = 0 ;
for ( ; ; )
{
MyTAT . activate ( ) ;
if ( Bench ) CHTimer : : startBench ( false , true , false ) ;
// count the amount of time to manage internal system
TTime bbefore = CTime : : getLocalTime ( ) ;
// every second, check for CPU usage
if ( bbefore - checkCpuProcTime > 1000 )
{
checkCpuProcTime = bbefore ;
_CPUUsageStats . peekMeasures ( ) ;
}
// call the user update and exit if the user update asks it
{
H_AUTO ( NLNETServiceUpdate ) ;
if ( ! update ( ) )
{
CHTimer : : endBench ( ) ;
break ;
}
}
// deal with any input waiting from stdin
{
H_AUTO ( NLNETStdinMonitorUpdate ) ;
IStdinMonitorSingleton : : getInstance ( ) - > update ( ) ;
}
// if the launching mode is 'quit after the first update' we set the exit signal
if ( haveArg ( ' Q ' ) )
{
// we use -100 so that the final return code ends as 0 (100+ExitSignalAsked) because
// otherwise we can't launch services with -Q in a makefile
ExitSignalAsked = - 100 ;
}
NbUserUpdate + + ;
// count the amount of time to manage internal system
TTime before = CTime : : getLocalTime ( ) ;
if ( WindowDisplayer ! = NULL )
{
// update the window displayer and quit if asked
if ( ! WindowDisplayer - > update ( ) )
{
nlinfo ( " SERVICE: The window displayer was closed by user, need to quit " ) ;
ExitSignalAsked = 1 ;
}
}
// stop the loop if the exit signal asked
if ( ExitSignalAsked ! = 0 )
{
if ( _RequestClosureClearanceCallback )
{
if ( _ClosureClearanceStatus = = CCClearedForClosure )
{
// Clearance has been granted
CHTimer : : endBench ( ) ;
break ;
}
else if ( _ClosureClearanceStatus = = CCMustRequestClearance )
{
if ( _RequestClosureClearanceCallback ( ) )
{
// Direct clearance
_ClosureClearanceStatus = CCClearedForClosure ;
CHTimer : : endBench ( ) ;
break ;
}
else
{
// Delayed clearance
_ClosureClearanceStatus = CCWaitingForClearance ;
}
}
else if ( _ClosureClearanceStatus > = CCCallbackThenClose )
{
// Always direct closure, because we don't have a connection to the naming service anymore
// But still call the callback
_RequestClosureClearanceCallback ( ) ;
CHTimer : : endBench ( ) ;
break ;
}
}
else
{
// Immediate closure, no clearance needed
CHTimer : : endBench ( ) ;
break ;
}
}
CConfigFile : : checkConfigFiles ( ) ;
updateAdmin ( ) ;
CFile : : checkFileChange ( ) ;
// update updatable interface
set < IServiceUpdatable * > : : iterator first ( _Updatables . begin ( ) ) , last ( _Updatables . end ( ) ) ;
for ( ; first ! = last ; + + first )
{
IServiceUpdatable * updatable = * first ;
updatable - > serviceLoopUpdate ( ) ;
}
// get and manage layer 5 messages
CUnifiedNetwork : : getInstance ( ) - > update ( _UpdateTimeout ) ;
// update modules
IModuleManager : : getInstance ( ) . updateModules ( ) ;
// Allow direct closure if the naming service was lost
if ( _RequestClosureClearanceCallback )
{
if ( ! CNamingClient : : connected ( ) )
{
if ( _ClosureClearanceStatus < CCCallbackThenClose )
_ClosureClearanceStatus + = CCCallbackThenClose ; // change status but backup old value
}
else
{
if ( _ClosureClearanceStatus > = CCCallbackThenClose )
_ClosureClearanceStatus - = CCCallbackThenClose ; // set the closure state back if the NS comes back
}
}
NetSpeedLoop = ( sint32 ) ( CTime : : getLocalTime ( ) - before ) ;
UserSpeedLoop = ( sint32 ) ( before - bbefore ) ;
L5CallbackTime = TimeInCallback - LastTimeInCallback ;
LastTimeInCallback = TimeInCallback ;
L5CallbackCount = TotalCallbackCalled - LastTotalCallbackCalled ;
LastTotalCallbackCalled = TotalCallbackCalled ;
if ( WindowDisplayer ! = NULL )
{
static TTime lt = 0 ;
TTime ct = CTime : : getLocalTime ( ) ;
if ( ct > lt + 100 )
{
lt = ct ;
uint64 rcv , snd , rcvq , sndq ;
rcv = CUnifiedNetwork : : getInstance ( ) - > getBytesReceived ( ) ;
snd = CUnifiedNetwork : : getInstance ( ) - > getBytesSent ( ) ;
rcvq = CUnifiedNetwork : : getInstance ( ) - > getReceiveQueueSize ( ) ;
sndq = CUnifiedNetwork : : getInstance ( ) - > getSendQueueSize ( ) ;
for ( uint i = 0 ; i < displayedVariables . size ( ) ; i + + )
{
// it s a separator, do nothing
if ( displayedVariables [ i ] . first . empty ( ) )
continue ;
// it s a command, do nothing
if ( displayedVariables [ i ] . first [ 0 ] = = ' @ ' )
continue ;
string dispName = displayedVariables [ i ] . first ;
string varName = dispName ;
string : : size_type pos = dispName . find ( " | " ) ;
if ( pos ! = string : : npos )
{
varName = displayedVariables [ i ] . first . substr ( pos + 1 ) ;
dispName = displayedVariables [ i ] . first . substr ( 0 , pos ) ;
}
if ( dispName . empty ( ) )
str = " " ;
else
str = dispName + " : " ;
mdDisplayVars . clear ( ) ;
ICommand : : execute ( varName , logDisplayVars , true ) ;
const std : : deque < std : : string > & strs = mdDisplayVars . lockStrings ( ) ;
if ( strs . size ( ) > 0 )
{
str + = strs [ 0 ] . substr ( 0 , strs [ 0 ] . size ( ) - 1 ) ;
}
else
{
str + = " ??? " ;
}
mdDisplayVars . unlockStrings ( ) ;
WindowDisplayer - > setLabel ( displayedVariables [ i ] . second , str ) ;
}
}
}
// nldebug ("SYNC: updatetimeout must be %d and is %d, sleep the rest of the time", _UpdateTimeout, delta);
CHTimer : : endBench ( ) ;
// Resetting the hierarchical timer must be done outside the top-level timer
if ( _ResetMeasures )
{
CHTimer : : clear ( ) ;
_ResetMeasures = false ;
}
MyTAT . desactivate ( ) ;
}
}
catch ( EFatalError & )
{
// Somebody call nlerror, so we have to quit now, the message already display
// so we don't have to to anything
setExitStatus ( EXIT_FAILURE ) ;
}
catch ( ESocket & e )
{
// Catch NeL network exception to release the system cleanly setExitStatus (EXIT_FAILURE);
ErrorLog - > displayNL ( " NeL Exception in \" %s \" : %s " , _ShortName . c_str ( ) , e . what ( ) ) ;
}
catch ( uint ) // SEH exceptions
{
ErrorLog - > displayNL ( " SERVICE: System exception " ) ;
}
try
{
nlinfo ( " SERVICE: Service starts releasing " ) ;
//
// Call the user service release() if the init() was called
//
if ( userInitCalled )
{
setCurrentStatus ( " Releasing " ) ;
release ( ) ;
}
CTransportClass : : release ( ) ;
//
// Delete all network connection (naming client also)
//
std : : vector < std : : string > namesOfOnlyServiceToFlushSendingV ;
explode ( NamesOfOnlyServiceToFlushSending . get ( ) , string ( " : " ) , namesOfOnlyServiceToFlushSendingV , true ) ;
CUnifiedNetwork : : getInstance ( ) - > release ( FlushSendingQueuesOnExit . get ( ) , namesOfOnlyServiceToFlushSendingV ) ;
// warn the module layer that the application is about to close
IModuleManager : : getInstance ( ) . applicationExit ( ) ;
// // release the network
// CSock::releaseNetwork ();
//
//
// Remove the window displayer
//
if ( WindowDisplayer ! = NULL )
{
DebugLog - > removeDisplayer ( WindowDisplayer ) ;
InfoLog - > removeDisplayer ( WindowDisplayer ) ;
WarningLog - > removeDisplayer ( WindowDisplayer ) ;
ErrorLog - > removeDisplayer ( WindowDisplayer ) ;
AssertLog - > removeDisplayer ( WindowDisplayer ) ;
CommandLog . removeDisplayer ( WindowDisplayer ) ;
delete WindowDisplayer ;
WindowDisplayer = NULL ;
}
nlinfo ( " SERVICE: Service released succesfully " ) ;
}
catch ( EFatalError & )
{
// Somebody call nlerror, so we have to quit now, the message already display
// so we don't have to to anything
setExitStatus ( EXIT_FAILURE ) ;
}
// remove the stdin monitor thread
IStdinMonitorSingleton : : getInstance ( ) - > release ( ) ; // does nothing if not initialized
// release the module manager
IModuleManager : : getInstance ( ) . releaseInstance ( ) ;
// release the network
CSock : : releaseNetwork ( ) ;
// stop the timeout thread
MyTAT . quit ( ) ;
if ( timeoutThread ! = NULL )
{
timeoutThread - > wait ( ) ;
delete timeoutThread ;
}
CHTimer : : display ( ) ;
CHTimer : : displayByExecutionPath ( ) ;
CHTimer : : displayHierarchical ( & CommandLog , true , 64 ) ;
CHTimer : : displayHierarchicalByExecutionPathSorted ( & CommandLog , CHTimer : : TotalTime , true , 64 ) ;
nlinfo ( " SERVICE: Service ends " ) ;
return ExitSignalAsked ? 100 + ExitSignalAsked : getExitStatus ( ) ;
}
void IService : : exit ( sint code )
{
nlinfo ( " SERVICE: Somebody called IService::exit(), I have to quit " ) ;
ExitSignalAsked = code ;
}
/// Push a new status on the status stack.
void IService : : setCurrentStatus ( const std : : string & status )
{
// remove the status if it is already in the stack
_ServiceStatusStack . erase ( std : : remove ( _ServiceStatusStack . begin ( ) , _ServiceStatusStack . end ( ) , status ) , _ServiceStatusStack . end ( ) ) ;
// insert the status on top of the stack
_ServiceStatusStack . push_back ( status ) ;
}
/// Remove a status from the status stack. If this status is at top of stack, the next status become the current status
void IService : : clearCurrentStatus ( const std : : string & status )
{
// remove the status of the stack
_ServiceStatusStack . erase ( std : : remove ( _ServiceStatusStack . begin ( ) , _ServiceStatusStack . end ( ) , status ) , _ServiceStatusStack . end ( ) ) ;
}
/// Add a tag in the status string
void IService : : addStatusTag ( const std : : string & statusTag )
{
_ServiveStatusTags . insert ( statusTag ) ;
}
/// Remove a tag from the status string
void IService : : removeStatusTag ( const std : : string & statusTag )
{
_ServiveStatusTags . erase ( statusTag ) ;
}
/// Get the current status with attached tags
std : : string IService : : getFullStatus ( ) const
{
string result ;
// get hold of the status at the top of the status stack
if ( ! _ServiceStatusStack . empty ( ) )
{
result = _ServiceStatusStack . back ( ) ;
}
// add status tags to the result so far
set < string > : : const_iterator first ( _ServiveStatusTags . begin ( ) ) , last ( _ServiveStatusTags . end ( ) ) ;
for ( ; first ! = last ; + + first )
{
if ( first ! = _ServiveStatusTags . begin ( ) | | ! result . empty ( ) )
result + = " " ;
result + = * first ;
}
// return the result
return result . empty ( ) ? " Online " : result ;
}
/*
* Require to reset the hierarchical timer
*/
void IService : : requireResetMeasures ( )
{
_ResetMeasures = true ;
}
/*
*
*/
std : : string IService : : getServiceUnifiedName ( ) const
{
nlassert ( ! _ShortName . empty ( ) ) ;
string res ;
if ( ! _AliasName . empty ( ) )
{
res = _AliasName + " / " ;
}
res + = _ShortName ;
if ( _SId . get ( ) ! = 0 )
{
res + = " - " ;
res + = toString ( _SId . get ( ) ) ;
}
return res ;
}
/*
* Returns the date of launch of the service . Unit : see CTime : : getSecondsSince1970 ( )
*/
uint32 IService : : getLaunchingDate ( ) const
{
return LaunchingDate ;
}
void IService : : registerUpdatable ( IServiceUpdatable * updatable )
{
_Updatables . insert ( updatable ) ;
}
void IService : : unregisterUpdatable ( IServiceUpdatable * updatable )
{
_Updatables . erase ( updatable ) ;
}
//
// Commands and Variables for controling all services
//
NLMISC_CATEGORISED_DYNVARIABLE ( nel , string , LaunchingDate , " date of the launching of the program " )
{
nlunreferenced ( human ) ;
if ( get ) * pointer = asctime ( localtime ( ( time_t * ) & LaunchingDate ) ) ;
}
NLMISC_CATEGORISED_DYNVARIABLE ( nel , string , Uptime , " time from the launching of the program " )
{
if ( get )
{
if ( human )
* pointer = secondsToHumanReadable ( CTime : : getSecondsSince1970 ( ) - LaunchingDate ) ;
else
* pointer = NLMISC : : toString ( CTime : : getSecondsSince1970 ( ) - LaunchingDate ) ;
}
else
{
NLMISC : : fromString ( * pointer , LaunchingDate ) ;
LaunchingDate = CTime : : getSecondsSince1970 ( ) - LaunchingDate ;
}
}
NLMISC_CATEGORISED_VARIABLE ( nel , string , CompilationDate , " date of the compilation " ) ;
NLMISC_CATEGORISED_VARIABLE ( nel , string , CompilationMode , " mode of the compilation " ) ;
NLMISC_CATEGORISED_VARIABLE ( nel , uint32 , NbUserUpdate , " number of time the user IService::update() called " ) ;
NLMISC_CATEGORISED_DYNVARIABLE ( nel , string , Scroller , " current size in bytes of the sent queue size " )
{
nlunreferenced ( human ) ;
if ( get )
{
// display the scroll text
static string foo = " Welcome to NeL Service! This scroll is used to see the update frequency of the main function and to see if the service is frozen or not. Have a nice day and hope you'll like NeL!!! "
" Welcome to NeL Service! This scroll is used to see the update frequency of the main function and to see if the service is frozen or not. Have a nice day and hope you'll like NeL!!! " ;
static int pos = 0 ;
* pointer = foo . substr ( ( pos + + ) % ( foo . size ( ) / 2 ) , 10 ) ;
}
}
NLMISC_CATEGORISED_COMMAND ( nel , quit , " exit the service " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
if ( args . size ( ) ! = 0 ) return false ;
log . displayNL ( " User ask me with a command to quit " ) ;
ExitSignalAsked = 0xFFFF ;
return true ;
}
NLMISC_CATEGORISED_COMMAND ( nel , brutalQuit , " exit the service brutally " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( log ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
if ( args . size ( ) ! = 0 ) return false ;
: : exit ( 0xFFFFFFFF ) ;
return true ;
}
# ifdef MUTEX_DEBUG
NLMISC_CATEGORISED_COMMAND ( nel , mutex , " display mutex values " , " " )
{
if ( args . size ( ) ! = 0 ) return false ;
map < CFairMutex * , TMutexLocks > acquiretimes = getNewAcquireTimes ( ) ;
map < CFairMutex * , TMutexLocks > : : iterator im ;
for ( im = acquiretimes . begin ( ) ; im ! = acquiretimes . end ( ) ; + + im )
{
log . displayNL ( " %d %p %s: %.0f %.0f, called %u times th(%d, %d wait)%s " , ( * im ) . second . MutexNum , ( * im ) . first , ( * im ) . second . MutexName . c_str ( ) ,
CTime : : cpuCycleToSecond ( ( * im ) . second . TimeToEnter ) * 1000.0 , CTime : : cpuCycleToSecond ( ( * im ) . second . TimeInMutex ) * 1000.0 ,
( * im ) . second . Nb , ( * im ) . second . ThreadHavingTheMutex , ( * im ) . second . WaitingMutex ,
( * im ) . second . Dead ? " DEAD " : " " ) ;
}
return true ;
}
# endif // MUTEX_DEBUG
NLMISC_CATEGORISED_COMMAND ( nel , serviceInfo , " display information about this service " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
if ( args . size ( ) ! = 0 ) return false ;
log . displayNL ( " Service %s '%s' using NeL ( " __DATE__ " " __TIME__ " ) " , IService : : getInstance ( ) - > getServiceLongName ( ) . c_str ( ) , IService : : getInstance ( ) - > getServiceUnifiedName ( ) . c_str ( ) ) ;
log . displayNL ( " Service listening port: %d " , IService : : getInstance ( ) - > ListeningPort . get ( ) ) ;
log . displayNL ( " Service running directory: '%s' " , IService : : getInstance ( ) - > RunningDirectory . c_str ( ) ) ;
log . displayNL ( " Service log directory: '%s' " , IService : : getInstance ( ) - > LogDirectory . c_str ( ) ) ;
log . displayNL ( " Service save files directory: '%s' " , IService : : getInstance ( ) - > SaveFilesDirectory . c_str ( ) ) ;
log . displayNL ( " Service write files directory: '%s' " , IService : : getInstance ( ) - > WriteFilesDirectory . c_str ( ) ) ;
log . displayNL ( " Service config directory: '%s' config filename: '%s.cfg' " , IService : : getInstance ( ) - > ConfigDirectory . c_str ( ) , IService : : getInstance ( ) - > _LongName . c_str ( ) ) ;
log . displayNL ( " Service id: %hu " , IService : : getInstance ( ) - > _SId . get ( ) ) ;
log . displayNL ( " Service update timeout: %dms " , IService : : getInstance ( ) - > _UpdateTimeout ) ;
log . displayNL ( " Service %suse naming service " , IService : : getInstance ( ) - > _DontUseNS ? " don't " : " " ) ;
log . displayNL ( " Service %suse admin executor service " , IService : : getInstance ( ) - > _DontUseAES ? " don't " : " " ) ;
log . displayNL ( " NeL is compiled in %s mode " , CompilationMode . c_str ( ) ) ;
log . displayNL ( " Services arguments: %d args " , IService : : getInstance ( ) - > _Args . size ( ) ) ;
for ( uint i = 0 ; i < IService : : getInstance ( ) - > _Args . size ( ) ; i + + )
{
log . displayNL ( " argv[%d] = '%s' " , i , IService : : getInstance ( ) - > _Args [ i ] . c_str ( ) ) ;
}
log . displayNL ( " Naming service info: %s " , CNamingClient : : info ( ) . c_str ( ) ) ;
ICommand : : execute ( " services " , log ) ;
return true ;
}
NLMISC_CATEGORISED_COMMAND ( nel , resetMeasures , " reset hierarchical timer " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( args ) ;
nlunreferenced ( log ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
IService : : getInstance ( ) - > requireResetMeasures ( ) ;
return true ;
}
NLMISC_CATEGORISED_COMMAND ( nel , getWinDisplayerInfo , " display the info about the pos and size of the window displayer " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( args ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
uint32 x , y , w , h ;
IService : : getInstance ( ) - > WindowDisplayer - > getWindowPos ( x , y , w , h ) ;
log . displayNL ( " Window Displayer : XWinParam = %d; YWinParam = %d; WWinParam = %d; HWinParam = %d; " , x , y , w , h ) ;
return true ;
}
NLMISC_CATEGORISED_COMMAND ( nel , displayConfigFile , " display the variables of the default configfile " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( args ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
IService : : getInstance ( ) - > ConfigFile . display ( & log ) ;
return true ;
}
NLMISC_CATEGORISED_COMMAND ( nel , getUnknownConfigFileVariables , " display the variables from config file that are called but not present " , " " )
{
nlunreferenced ( rawCommandString ) ;
nlunreferenced ( args ) ;
nlunreferenced ( quiet ) ;
nlunreferenced ( human ) ;
log . displayNL ( " %d Variables not found in the configfile '%s' " , IService : : getInstance ( ) - > ConfigFile . UnknownVariables . size ( ) , IService : : getInstance ( ) - > ConfigFile . getFilename ( ) . c_str ( ) ) ;
for ( uint i = 0 ; i < IService : : getInstance ( ) - > ConfigFile . UnknownVariables . size ( ) ; i + + )
{
log . displayNL ( " %s " , IService : : getInstance ( ) - > ConfigFile . UnknownVariables [ i ] . c_str ( ) ) ;
}
return true ;
}
// -1 = service is quitting
// 0 = service is not connected
// 1 = service is running
// 2 = service is launching
// 3 = service failed launching
NLMISC_CATEGORISED_DYNVARIABLE ( nel , string , State , " Set this value to 0 to shutdown the service and 1 to start the service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get )
{
}
else
{
if ( IService : : getInstance ( ) - > getServiceShortName ( ) = = " AES " | | IService : : getInstance ( ) - > getServiceShortName ( ) = = " AS " )
{
nlinfo ( " SERVICE: I can't set State=0 because I'm the admin and I should never quit " ) ;
}
else if ( * pointer = = " 0 " | | * pointer = = " 2 " )
{
// ok, we want to set the value to false, just quit
nlinfo ( " SERVICE: User ask me with a command to quit using the State variable " ) ;
ExitSignalAsked = 0xFFFE ;
IService * srv = IService : : getInstance ( ) ;
if ( srv = = NULL )
{
return ;
}
srv - > setCurrentStatus ( " Quitting " ) ;
}
else
{
nlwarning ( " SERVICE: Unknown value for State '%s' " , ( * pointer ) . c_str ( ) ) ;
}
}
// whether reading or writing, the internal value of the state variable should end up as the result of getFullStatus()
IService * srv = IService : : getInstance ( ) ;
if ( srv = = NULL )
{
return ;
}
* pointer = srv - > getFullStatus ( ) ;
}
NLMISC_CATEGORISED_DYNVARIABLE ( nel , uint32 , ShardId , " Get value of shardId set for this particular service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get )
{
* pointer = IService : : getInstance ( ) - > getShardId ( ) ;
}
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , CPULoad , " Get instant CPU load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPULoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , ProcessLoad , " Get instant CPU load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , CPUUserLoad , " Get instant CPU user load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUUserLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , CPUSytemLoad , " Get instant CPU system load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUSystemLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , CPUNiceLoad , " Get instant CPU nice processes load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUNiceLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , CPUIOWaitLoad , " Get instant CPU IO wait load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUIOWaitLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , ProcessUserLoad , " Get instant CPU user load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessUserLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , ProcessSystemLoad , " Get instant CPU system load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessSystemLoad ( ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanCPULoad , " Get instant CPU load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPULoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanProcessLoad , " Get instant CPU load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanCPUUserLoad , " Get instant CPU user load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUUserLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanCPUSytemLoad , " Get instant CPU system load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUSystemLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanCPUNiceLoad , " Get instant CPU nice processes load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUNiceLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanCPUIOWaitLoad , " Get instant CPU IO wait load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUIOWaitLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanProcessUserLoad , " Get instant CPU user load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessUserLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , MeanProcessSystemLoad , " Get instant CPU system load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessSystemLoad ( CCPUTimeStat : : Mean ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakCPULoad , " Get instant CPU load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPULoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakProcessLoad , " Get instant CPU load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessLoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakCPUUserLoad , " Get instant CPU user load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUUserLoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakCPUSytemLoad , " Get instant CPU system load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUSystemLoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakCPUNiceLoad , " Get instant CPU nice processes load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUNiceLoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakCPUIOWaitLoad , " Get instant CPU IO wait load of the server " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getCPUIOWaitLoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakProcessUserLoad , " Get instant CPU user load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessUserLoad ( CCPUTimeStat : : Peak ) ; }
}
NLMISC_CATEGORISED_DYNVARIABLE ( cpu , float , PeakProcessSystemLoad , " Get instant CPU system load of the process/service " )
{
nlunreferenced ( human ) ;
// read or write the variable
if ( get ) { * pointer = IService : : getInstance ( ) - > getCPUUsageStats ( ) . getProcessSystemLoad ( CCPUTimeStat : : Peak ) ; }
}
} //NLNET