|
|
/*++
Copyright (c) 1996 Microsoft Corporation All rights reserved.
Module Name:
trace.cxx
Abstract:
Holds logging routines.
Author:
Albert Ting (AlbertT) 24-May-1996
Revision History:
--*/
#include "spllibp.hxx"
#pragma hdrstop
#if DBG
#include "trace.hxx"
/*
** Turn off memory tracing. Turning this on keeps all the back traces in ** memory. #if i386
#define BACKTRACE_ENABLED
#endif
*/
CRITICAL_SECTION gcsBackTrace;
#ifdef TRACE_ENABLED
TBackTraceDB* gpBackTraceDB;
/********************************************************************
BackTrace DB
********************************************************************/
TBackTraceDB:: TBackTraceDB( VOID ) : _pTraceHead( NULL )
/*++
Routine Description:
Initialize the trace database.
Generally you will have just one database that holds all the traces.
Arguments:
Return Value:
--*/
{ _pMemBlock = new TMemBlock( kBlockSize, TMemBlock::kFlagGlobalNew ); }
TBackTraceDB:: ~TBackTraceDB( VOID )
/*++
Routine Description:
Destroy the back trace database.
Arguments:
Return Value:
--*/
{ delete _pMemBlock; }
BOOL TBackTraceDB:: bValid( VOID ) { return _pMemBlock && _pMemBlock->bValid(); }
HANDLE TBackTraceDB:: hStore( IN ULONG ulHash, IN PVOID pvBackTrace )
/*++
Routine Description:
Store a backtrace into the database.
Arguments:
ulHash - Hash for this backtrace.
pvBackTrace - Actual backtrace; must be NULL terminated.
Return Value:
HANDLE - backtrace handle.
--*/
{ TTrace *ptRet; TTrace **ppTrace;
//
// First see if we can find a backtrace. If we can't, then
// pTrace will hold the slot where it should be.
//
ptRet = ptFind( ulHash, pvBackTrace, &ppTrace );
if( !ptRet ){
//
// Didn't find one; add it.
//
ptRet = TTrace::pNew( this, ulHash, pvBackTrace, ppTrace ); }
return ptRet; }
PLONG TBackTraceDB:: plGetCount( HANDLE hData )
/*++
Routine Description:
Get data from a HANDLE retrieved from hStore. There is one ULONG per stack backtrace.
Arguments:
hData - Returned from hStore.
Return Value:
PLONG.
--*/
{ TTrace *ptTrace = static_cast<TTrace*>( hData ); return &ptTrace->lCount(); }
TBackTraceDB::TTrace* TBackTraceDB:: ptFind( IN ULONG ulHash, IN PVOID pvBackTrace, OUT TTrace ***pppTrace OPTIONAL )
/*++
Routine Description:
Find a backtrace in the database. If one does not exist, then return NULL and a pointer to where it would exist in the database.
Arguments:
ulHash - Hash of the backtrace.
pvBackTrace - Backtrace to find.
pppTrace - If not found, this holds the address of where it should be stored in the database. Adding the trace here is sufficient to add it.
Return Value:
TTrace* the actual trace, NULL if not found.
--*/
{ //
// Traverse the binary tree until we find the end or the
// right one.
//
TTrace **ppTrace = &_pTraceHead;
while( *ppTrace ){
//
// Check if this one matches ours.
//
COMPARE Compare = (*ppTrace)->eCompareHash( ulHash );
if( Compare == kEqual ){
//
// Now do slow compare in case the hash is a collision.
//
Compare = (*ppTrace)->eCompareBackTrace( pvBackTrace );
if( Compare == kEqual ){
//
// Break out of while loop and quit.
//
break; } }
ppTrace = ( Compare == kLess ) ? &(*ppTrace)->_pLeft : &(*ppTrace)->_pRight; }
if( pppTrace ){ *pppTrace = ppTrace; } return *ppTrace; }
/********************************************************************
TBackTraceDB::TTrace
********************************************************************/
COMPARE TBackTraceDB:: TTrace:: eCompareHash( ULONG ulHash ) const
/*++
Routine Description:
Quickly compare two trace hashes.
Arguments:
ulHash - Input hash.
Return Value:
--*/
{ if( _ulHash < ulHash ){ return kLess; }
if( _ulHash > ulHash ){ return kGreater; }
return kEqual; }
COMPARE TBackTraceDB:: TTrace:: eCompareBackTrace( PVOID pvBackTrace ) const
/*++
Routine Description:
Compare backtrace to one stored in this.
Arguments:
pvBackTrace - Must be NULL terminated.
Return Value:
COMAARE: kLess, kEqual, kGreater.
--*/
{ PVOID *pSrc; PVOID *pDest;
for( pSrc = (PVOID*)this, pDest = (PVOID*)&pvBackTrace; *pSrc && *pDest; pSrc++, pDest++ ) {
if ( *pSrc != *pDest ){ return (ULONG_PTR)*pSrc < (ULONG_PTR)*pDest ? kLess : kGreater; } } return kEqual; }
TBackTraceDB::TTrace* TBackTraceDB:: TTrace:: pNew( IN TBackTraceDB *pBackTraceDB, IN ULONG ulHash, IN PVOID pvBackTrace, OUT TTrace ** ppTrace )
/*++
Routine Description:
Constructs a new TTrace and puts it in pBackTraceDB.
Assumes the trace does _not_ exist already, and ppTrace points to the place where it should be stored to ensure the database is kept consistent.
Arguments:
pBackTraceDB - Storage for the new trace.
ulHash - Hash for the trace.
pvBackTrace - The actual backtrace.
ppTrace - Where the trace should be stored in the database.
Return Value:
TTrace* - New trace, NULL if failed.
--*/
{ COUNT cCalls; PVOID *ppvCalls;
//
// Calculate size of backtrace. Start with cCalls = 1 so that
// we include 1 extra for the NULL terminator.
//
for( ppvCalls = (PVOID*)pvBackTrace, cCalls = 1; *ppvCalls; ++ppvCalls, ++cCalls )
;
++cCalls;
COUNTB cbSize = OFFSETOF( TTrace, apvBackTrace ) + cCalls * sizeof( PVOID );
TTrace* pTrace = (TTrace*)pBackTraceDB->_pMemBlock->pvAlloc( cbSize );
if( pTrace ){
pTrace->_pLeft = NULL; pTrace->_pRight = NULL; pTrace->_ulHash = ulHash; pTrace->_lCount = -1;
CopyMemory( pTrace->apvBackTrace, (PVOID*)pvBackTrace, cCalls * sizeof( PVOID ));
//
// Add it in the right spot into the database.
//
*ppTrace = pTrace; }
return pTrace; }
/********************************************************************
Back tracing: abstract base class.
********************************************************************/
BOOL VBackTrace::gbInitialized = FALSE;
#endif // TRACE_ENABLED
VBackTrace:: VBackTrace( ULONG_PTR fOptions1, ULONG_PTR fOptions2 ) : _fOptions1( fOptions1 ), _fOptions2( fOptions2 ) { }
VBackTrace:: ~VBackTrace( VOID ) { }
BOOL VBackTrace:: bInit( VOID ) {
#ifdef TRACE_ENABLED
gbInitialized = InitializeCriticalSectionAndSpinCount(&gcsBackTrace, 0x80000000); gpBackTraceDB = new TBackTraceDB();
return gbInitialized && (gpBackTraceDB != NULL);
#else
return TRUE;
#endif
}
VOID VBackTrace:: vDone( VOID ) {
#ifdef TRACE_ENABLED
if( gbInitialized ) { DeleteCriticalSection(&gcsBackTrace); }
#endif
}
PLONG VBackTrace:: plGetCount( HANDLE hData ) {
#ifdef TRACE_ENABLED
return gpBackTraceDB->plGetCount( hData );
#else
return NULL;
#endif
}
#ifndef TRACE_ENABLED
HANDLE VBackTrace:: hCapture( ULONG_PTR Info1, ULONG_PTR Info2, ULONG_PTR Info3, PULONG pHash )
/*++
Routine Description:
In the case that tracing is disabled, this function is coded to return NULL.
Arguments:
Return Value:
NULL
--*/
{ return NULL; }
#endif // ndef TRACE_ENABLED
#ifdef TRACE_ENABLED
/********************************************************************
Back tracing to memory.
********************************************************************/
TBackTraceMem:: TBackTraceMem( ULONG_PTR fOptions1, ULONG_PTR fOptions2 ) : VBackTrace( fOptions1, fOptions2 ), _uNextFree( 0 ) { _pLines = new TLine[kMaxCall]; if( _pLines ){ ZeroMemory( _pLines, sizeof( TLine[kMaxCall] )); } }
TBackTraceMem:: ~TBackTraceMem( VOID ) { UINT i; TLine* pLine;
if( _pLines ){ for( i=0, pLine = _pLines; i< kMaxCall; i++, pLine++ ){
if( _fOptions1 & kString ){ DbgFreeMem( (PVOID)pLine->_Info1 ); }
if( _fOptions2 & kString ){ DbgFreeMem( (PVOID)pLine->_Info2 ); } } delete [] _pLines; } }
VOID TBackTraceMem:: vCaptureLine( IN OUT TLine* pLine, IN ULONG_PTR Info1, IN ULONG_PTR Info2, IN ULONG_PTR Info3, OUT PVOID apvBackTrace[kMaxDepth+1], OPTIONAL OUT PULONG pulHash OPTIONAL )
/*++
Routine Description:
Captures information into a TLine structure; freeing previous contents if necessary.
Arguments:
pLine - Fully initialized pLine structure. On output, everything _except_ _hTrace is filled in.
** Both apvBackTrace && pulHash must both be valid if either is valid **
apvBackTrace - Buffer to receive backtrace.
pulHash - Buffer to receive ulHash.
Return Value:
--*/
{ //
// Free memory if necessary.
//
if( _fOptions1 & kString ) { DbgFreeMem( (PVOID)pLine->_Info1 ); }
if( _fOptions2 & kString ) { DbgFreeMem( (PVOID)pLine->_Info2 ); }
pLine->_TickCount = GetTickCount(); pLine->_Info1 = Info1; pLine->_Info2 = Info2; pLine->_Info3 = Info3;
pLine->_ThreadId = GetCurrentThreadId(); pLine->_hTrace = NULL;
#ifdef BACKTRACE_ENABLED
if( apvBackTrace && pulHash ){
ULONG ulHash;
//
// Capture a backtrace at this spot for debugging.
//
UINT uDepth = RtlCaptureStackBackTrace( 2, kMaxDepth, apvBackTrace, pulHash );
//
// NULL terminate.
//
apvBackTrace[uDepth] = NULL; } #else
apvBackTrace[0] = NULL; *pulHash = 0; #endif
}
HANDLE TBackTraceMem:: hCapture( ULONG_PTR Info1, ULONG_PTR Info2, ULONG_PTR Info3, PULONG pHash ) { UINT uDepth; TLine* pLine; ULONG ulHash; PVOID apvBackTrace[kMaxDepth+1];
if( !_pLines ){ return NULL; }
EnterCriticalSection( &gcsBackTrace );
pLine = &_pLines[_uNextFree];
vCaptureLine( pLine, Info1, Info2, Info3, apvBackTrace, &ulHash ); pLine->_hTrace = gpBackTraceDB->hStore( ulHash, apvBackTrace );
_uNextFree++;
if( _uNextFree == kMaxCall ) _uNextFree = 0;
LeaveCriticalSection( &gcsBackTrace );
if( pHash ) { *pHash = ulHash; }
return (PVOID)pLine->_hTrace; }
/********************************************************************
Backtracing to File.
********************************************************************/
COUNT TBackTraceFile::gcInstances;
TBackTraceFile:: TBackTraceFile( ULONG_PTR fOptions1, ULONG_PTR fOptions2 ) : VBackTrace( fOptions1, fOptions2 ) { TCHAR szFile[kMaxPath];
EnterCriticalSection( &gcsBackTrace );
StringCchPrintf( szFile, COUNTOF(szFile), TEXT( "spl_%d.%d.log" ), GetCurrentProcessId(), gcInstances );
++gcInstances;
LeaveCriticalSection( &gcsBackTrace );
_hFile = CreateFile( szFile, GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_COMPRESSED, NULL );
if( _hFile == INVALID_HANDLE_VALUE ){
OutputDebugStringA( "SPLLIB: Unable to open file " ); OutputDebugString( szFile ); OutputDebugStringA( "\n" ); return; } }
TBackTraceFile:: ~TBackTraceFile( VOID ) { if( _hFile != INVALID_HANDLE_VALUE ){ CloseHandle( _hFile ); } }
HANDLE TBackTraceFile:: hCapture( ULONG_PTR Info1, ULONG_PTR Info2, ULONG_PTR Info3, PULONG pHash ) { TLine Line; PVOID apvBackTrace[kMaxDepth+1]; DWORD cbWritten;
CHAR szLine[kMaxLineStr]; szLine[0] = 0;
#ifdef BACKTRACE_ENABLED
ULONG ulHash;
//
// Capture a backtrace at this spot for debugging.
//
UINT uDepth = RtlCaptureStackBackTrace( 2, kMaxDepth, apvBackTrace, &ulHash ); #endif
EnterCriticalSection( &gcsBackTrace );
//
// Print out strings as appropriate.
//
if( _fOptions1 & kString ) { WriteFile( _hFile, (LPCVOID)Info1, lstrlenA( (LPCSTR)Info1 ), &cbWritten, NULL ); }
if( _fOptions2 & kString ) { WriteFile( _hFile, (LPCVOID)Info2, lstrlenA( (LPCSTR)Info2 ), &cbWritten, NULL ); }
//
// Print out the hex info.
//
StringCchPrintfA(szLine, COUNTOF(szLine), "\n\t%08x: %08x %08x %08x threadid=%x tc=%x < %x >: ", this, Info1, Info2, Info3, GetCurrentThreadId(), GetTickCount(), Info1 + Info2 );
if( _hFile ) { WriteFile( _hFile, szLine, lstrlenA( szLine ), &cbWritten, NULL ); }
#ifdef BACKTRACE_ENABLED
//
// Print out the backtrace.
//
UINT i; szLine[0] = '\t'; CHAR *pszLineEnd = &szLine[1]; size_t cRemaining = COUNTOF(szLine)-1; for( i=0; i < uDepth; ++i ) { StringCchPrintfAEx( pszLineEnd, cRemaining, pszLineEnd, cRemaining, "%08x ", apvBackTrace[i] ); }
if( _hFile && i ) { //
// Countof(szLine)-cRemaining is the number of characters that were copied to szLine.
// We add 2 for the first character and the null character at the end.
//
WriteFile( _hFile, szLine, COUNTOF(szLine)-cRemaining+2, &cbWritten, NULL ); }
#endif
//
// Add extra blank line.
//
szLine[0] = '\n'; WriteFile( _hFile, szLine, 1, &cbWritten, NULL );
LeaveCriticalSection( &gcsBackTrace );
//
// Free memory if necessary.
//
if( _fOptions1 & kString ) { DbgFreeMem( (PVOID)Info1 ); }
if( _fOptions2 & kString ) { DbgFreeMem( (PVOID)Info2 ); }
#ifdef BACKTRACE_ENABLED
if( pHash ) { *pHash = ulHash; }
#endif
return NULL; }
#endif // TRACE_ENABLED
#endif // #ifdef DBG
|