207 lines
4.4 KiB
C
207 lines
4.4 KiB
C
/*++
|
||
|
||
Copyright (c) 1989 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
blkdebug.c
|
||
|
||
Abstract:
|
||
|
||
Contains routines for debugging reference count problems.
|
||
|
||
Author:
|
||
|
||
David Treadwell (davidtr) 30-Sept-1991
|
||
|
||
Revision History:
|
||
|
||
--*/
|
||
|
||
#include "precomp.h"
|
||
#pragma hdrstop
|
||
|
||
//
|
||
// This entire module is conditionalized out if SRVDBG2 is not defined.
|
||
//
|
||
|
||
#if SRVDBG2
|
||
|
||
#ifdef ALLOC_PRAGMA
|
||
#pragma alloc_text( PAGE, SrvInitializeReferenceHistory )
|
||
#pragma alloc_text( PAGE, SrvTerminateReferenceHistory )
|
||
#endif
|
||
#if 0
|
||
NOT PAGEABLE -- SrvUpdateReferenceHistory
|
||
NOT PAGEABLE -- SrvdbgClaimOrReleaseHandle
|
||
#endif
|
||
|
||
|
||
VOID
|
||
SrvInitializeReferenceHistory (
|
||
IN PBLOCK_HEADER Block,
|
||
IN ULONG InitialReferenceCount
|
||
)
|
||
|
||
{
|
||
PVOID caller, callersCaller;
|
||
|
||
ULONG historyTableSize = sizeof(REFERENCE_HISTORY_ENTRY) *
|
||
REFERENCE_HISTORY_LENGTH;
|
||
|
||
PAGED_CODE( );
|
||
|
||
Block->History.HistoryTable = ALLOCATE_NONPAGED_POOL(
|
||
historyTableSize,
|
||
BlockTypeDataBuffer
|
||
);
|
||
//
|
||
// It we weren't able to allocate the memory, don't track references
|
||
// and dereferences.
|
||
//
|
||
|
||
if ( Block->History.HistoryTable == NULL ) {
|
||
Block->History.NextEntry = -1;
|
||
} else {
|
||
Block->History.NextEntry = 0;
|
||
RtlZeroMemory( Block->History.HistoryTable, historyTableSize );
|
||
}
|
||
|
||
Block->History.TotalReferences = 0;
|
||
Block->History.TotalDereferences = 0;
|
||
|
||
//
|
||
// Account for the initial reference(s).
|
||
//
|
||
|
||
RtlGetCallersAddress( &caller, &callersCaller );
|
||
|
||
while ( InitialReferenceCount-- > 0 ) {
|
||
SrvUpdateReferenceHistory( Block, caller, callersCaller, FALSE );
|
||
}
|
||
|
||
return;
|
||
|
||
} // SrvInitializeReferenceHistory
|
||
|
||
|
||
VOID
|
||
SrvUpdateReferenceHistory (
|
||
IN PBLOCK_HEADER Block,
|
||
IN PVOID Caller,
|
||
IN PVOID CallersCaller,
|
||
IN BOOLEAN IsDereference
|
||
)
|
||
|
||
{
|
||
KIRQL oldIrql;
|
||
|
||
ACQUIRE_GLOBAL_SPIN_LOCK( Debug, &oldIrql );
|
||
|
||
if ( IsDereference ) {
|
||
Block->History.TotalDereferences++;
|
||
} else {
|
||
Block->History.TotalReferences++;
|
||
}
|
||
|
||
if ( Block->History.HistoryTable != 0 ) {
|
||
|
||
PREFERENCE_HISTORY_ENTRY entry;
|
||
PREFERENCE_HISTORY_ENTRY priorEntry;
|
||
|
||
entry = &Block->History.HistoryTable[ Block->History.NextEntry ];
|
||
|
||
if ( Block->History.NextEntry == 0 ) {
|
||
priorEntry =
|
||
&Block->History.HistoryTable[ REFERENCE_HISTORY_LENGTH-1 ];
|
||
} else {
|
||
priorEntry =
|
||
&Block->History.HistoryTable[ Block->History.NextEntry-1 ];
|
||
}
|
||
|
||
entry->Caller = Caller;
|
||
entry->CallersCaller = CallersCaller;
|
||
|
||
if ( IsDereference ) {
|
||
entry->NewReferenceCount = priorEntry->NewReferenceCount - 1;
|
||
entry->IsDereference = (ULONG)TRUE;
|
||
} else {
|
||
entry->NewReferenceCount = priorEntry->NewReferenceCount + 1;
|
||
entry->IsDereference = (ULONG)FALSE;
|
||
}
|
||
|
||
Block->History.NextEntry++;
|
||
|
||
if ( Block->History.NextEntry >= REFERENCE_HISTORY_LENGTH ) {
|
||
Block->History.NextEntry = 0;
|
||
}
|
||
}
|
||
|
||
RELEASE_GLOBAL_SPIN_LOCK( Debug, oldIrql );
|
||
|
||
} // SrvUpdateReferenceHistory
|
||
|
||
|
||
VOID
|
||
SrvTerminateReferenceHistory (
|
||
IN PBLOCK_HEADER Block
|
||
)
|
||
|
||
{
|
||
PAGED_CODE( );
|
||
|
||
if ( Block->History.HistoryTable != 0 ) {
|
||
DEALLOCATE_NONPAGED_POOL( Block->History.HistoryTable );
|
||
}
|
||
|
||
return;
|
||
|
||
} // SrvTerminateReferenceHistory
|
||
|
||
#endif // SRVDBG2
|
||
|
||
|
||
#if SRVDBG_HANDLES
|
||
|
||
#define HANDLE_HISTORY_SIZE 512
|
||
|
||
struct {
|
||
ULONG HandleTypeAndOperation;
|
||
PVOID Handle;
|
||
ULONG Location;
|
||
PVOID Data;
|
||
} HandleHistory[HANDLE_HISTORY_SIZE];
|
||
|
||
ULONG HandleHistoryIndex = 0;
|
||
|
||
VOID
|
||
SrvdbgClaimOrReleaseHandle (
|
||
IN HANDLE Handle,
|
||
IN PSZ HandleType,
|
||
IN ULONG Location,
|
||
IN BOOLEAN Release,
|
||
IN PVOID Data
|
||
)
|
||
{
|
||
ULONG index;
|
||
KIRQL oldIrql;
|
||
|
||
ACQUIRE_GLOBAL_SPIN_LOCK( Debug, &oldIrql );
|
||
index = HandleHistoryIndex;
|
||
if ( ++HandleHistoryIndex >= HANDLE_HISTORY_SIZE ) {
|
||
HandleHistoryIndex = 0;
|
||
}
|
||
RELEASE_GLOBAL_SPIN_LOCK( Debug, oldIrql );
|
||
|
||
HandleHistory[index].HandleTypeAndOperation =
|
||
(*(PULONG)HandleType << 8) | (Release ? 'c' : 'o');
|
||
HandleHistory[index].Handle = Handle;
|
||
HandleHistory[index].Location = Location;
|
||
HandleHistory[index].Data = Data;
|
||
|
||
return;
|
||
|
||
} // SrvdbgClaimOrReleaseHandle
|
||
|
||
#endif // SRVDBG_HANDLES
|