You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
881 lines
16 KiB
881 lines
16 KiB
/*++
|
|
|
|
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
|