/*
* Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
* All rights reserved.
* This component and the accompanying materials are made available
* under the terms of the License "Symbian Foundation License v1.0"
* which accompanies this distribution, and is available
* at the URL "http://www.symbianfoundation.org/legal/sfl-v10.html".
*
* Initial Contributors:
* Nokia Corporation - initial contribution.
*
* Contributors:
*
* Description:
* Trace macro definitions.
*
*/
#ifndef TRACE_H
#define TRACE_H
#include <e32base.h> // TCleanupItem
#include "traceconfiguration.hrh"
#ifdef TRACE_INTO_FILE
#include <flogger.h> // RFileLogger
#else
#include <e32debug.h> // RDebug
#endif
//-----------------------------------------------------------------------------
// Constants
//-----------------------------------------------------------------------------
//
// NOTE!
// Replace all COMPONENT_NAME occurnaces with your own component / module name.
/**
* Prefix trace macro to complete tracing with component name.
* Returns TDesC which can be used directly with RDebug or RFileLogger.
*/
#define _PREFIX_TRACE( aMsg ) TPtrC( (const TText*)L"[Settings]: " L##aMsg )
/**
* Prefix error trace
*/
#define _PREFIX_ERROR( aMsg ) _PREFIX_TRACE( "[ERROR: %d]: " L##aMsg )
/**
* Prefix info trace.
*/
#define _PREFIX_INFO( aMsg ) _PREFIX_TRACE( "[INFO]: " L##aMsg )
/**
* Prefix macro for strings
*/
#define _PREFIX_CHAR( aMsg ) (const char*)"[Settings]: " ##aMsg
/**
* Define needed directories if TRACE_INTO_FILE macro in use
*/
#ifdef TRACE_INTO_FILE
_LIT( KDir, "settings" );
_LIT( KFile, "settings.txt" );
_LIT( KFullPath, "c:\\logs\\settings\\" );
#endif
//-----------------------------------------------------------------------------
// Error trace macros
//-----------------------------------------------------------------------------
//
#ifdef ERROR_TRACE
/**
* Error trace definitions.
*/
#ifdef TRACE_INTO_FILE
#define ERROR( aErr, aMsg )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr );\
}\
}
#define ERROR_1( aErr, aMsg, aP1 )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr, aP1 );\
}\
}
#define ERROR_2( aErr, aMsg, aP1, aP2 )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr, aP1, aP2 );\
}\
}
#define ERROR_3( aErr, aMsg, aP1, aP2, aP3 )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3 );\
}\
}
#define ERROR_4( aErr, aMsg, aP1, aP2, aP3, aP4 )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3, aP4 );\
}\
}
#define ERROR_5( aErr, aMsg, aP1, aP2, aP3, aP4, aP5 )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3, aP4, aP5 );\
}\
}
#define ERROR_6( aErr, aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )\
{\
if( aErr < KErrNone )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3, aP4, aP5 );\
}\
}
#else//TRACE_INTO_FILE not defined
#define ERROR( aErr, aMsg )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr );\
}\
}
#define ERROR_1( aErr, aMsg, aP1 )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr, aP1 );\
}\
}
#define ERROR_2( aErr, aMsg, aP1, aP2 )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr, aP1, aP2 );\
}\
}
#define ERROR_3( aErr, aMsg, aP1, aP2, aP3 )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3 );\
}\
}
#define ERROR_4( aErr, aMsg, aP1, aP2, aP3, aP4 )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3, aP4 );\
}\
}
#define ERROR_5( aErr, aMsg, aP1, aP2, aP3, aP4, aP5 )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3, aP4, aP5 );\
}\
}
#define ERROR_6( aErr, aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )\
{\
if( aErr < KErrNone )\
{\
RDebug::Print( _PREFIX_ERROR( aMsg ), aErr, aP1, aP2, aP3, aP4, aP5, aP6 );\
}\
}
#endif//TRACE_INTO_FILE
#define ERROR_GEN( aMsg ) ERROR( KErrGeneral, aMsg )
#define ERROR_GEN_1( aMsg, aP1 ) ERROR_1( KErrGeneral, aMsg, aP1 )
#define ERROR_GEN_2( aMsg, aP1, aP2 ) ERROR_2( KErrGeneral, aMsg, aP1, aP2 )
#define ERROR_GEN_3( aMsg, aP1, aP2, aP3 ) ERROR_3( KErrGeneral, aMsg, aP1, aP3 )
#define ERROR_GEN_4( aMsg, aP1, aP2, aP3, aP4 ) ERROR_4( KErrGeneral, aMsg, aP1, aP3, aP4 )
#define ERROR_GEN_5( aMsg, aP1, aP2, aP3, aP4, aP5 ) ERROR_5( KErrGeneral, aMsg, aP1, aP3, aP4, aP5 )
#define ERROR_GEN_6( aMsg, aP1, aP2, aP3, aP4, aP5, aP6 ) ERROR_6( KErrGeneral, aMsg, aP1, aP3, aP4, aP5, aP6 )
#else//ERROR_TRACE not defined
#define ERROR( aErr, aMsg )
#define ERROR_1( aErr, aMsg, aP1 )
#define ERROR_2( aErr, aMsg, aP1, aP2 )
#define ERROR_3( aErr, aMsg, aP1, aP2, aP3 )
#define ERROR_4( aErr, aMsg, aP1, aP2, aP3, aP4 )
#define ERROR_5( aErr, aMsg, aP1, aP2, aP3, aP4, aP5 )
#define ERROR_6( aErr, aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )
#define ERROR_GEN( aMsg )
#define ERROR_GEN_1( aMsg, aP1 )
#define ERROR_GEN_2( aMsg, aP1, aP2 )
#define ERROR_GEN_3( aMsg, aP1, aP2, aP3 )
#define ERROR_GEN_4( aMsg, aP1, aP2, aP3, aP4 )
#define ERROR_GEN_5( aMsg, aP1, aP2, aP3, aP4, aP5 )
#define ERROR_GEN_6( aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )
#endif//ERROR_TRACE
//-----------------------------------------------------------------------------
// TRAP and trace with error macro
//-----------------------------------------------------------------------------
//
#define TRAP_ERROR( aErr, aFunction )\
{\
TRAP( aErr, aFunction );\
TPtrC8 file( ( TText8* )__FILE__ );\
ERROR_2( aErr, "Trapped leave in '%S' line %d", &file, __LINE__);\
}
//-----------------------------------------------------------------------------
// Info trace macros
//-----------------------------------------------------------------------------
//
#ifdef INFO_TRACE
/**
* Info log message definitions.
*/
#ifdef TRACE_INTO_FILE
#define INFO( aMsg )\
{\
RFileLogger::Write( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ) );\
}
#define INFO_1( aMsg, aP1 )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ), aP1 );\
}
#define INFO_2( aMsg, aP1, aP2 )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ), aP1, aP2 );\
}
#define INFO_3( aMsg, aP1, aP2, aP3 )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ), aP1, aP2, aP3 );\
}
#define INFO_4( aMsg, aP1, aP2, aP3, aP4 )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ), aP1, aP2, aP3, aP4 );\
}
#define INFO_5( aMsg, aP1, aP2, aP3, aP4, aP5 )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ), aP1, aP2, aP3, aP4, aP5 );\
}
#define INFO_6( aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )\
{\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_INFO( aMsg ), aP1, aP2, aP3, aP4, aP5, aP6 );\
}
#else//TRACE_INTO_FILE not defined
#define INFO( aMsg )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ) );\
}
#define INFO_1( aMsg, aP1 )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ), aP1 );\
}
#define INFO_2( aMsg, aP1, aP2 )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ), aP1, aP2 );\
}
#define INFO_3( aMsg, aP1, aP2, aP3 )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ), aP1, aP2, aP3 );\
}
#define INFO_4( aMsg, aP1, aP2, aP3, aP4 )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ), aP1, aP2, aP3, aP4 );\
}
#define INFO_5( aMsg, aP1, aP2, aP3, aP4, aP5 )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ), aP1, aP2, aP3, aP4, aP5 );\
}
#define INFO_6( aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )\
{\
RDebug::Print( _PREFIX_INFO( aMsg ), aP1, aP2, aP3, aP4, aP5, aP6 );\
}
#endif//TRACE_INTO_FILE
#else//INFO_TRACE not defined
#define INFO( aMsg )
#define INFO_1( aMsg, aP1 )
#define INFO_2( aMsg, aP1, aP2 )
#define INFO_3( aMsg, aP1, aP2, aP3 )
#define INFO_4( aMsg, aP1, aP2, aP3, aP4 )
#define INFO_5( aMsg, aP1, aP2, aP3, aP4, aP5 )
#define INFO_6( aMsg, aP1, aP2, aP3, aP4, aP5, aP6 )
#endif//INFO_TRACE
//-----------------------------------------------------------------------------
// Trace current client thread name and process id
//-----------------------------------------------------------------------------
//
#ifdef CLIENT_TRACE
#define CLIENT_PROCESS\
{\
CLIENT_PROCESS_PREFIX( "" );\
}
#define CLIENT_PROCESS_PREFIX( aPrefix )\
{\
RProcess process;\
TPtrC name( process.Name() );\
TSecureId sid( process.SecureId() );\
TPtrC prefix( _S( aPrefix ) );\
if( prefix.Length() )\
{\
INFO_3( "%S: CLIENT - Name: [%S], Sid: [0x%x]", &prefix, &name, sid.iId );\
}\
else\
{\
INFO_2( "CLIENT - Name: [%S], Sid: [0x%x]", &name, sid.iId );\
}\
process.Close();\
}
#define CLIENT_MESSAGE( aMsg )\
{\
CLIENT_MESSAGE_PREFIX( "", aMsg );\
}
#define CLIENT_MESSAGE_PREFIX( aPrefix, aMsg )\
{\
RThread thread;\
TInt err = aMsg.Client( thread );\
if( err == KErrNone )\
{\
RProcess process;\
err = thread.Process( process );\
if( err == KErrNone )\
{\
TPtrC threadName( thread.Name() );\
TUid processUid( process.SecureId() );\
TPtrC prefix( _S( aPrefix ) );\
if( prefix.Length() )\
{\
INFO_4( "%S: MSG - Name: [%S], Sid: [0x%x], Message ID: [%d]",\
&prefix,\
&threadName,\
processUid,\
aMsg.Function() );\
}\
else\
{\
INFO_3( "MSG - Name: [%S], Sid: [0x%x], Message ID: [%d]",\
&threadName,\
processUid,\
aMsg.Function() );\
}\
}\
process.Close();\
}\
thread.Close();\
}
#else
#define CLIENT_PROCESS
#define CLIENT_PROCESS_PREFIX( aPrefix )
#define CLIENT_MESSAGE( aMsg )
#define CLIENT_MESSAGE_PREFIX( aPrefix, aMsg )
#endif
//-----------------------------------------------------------------------------
// Function trace macros
//-----------------------------------------------------------------------------
//
#ifdef FUNC_TRACE
/**
* Function logging definitions.
*/
#ifdef TRACE_INTO_FILE
#define FUNC( aMsg, aP1 )\
{\
TPtrC8 trace( _S8( aMsg ) );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, trace, aP1 );\
}\
#else//TRACE_INTO_FILE not defined
#define FUNC( aMsg, aP1 )\
{\
RDebug::Printf( aMsg, aP1 );\
}\
#endif//TRACE_INTO_FILE
/**
* Function trace helper class.
*
* NOTE:
* LC -methods cannot be trapped. Therefore if LC -method leaves
* END trace is used instead of LEAVE trace.
* If you have an idea how to round this problem please tell.
*/
_LIT8( KFuncNameTerminator, "(" );
_LIT8( KFuncLeavePatternL, "L" );
class TFuncLog
{
public:
static void Cleanup( TAny* aPtr )
{
TFuncLog* self = static_cast< TFuncLog* >( aPtr );
self->iLeft = ETrue;
FUNC( _PREFIX_CHAR("%S-LEAVE"), &self->iFunc ); // Leave detected
}
inline TFuncLog( const char* aFunc ) :
iFunc( aFunc ? _S8( aFunc ) : _S8("") ),
iLeft( EFalse ),
iCleanupItem( Cleanup, this ),
iCanLeave( EFalse )
{
TInt pos( iFunc.Find( KFuncNameTerminator ) );
if( pos != KErrNotFound )
{
iFunc.Set( iFunc.Left( pos ) );
iCanLeave = !iFunc.Right( KFuncLeavePatternL().Length() ).Compare( KFuncLeavePatternL );
if ( iCanLeave )
{
CleanupStack::PushL( iCleanupItem ); // Ignore warnings
}
}
FUNC( _PREFIX_CHAR("%S-START"), &iFunc );
}
inline ~TFuncLog()
{
if ( !iLeft )
{
if ( iCanLeave )
{
CleanupStack::Pop( this ); // Pop the cleanup item
}
FUNC( _PREFIX_CHAR("%S-END"), &iFunc ); // Normally finished
}
}
private: // Data
TPtrC8 iFunc;
TBool iLeft;
TCleanupItem iCleanupItem;
TBool iCanLeave;
};
#define FUNC_LOG TFuncLog _fl( __PRETTY_FUNCTION__ );
#else//FUNC_TRACE not defined
#define FUNC_LOG
#endif//FUNC_TRACE
//-----------------------------------------------------------------------------
// Timestamp trace macros
//-----------------------------------------------------------------------------
//
#ifdef TIMESTAMP_TRACE
#ifdef TRACE_INTO_FILE
#define TIMESTAMP( aCaption )\
{\
TTime t;\
t.HomeTime();\
TDateTime dt = t.DateTime();\
_LIT( KCaption, aCaption );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend,\
_PREFIX_TRACE("[TIMESTAMP] %S %d:%02d:%02d.%d us"),\
&KCaption, dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond() );\
}
#else//TRACE_INTO_FILE not defined
#define TIMESTAMP( aCaption )\
{\
TTime t;\
t.HomeTime();\
TDateTime dt = t.DateTime();\
_LIT( KCaption, aCaption );\
RDebug::Print( _PREFIX_TRACE("[TIMESTAMP] %S %d:%02d:%02d.%d us"),\
&KCaption, dt.Hour(), dt.Minute(), dt.Second(), dt.MicroSecond() );\
}
#endif//TRACE_INTO_FILE
#else//TIMESTAMP_TRACE not defined
#define TIMESTAMP( aCaption )
#endif//TIMESTAMP_TRACE
#ifdef HEAP_TRACE
#ifdef TRACE_INTO_FILE
#define HEAP( aMsg )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_HEAP( aMsg ), totalAllocSpace );\
}
#define HEAP_1( aMsg, aP1 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1 );\
}
#define HEAP_2( aMsg, aP1, aP2 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1, aP2 );\
}
#define HEAP_3( aMsg, aP1, aP2, aP3 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1, aP2, aP3 );\
}
#define HEAP_4( aMsg, aP1, aP2, aP3, aP4 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RFileLogger::WriteFormat( KDir, KFile, EFileLoggingModeAppend, _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1, aP2, aP3, aP4 );\
}
#else//TRACE_INTO_FILE not defined
#define HEAP( aMsg )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RDebug::Print( _PREFIX_HEAP( aMsg ), totalAllocSpace );\
}
#define HEAP_1( aMsg, aP1 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RDebug::Print( _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1 );\
}
#define HEAP_2( aMsg, aP1, aP2 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RDebug::Print( _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1, aP2 );\
}
#define HEAP_3( aMsg, aP1, aP2, aP3 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RDebug::Print( _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1, aP2, aP3 );\
}
#define HEAP_4( aMsg, aP1, aP2, aP3, aP4 )\
{\
TInt totalAllocSpace = 0;\
User::AllocSize( totalAllocSpace );\
RDebug::Print( _PREFIX_HEAP( aMsg ), totalAllocSpace, aP1, aP2, aP3, aP4 );\
}
#endif//TRACE_INTO_FILE
#else//HEAP_TRACE not defined
#define HEAP( aMsg )
#define HEAP_1( aMsg, aP1 )
#define HEAP_2( aMsg, aP1, aP2 )
#define HEAP_3( aMsg, aP1, aP2, aP3 )
#define HEAP_4( aMsg, aP1, aP2, aP3, aP4 )
#endif//HEAP_TRACE
#endif