2798 lines
72 KiB
C++
2798 lines
72 KiB
C++
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1995 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
debugmem.cxx
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
Debug memory allocator
|
|||
|
|
|||
|
Contents:
|
|||
|
InternetDebugMemInitialize
|
|||
|
InternetDebugMemTerminate
|
|||
|
InternetDebugAllocMem
|
|||
|
InternetDebugFreeMem
|
|||
|
InternetDebugReAllocMem
|
|||
|
InternetDebugSizeMem
|
|||
|
InternetDebugCheckMemFreed
|
|||
|
InternetDebugMemReport
|
|||
|
(InternetDebugCheckMemBlock)
|
|||
|
(DebugFillMem)
|
|||
|
(InternetAlloc)
|
|||
|
(InternetFree)
|
|||
|
(InternetReAlloc)
|
|||
|
(InternetSize)
|
|||
|
(InternetHeapAlloc)
|
|||
|
(InternetHeapReAlloc)
|
|||
|
(InternetHeapFree)
|
|||
|
(InternetHeapSize)
|
|||
|
(InternetDebugMemTest)
|
|||
|
(ReportMemoryUsage)
|
|||
|
(ReportMemoryBlocks)
|
|||
|
(DumpDeferredFreeList)
|
|||
|
(DumpMemoryList)
|
|||
|
(FindAndDumpDeferredBlock)
|
|||
|
(DumpBlock)
|
|||
|
(DumpDebugMemoryHeader)
|
|||
|
(DumpDebugMemoryFooter)
|
|||
|
(DumpUserData)
|
|||
|
(MapLastAccessOperation)
|
|||
|
(MapMemoryFlags)
|
|||
|
(DbgMemGetDebugSymbol)
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Richard L Firth (rfirth) 02-Feb-1995
|
|||
|
|
|||
|
Environment:
|
|||
|
|
|||
|
Win32 user mode
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
02-Feb-1995
|
|||
|
Created
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include <wininetp.h>
|
|||
|
#include "rprintf.h"
|
|||
|
|
|||
|
#if defined(USE_DEBUG_MEMORY)
|
|||
|
|
|||
|
//
|
|||
|
// manifests
|
|||
|
//
|
|||
|
|
|||
|
#define DEFAULT_INITIAL_HEAP_SIZE (64 K)
|
|||
|
#define DEFAULT_MAXIMUM_HEAP_SIZE (1 M)
|
|||
|
#define DEFAULT_HEADER_GUARD_SIZE 32
|
|||
|
#define DEFAULT_FOOTER_GUARD_SIZE 32
|
|||
|
#define DEFAULT_ALLOC_ALIGNMENT 4
|
|||
|
#define HEADER_SIGNATURE 0x414d454d // "MEMA"
|
|||
|
#define FOOTER_SIGNATURE 0x434f4c4c // "LLOC"
|
|||
|
#define DWORD_ALLOC_FILL 0xc5c5c5c5
|
|||
|
#define BYTE_ALLOC_FILL 0xc5
|
|||
|
#define BYTE_ALLOC_FILL_EXTRA 0x88
|
|||
|
#define GUARD_DWORD_FILL 0x44524147 // "GARD"
|
|||
|
#define DWORD_FREE_FILL 0xb7b7b7b7
|
|||
|
#define BYTE_FREE_FILL 0xb7
|
|||
|
#define DEFAULT_MAX_BLOCKS_DUMPED 1024
|
|||
|
#define DEFAULT_MAX_DATA_DUMPED 65536
|
|||
|
#define DEFAULT_BACKTRACE_DEPTH 2
|
|||
|
|
|||
|
//
|
|||
|
// only perform stack dump for x86 (or other stack-based processors)
|
|||
|
//
|
|||
|
|
|||
|
#if defined(i386)
|
|||
|
#define DUMP_STACK 1
|
|||
|
#else
|
|||
|
#define DUMP_STACK 0
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// just using one stack these days
|
|||
|
//
|
|||
|
|
|||
|
#define ONE_STACK 1
|
|||
|
|
|||
|
//
|
|||
|
// private types
|
|||
|
//
|
|||
|
|
|||
|
typedef enum {
|
|||
|
MemAllocate = 0x6f6c6c41, // "Allo"
|
|||
|
MemReallocate = 0x6c416552, // "ReAl"
|
|||
|
MemLock = 0x6b636f4c, // "Lock"
|
|||
|
MemUnlock = 0x6f6c6e55, // "Unlo"
|
|||
|
MemFree = 0x65657246, // "Free"
|
|||
|
MemSize = 0x657a6953 // "Size"
|
|||
|
} MEMORY_ACTION;
|
|||
|
|
|||
|
typedef enum {
|
|||
|
HEAP_COMPACT_NEVER = 0,
|
|||
|
HEAP_COMPACT_ON_ALLOC_FAIL,
|
|||
|
HEAP_COMPACT_ON_FREE
|
|||
|
} HEAP_COMPACT_TYPE;
|
|||
|
|
|||
|
typedef enum {
|
|||
|
HEAP_VALIDATE_NEVER = 0,
|
|||
|
HEAP_VALIDATE_ON_ALLOC,
|
|||
|
HEAP_VALIDATE_ON_FREE
|
|||
|
} HEAP_VALIDATE_TYPE;
|
|||
|
|
|||
|
//
|
|||
|
// DEBUG_MEMORY_HEADER - keeps debug memory on list
|
|||
|
//
|
|||
|
|
|||
|
typedef struct {
|
|||
|
LIST_ENTRY List;
|
|||
|
DWORD ThreadId;
|
|||
|
LPSTR CreatedFile;
|
|||
|
DWORD CreatedLine;
|
|||
|
LPSTR AccessedFile;
|
|||
|
DWORD AccessedLine;
|
|||
|
SIZE_T RequestedLength;
|
|||
|
SIZE_T BlockLength;
|
|||
|
SIZE_T ActualLength;
|
|||
|
DWORD Signature;
|
|||
|
DWORD Flags;
|
|||
|
DWORD TimeDeferred;
|
|||
|
LONG ClashTest;
|
|||
|
MEMORY_ACTION LastAccessOperation;
|
|||
|
#if DUMP_STACK
|
|||
|
#if ONE_STACK
|
|||
|
LPVOID Stack[8]; // should be variable
|
|||
|
#else
|
|||
|
LPVOID CreateStack[4];
|
|||
|
LPVOID LastAccessStack[4];
|
|||
|
#endif // ONE_STACK
|
|||
|
#endif // DUMP_STACK
|
|||
|
DWORD Guard[2];
|
|||
|
|
|||
|
//
|
|||
|
// sizeof(MEMORY_SIGNATURE) currently 24 DWORDs in Win32
|
|||
|
//
|
|||
|
|
|||
|
} DEBUG_MEMORY_HEADER, *LPDEBUG_MEMORY_HEADER;
|
|||
|
|
|||
|
//
|
|||
|
// DEBUG_MEMORY_FOOTER - used to check for overwrites
|
|||
|
//
|
|||
|
|
|||
|
typedef struct {
|
|||
|
DWORD Guard[4];
|
|||
|
DWORD Signature;
|
|||
|
SIZE_T BlockLength; // should be the same as the header
|
|||
|
DWORD Guard2[2];
|
|||
|
|
|||
|
//
|
|||
|
// sizeof(DEBUG_MEMORY_FOOTER) currently 8 DWORDs in Win32
|
|||
|
//
|
|||
|
|
|||
|
} DEBUG_MEMORY_FOOTER, *LPDEBUG_MEMORY_FOOTER;
|
|||
|
|
|||
|
//
|
|||
|
// private data
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE BOOL MemoryPackageInitialized = FALSE;
|
|||
|
|
|||
|
//
|
|||
|
// InternetDebugMemFlags - bitfield of flags controlling debug memory usage.
|
|||
|
// The default is no debug alloc (don't create header + footers) and to use
|
|||
|
// LocalAlloc() etc.
|
|||
|
//
|
|||
|
|
|||
|
//
|
|||
|
// BUGBUG - I'm making an assumption that the compiler thinks the bits have the
|
|||
|
// same values as I think they have. If not, it could mess up the
|
|||
|
// registry/environment flags
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE struct { // default value
|
|||
|
DWORD bNoDebugAlloc : 1; // 0x00000001 TRUE
|
|||
|
DWORD bUseLocalAlloc : 1; // 0x00000002 TRUE
|
|||
|
DWORD bUseSymbols : 1; // 0x00000004 FALSE
|
|||
|
DWORD bAssertOnMemoryErrors : 1; // 0x00000008 FALSE
|
|||
|
DWORD bFillMemoryOnAlloc : 1; // 0x00000010 FALSE
|
|||
|
DWORD bFillMemoryOnFree : 1; // 0x00000020 FALSE
|
|||
|
DWORD bReportMemoryUsage : 1; // 0x00000040 FALSE
|
|||
|
DWORD bReportUnfreedBlocks : 1; // 0x00000080 FALSE
|
|||
|
DWORD bReportMemoryFooters : 1; // 0x00000100 FALSE
|
|||
|
DWORD bReportUserData : 1; // 0x00000200 FALSE
|
|||
|
DWORD bStopDumpIfBadBlock : 1; // 0x00000400 FALSE
|
|||
|
DWORD bLimitUnfreedBlocks : 1; // 0x00000800 FALSE
|
|||
|
DWORD bLimitUserData : 1; // 0x00001000 FALSE
|
|||
|
DWORD bDumpAsDwords : 1; // 0x00002000 FALSE
|
|||
|
DWORD bHeapNoSerialize : 1; // 0x00004000 FALSE
|
|||
|
DWORD bHeapGenerateExceptions : 1; // 0x00008000 FALSE
|
|||
|
DWORD bHeapIsGrowable : 1; // 0x00010000 FALSE
|
|||
|
DWORD bDeferFree : 1; // 0x00020000 FALSE
|
|||
|
DWORD bDumpToFile : 1; // 0x00040000 FALSE
|
|||
|
} InternetDebugMemFlags = {
|
|||
|
TRUE, // no debug alloc
|
|||
|
TRUE, // use LocalAlloc()
|
|||
|
FALSE, // don't load debug symbols
|
|||
|
FALSE, // don't assert on memory errors
|
|||
|
FALSE, // don't fill memory on alloc
|
|||
|
FALSE, // don't fill memory on free
|
|||
|
FALSE, // don't report memory usage (stats)
|
|||
|
FALSE, // don't report unfreed blocks
|
|||
|
FALSE, // don't report memory footers (irrelevant)
|
|||
|
FALSE, // don't report user data (irrelevant)
|
|||
|
FALSE, // don't stop dump if bad block (irrelevant)
|
|||
|
FALSE, // don't limit dump of unfreed blocks (irrelevant)
|
|||
|
FALSE, // don't limit dump of user data (irrelevant)
|
|||
|
FALSE, // don't dump user data as DWORDs (irrelevant)
|
|||
|
FALSE, // serialize access to heap (irrelevant)
|
|||
|
FALSE, // don't generate heap exceptions (irrelevant)
|
|||
|
TRUE, // heap is growable (irrelevant)
|
|||
|
FALSE, // don't defer frees
|
|||
|
FALSE // don't dump to wininet log file
|
|||
|
};
|
|||
|
|
|||
|
//
|
|||
|
// defines to make using InternetDebugMemFlags easier
|
|||
|
//
|
|||
|
|
|||
|
#define bNoDebugAlloc InternetDebugMemFlags.bNoDebugAlloc
|
|||
|
#define bUseLocalAlloc InternetDebugMemFlags.bUseLocalAlloc
|
|||
|
#define bUseSymbols InternetDebugMemFlags.bUseSymbols
|
|||
|
#define bAssertOnMemoryErrors InternetDebugMemFlags.bAssertOnMemoryErrors
|
|||
|
#define bFillMemoryOnAlloc InternetDebugMemFlags.bFillMemoryOnAlloc
|
|||
|
#define bFillMemoryOnFree InternetDebugMemFlags.bFillMemoryOnFree
|
|||
|
#define bReportMemoryUsage InternetDebugMemFlags.bReportMemoryUsage
|
|||
|
#define bReportUnfreedBlocks InternetDebugMemFlags.bReportUnfreedBlocks
|
|||
|
#define bReportMemoryFooters InternetDebugMemFlags.bReportMemoryFooters
|
|||
|
#define bReportUserData InternetDebugMemFlags.bReportUserData
|
|||
|
#define bStopDumpIfBadBlock InternetDebugMemFlags.bStopDumpIfBadBlock
|
|||
|
#define bLimitUnfreedBlocks InternetDebugMemFlags.bLimitUnfreedBlocks
|
|||
|
#define bLimitUserData InternetDebugMemFlags.bLimitUserData
|
|||
|
#define bDumpAsDwords InternetDebugMemFlags.bDumpAsDwords
|
|||
|
#define bHeapNoSerialize InternetDebugMemFlags.bHeapNoSerialize
|
|||
|
#define bHeapGenerateExceptions InternetDebugMemFlags.bHeapGenerateExceptions
|
|||
|
#define bHeapIsGrowable InternetDebugMemFlags.bHeapIsGrowable
|
|||
|
#define bDeferFree InternetDebugMemFlags.bDeferFree
|
|||
|
#define bDumpToFile InternetDebugMemFlags.bDumpToFile
|
|||
|
|
|||
|
PRIVATE DWORD MaxBlocksDumped = DEFAULT_MAX_BLOCKS_DUMPED;
|
|||
|
PRIVATE DWORD MaxUserDataDumped = DEFAULT_MAX_DATA_DUMPED;
|
|||
|
PRIVATE DWORD StackBacktraceDepth = DEFAULT_BACKTRACE_DEPTH;
|
|||
|
|
|||
|
//
|
|||
|
// heap variables
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE HANDLE hDebugHeap = NULL;
|
|||
|
PRIVATE DWORD InitialHeapSize = DEFAULT_INITIAL_HEAP_SIZE;
|
|||
|
PRIVATE DWORD MaximumHeapSize = DEFAULT_MAXIMUM_HEAP_SIZE;
|
|||
|
PRIVATE HEAP_COMPACT_TYPE HeapCompactControl = HEAP_COMPACT_NEVER;
|
|||
|
PRIVATE HEAP_VALIDATE_TYPE HeapValidateControl = HEAP_VALIDATE_NEVER;
|
|||
|
|
|||
|
//
|
|||
|
// debug mem signatures etc.
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE DWORD AllocAlignment = DEFAULT_ALLOC_ALIGNMENT;
|
|||
|
PRIVATE DWORD HeaderGuardSize = DEFAULT_HEADER_GUARD_SIZE;
|
|||
|
PRIVATE DWORD FooterGuardSize = DEFAULT_FOOTER_GUARD_SIZE;
|
|||
|
PRIVATE DWORD AllocMemoryFiller = DWORD_ALLOC_FILL;
|
|||
|
PRIVATE DWORD FreeMemoryFiller = DWORD_FREE_FILL;
|
|||
|
|
|||
|
//
|
|||
|
// usage variables - access using some sort of lock (critsec/interlocked)
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE CRITICAL_SECTION MemoryVarsCritSec;
|
|||
|
PRIVATE SIZE_T TotalActualMemoryAllocated = 0; // cumulative
|
|||
|
PRIVATE SIZE_T TotalBlockMemoryAllocated = 0; // "
|
|||
|
PRIVATE SIZE_T TotalRealMemoryAllocated = 0; // "
|
|||
|
PRIVATE SIZE_T TotalActualMemoryFreed = 0; // "
|
|||
|
PRIVATE SIZE_T TotalBlockMemoryFreed = 0; // "
|
|||
|
PRIVATE SIZE_T TotalRealMemoryFreed = 0; // "
|
|||
|
PRIVATE SIZE_T ActualMemoryAllocated = 0; // difference
|
|||
|
PRIVATE SIZE_T BlockLengthAllocated = 0; // "
|
|||
|
PRIVATE SIZE_T RealLengthAllocated = 0; // "
|
|||
|
PRIVATE DWORD MemoryAllocations = 0; // cumulative
|
|||
|
PRIVATE DWORD GoodMemoryAllocations = 0; // "
|
|||
|
PRIVATE DWORD MemoryReAllocations = 0; // "
|
|||
|
PRIVATE DWORD GoodMemoryReAllocations = 0; // "
|
|||
|
PRIVATE DWORD MemoryFrees = 0; // "
|
|||
|
PRIVATE DWORD GoodMemoryFrees = 0; // "
|
|||
|
PRIVATE SIZE_T LargestBlockRequested = 0;
|
|||
|
PRIVATE SIZE_T LargestBlockAllocated = 0;
|
|||
|
PRIVATE LPSTR LargestBlockRequestedFile = NULL;
|
|||
|
PRIVATE DWORD LargestBlockRequestedLine = 0;
|
|||
|
PRIVATE SIZE_T SmallestBlockRequested = (SIZE_T)-1;
|
|||
|
PRIVATE SIZE_T SmallestBlockAllocated = (SIZE_T)-1;
|
|||
|
PRIVATE LPSTR SmallestBlockRequestedFile = NULL;
|
|||
|
PRIVATE DWORD SmallestBlockRequestedLine = 0;
|
|||
|
PRIVATE DWORD DeferFreeTime = 0;
|
|||
|
|
|||
|
//
|
|||
|
// lists
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE SERIALIZED_LIST AllocatedBlockList;
|
|||
|
PRIVATE SERIALIZED_LIST DeferredFreeList;
|
|||
|
|
|||
|
//
|
|||
|
// macros
|
|||
|
//
|
|||
|
|
|||
|
#define MEMORY_ASSERT(x) \
|
|||
|
if (bAssertOnMemoryErrors) { \
|
|||
|
INET_ASSERT(x); \
|
|||
|
} else { \
|
|||
|
/* NOTHING */ \
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// private prototypes
|
|||
|
//
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DebugFillMem(
|
|||
|
IN LPVOID Pointer,
|
|||
|
IN SIZE_T Size,
|
|||
|
IN DWORD dwFiller
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetAlloc(
|
|||
|
IN UINT Flags,
|
|||
|
IN SIZE_T Size
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetFree(
|
|||
|
IN HLOCAL hLocal
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetReAlloc(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN SIZE_T Size,
|
|||
|
IN UINT Flags
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
SIZE_T
|
|||
|
InternetSize(
|
|||
|
IN HLOCAL hLocal
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetHeapAlloc(
|
|||
|
IN UINT Flags,
|
|||
|
IN SIZE_T Size
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetHeapReAlloc(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN SIZE_T Size,
|
|||
|
IN UINT Flags
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetHeapFree(
|
|||
|
IN HLOCAL hLocal
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
SIZE_T
|
|||
|
InternetHeapSize(
|
|||
|
IN HLOCAL hLocal
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
InternetDebugCheckMemBlock(
|
|||
|
IN LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
InternetDebugMemTest(
|
|||
|
VOID
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
ReportMemoryUsage(
|
|||
|
VOID
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
ReportMemoryBlocks(
|
|||
|
VOID
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DumpDeferredFreeList(
|
|||
|
VOID
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DumpMemoryList(
|
|||
|
IN LPSERIALIZED_LIST lpList
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
FindAndDumpDeferredBlock(
|
|||
|
IN HLOCAL hLocal
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
DumpBlock(
|
|||
|
IN LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
DumpDebugMemoryHeader(
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
DumpDebugMemoryFooter(
|
|||
|
LPDEBUG_MEMORY_FOOTER lpFooter
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DumpUserData(
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
LPSTR
|
|||
|
MapLastAccessOperation(
|
|||
|
MEMORY_ACTION Action
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
LPSTR
|
|||
|
MapMemoryFlags(
|
|||
|
DWORD Flags,
|
|||
|
LPSTR Buffer
|
|||
|
);
|
|||
|
|
|||
|
PRIVATE
|
|||
|
LPSTR
|
|||
|
DbgMemGetDebugSymbol(
|
|||
|
DWORD Address,
|
|||
|
LPDWORD Offset
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// functions
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
InternetDebugMemInitialize(
|
|||
|
VOID
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Initializes debug memory allocator
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
BOOL init;
|
|||
|
|
|||
|
init = (BOOL)InterlockedExchange((LPLONG)&MemoryPackageInitialized, TRUE);
|
|||
|
if (init) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("Memory package already initialized\n"
|
|||
|
));
|
|||
|
|
|||
|
DEBUG_BREAK(MEMALLOC);
|
|||
|
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
InitializeSerializedList(&AllocatedBlockList);
|
|||
|
InitializeSerializedList(&DeferredFreeList);
|
|||
|
InitializeCriticalSection(&MemoryVarsCritSec);
|
|||
|
|
|||
|
//
|
|||
|
// sleaze: disable any debug output until we finish this. Debug log
|
|||
|
// routines want to allocate memory(!). InternetReadRegistryDword()
|
|||
|
// (called from InternetGetDebugVariable()) wants to perform DEBUG_ENTER
|
|||
|
// etc.
|
|||
|
//
|
|||
|
|
|||
|
DWORD debugControlFlags = InternetDebugControlFlags;
|
|||
|
|
|||
|
InternetDebugControlFlags = DBG_NO_DEBUG;
|
|||
|
|
|||
|
//
|
|||
|
// if "WininetMem" is set then we set up to use debug memory - we use our
|
|||
|
// own heap, full debugging & reporting etc. (basically max memory debugging
|
|||
|
// as defined by me)
|
|||
|
//
|
|||
|
|
|||
|
DWORD useDefaultDebugMemoryFlags = FALSE;
|
|||
|
|
|||
|
InternetGetDebugVariable("WininetMem",
|
|||
|
&useDefaultDebugMemoryFlags
|
|||
|
);
|
|||
|
if (useDefaultDebugMemoryFlags) {
|
|||
|
bNoDebugAlloc = FALSE; // use full debug allocator (header + footers, etc.)
|
|||
|
bUseLocalAlloc = FALSE; // use our own heap
|
|||
|
bUseSymbols = FALSE; // don't load debug symbols
|
|||
|
bAssertOnMemoryErrors = TRUE; // assert to debugger/log if memory errors
|
|||
|
bFillMemoryOnAlloc = TRUE; // fill user data w/ signature if not zeroinit
|
|||
|
bFillMemoryOnFree = TRUE; // fill freed memory (useful on Win95/non-debug on NT)
|
|||
|
bReportMemoryUsage = TRUE; // dump memory usage stats
|
|||
|
bReportUnfreedBlocks = TRUE; // dump unfreed blocks (headers)
|
|||
|
bReportMemoryFooters = TRUE; // dump unfreed block footers
|
|||
|
bReportUserData = TRUE; // dump unfreed block user data
|
|||
|
bStopDumpIfBadBlock = TRUE; // stop dumping if error occurs
|
|||
|
bLimitUnfreedBlocks = TRUE; // limit block dump in case of loop in list
|
|||
|
bLimitUserData = TRUE; // limit user data dump in case of bad length
|
|||
|
bDumpAsDwords = TRUE; // dump data in dc format vs. db
|
|||
|
bHeapNoSerialize = FALSE; // heap functions are serialized
|
|||
|
bHeapGenerateExceptions = FALSE;// heap functions return errors
|
|||
|
bHeapIsGrowable = FALSE; // limit heap to maximum size (1 Meg)
|
|||
|
if (useDefaultDebugMemoryFlags == 2) {
|
|||
|
bDumpToFile = TRUE;
|
|||
|
}
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// no use-debug-mem override, see if there are any specific flags set
|
|||
|
//
|
|||
|
|
|||
|
InternetGetDebugVariable("WininetDebugMemFlags",
|
|||
|
(LPDWORD)&InternetDebugMemFlags
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// we used to load IMAGEHLP.DLL here and not use its functions until we were
|
|||
|
// dumping still in-use memory during DLL shutdown. Problem is that the
|
|||
|
// system has probably already freed IMAGEHLP.DLL by the time we come to use
|
|||
|
// it, resulting in GPF, so now we only load it at the time we're about to
|
|||
|
// use it
|
|||
|
//
|
|||
|
|
|||
|
//if (bUseSymbols) {
|
|||
|
// InitSymLib();
|
|||
|
//}
|
|||
|
|
|||
|
if (!bUseLocalAlloc) {
|
|||
|
|
|||
|
//
|
|||
|
// not using LocalAlloc(), using HeapAlloc(). Create heap
|
|||
|
//
|
|||
|
|
|||
|
InitialHeapSize = DEFAULT_INITIAL_HEAP_SIZE;
|
|||
|
InternetGetDebugVariable("WininetDebugHeapInitialSize",
|
|||
|
&InitialHeapSize
|
|||
|
);
|
|||
|
|
|||
|
MaximumHeapSize = DEFAULT_MAXIMUM_HEAP_SIZE;
|
|||
|
InternetGetDebugVariable("WininetDebugHeapMaximumSize",
|
|||
|
&MaximumHeapSize
|
|||
|
);
|
|||
|
|
|||
|
if (bHeapIsGrowable) {
|
|||
|
MaximumHeapSize = 0;
|
|||
|
}
|
|||
|
|
|||
|
hDebugHeap = HeapCreate((bHeapGenerateExceptions
|
|||
|
? HEAP_GENERATE_EXCEPTIONS
|
|||
|
: 0)
|
|||
|
| (bHeapNoSerialize
|
|||
|
? HEAP_NO_SERIALIZE
|
|||
|
: 0),
|
|||
|
InitialHeapSize,
|
|||
|
MaximumHeapSize
|
|||
|
);
|
|||
|
if (hDebugHeap == NULL) {
|
|||
|
|
|||
|
DEBUG_PUT(("HeapCreate() failed - %d\n",
|
|||
|
GetLastError()
|
|||
|
));
|
|||
|
|
|||
|
bUseLocalAlloc = TRUE;
|
|||
|
} else {
|
|||
|
HeapCompactControl = HEAP_COMPACT_NEVER;
|
|||
|
InternetGetDebugVariable("WininetDebugHeapCompactControl",
|
|||
|
(LPDWORD)&HeapCompactControl
|
|||
|
);
|
|||
|
|
|||
|
HeapValidateControl = HEAP_VALIDATE_NEVER;
|
|||
|
InternetGetDebugVariable("WininetDebugHeapValidateControl",
|
|||
|
(LPDWORD)&HeapValidateControl
|
|||
|
);
|
|||
|
|
|||
|
DEBUG_PUT(("Wininet heap = %#x\n",
|
|||
|
hDebugHeap
|
|||
|
));
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// restore default debug flags
|
|||
|
//
|
|||
|
|
|||
|
InternetDebugControlFlags = debugControlFlags;
|
|||
|
|
|||
|
//InternetDebugMemTest();
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
InternetDebugMemTerminate(
|
|||
|
IN BOOL bReport
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Frees resources allocated in InternetDebugMemInitialize, after checking that
|
|||
|
all memory is freed
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
bReport - TRUE if in-use blocks reported at termination
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
BOOL bOpened = bReport ? InternetDebugMemReport(TRUE, FALSE) : FALSE;
|
|||
|
|
|||
|
InternetDebugCheckMemFreed(FALSE);
|
|||
|
DeleteCriticalSection(&MemoryVarsCritSec);
|
|||
|
TerminateSerializedList(&AllocatedBlockList);
|
|||
|
TerminateSerializedList(&DeferredFreeList);
|
|||
|
|
|||
|
if (hDebugHeap != NULL) {
|
|||
|
|
|||
|
//
|
|||
|
// any future allocations(!) must use process heap
|
|||
|
//
|
|||
|
|
|||
|
bUseLocalAlloc = TRUE;
|
|||
|
|
|||
|
if (!HeapDestroy(hDebugHeap)) {
|
|||
|
|
|||
|
DWORD error = GetLastError();
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("HeapDestroy(%#x) returns %s (%d)\n",
|
|||
|
hDebugHeap,
|
|||
|
InternetMapError(error),
|
|||
|
error
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
if (bOpened) {
|
|||
|
InternetCloseDebugFile();
|
|||
|
}
|
|||
|
MemoryPackageInitialized = FALSE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
HLOCAL
|
|||
|
InternetDebugAllocMem(
|
|||
|
IN UINT Flags,
|
|||
|
IN UINT Size,
|
|||
|
IN LPSTR File,
|
|||
|
IN DWORD Line
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Debug memory allocator. If this succeeds, then the real block is put on our
|
|||
|
list and has its head & tail (& possibly contents) initialized. The caller
|
|||
|
gets an pointer which is an offset to the user area in the block
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Flags - controlling flags (normally passed to LocalAlloc)
|
|||
|
|
|||
|
Size - of block to allocate
|
|||
|
|
|||
|
File - from where alloc called
|
|||
|
|
|||
|
Line - in File
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - pointer to caller's start of allocated block
|
|||
|
|
|||
|
Failure - NULL
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (!MemoryPackageInitialized) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//dprintf("InternetDebugAllocMem(%#x, %d) = ", Flags, Size);
|
|||
|
InterlockedIncrement((LPLONG)&MemoryAllocations);
|
|||
|
|
|||
|
//
|
|||
|
// keep these tests separate so we don't have to look up the flags #defines
|
|||
|
//
|
|||
|
|
|||
|
INET_ASSERT(!(Flags & LMEM_MOVEABLE));
|
|||
|
INET_ASSERT(!(Flags & LMEM_DISCARDABLE));
|
|||
|
|
|||
|
if (Size == 0) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
WARNING,
|
|||
|
("InternetDebugAllocMem(%#x, %d)\n",
|
|||
|
Flags,
|
|||
|
Size
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
SIZE_T blockLength;
|
|||
|
|
|||
|
if (bNoDebugAlloc) {
|
|||
|
blockLength = Size;
|
|||
|
} else {
|
|||
|
if (Size > LargestBlockRequested) {
|
|||
|
LargestBlockRequested = Size;
|
|||
|
LargestBlockRequestedFile = File;
|
|||
|
LargestBlockRequestedLine = Line;
|
|||
|
} else if (Size < SmallestBlockRequested) {
|
|||
|
SmallestBlockRequested = Size;
|
|||
|
SmallestBlockRequestedFile = File;
|
|||
|
SmallestBlockRequestedLine = Line;
|
|||
|
}
|
|||
|
blockLength = ROUND_UP_DWORD(Size)
|
|||
|
+ sizeof(DEBUG_MEMORY_HEADER)
|
|||
|
+ sizeof(DEBUG_MEMORY_FOOTER);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// possible problem: if Size + signatures would overflow UINT. Only really
|
|||
|
// problematic on 16-bit platforms
|
|||
|
//
|
|||
|
|
|||
|
if (blockLength < Size) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("can't allocate %lu bytes: would overflow\n",
|
|||
|
(DWORD)Size
|
|||
|
));
|
|||
|
|
|||
|
DEBUG_BREAK(MEMALLOC);
|
|||
|
|
|||
|
//dprintf("NULL\n");
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// BUGBUG - allocating 0 bytes?
|
|||
|
//
|
|||
|
|
|||
|
HLOCAL hLocal = InternetAlloc(Flags, blockLength);
|
|||
|
|
|||
|
if (hLocal != NULL) {
|
|||
|
InterlockedIncrement((LPLONG)&GoodMemoryAllocations);
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("failed to allocate %u bytes memory\n",
|
|||
|
blockLength
|
|||
|
));
|
|||
|
|
|||
|
DEBUG_BREAK(MEMALLOC);
|
|||
|
|
|||
|
//dprintf("NULL\n");
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
SIZE_T actualLength = InternetSize(hLocal);
|
|||
|
SIZE_T requestedLength;
|
|||
|
|
|||
|
if (bNoDebugAlloc) {
|
|||
|
blockLength = actualLength;
|
|||
|
requestedLength = actualLength;
|
|||
|
} else {
|
|||
|
requestedLength = Size;
|
|||
|
if (actualLength > LargestBlockAllocated) {
|
|||
|
LargestBlockAllocated = actualLength;
|
|||
|
} else if (actualLength < SmallestBlockAllocated) {
|
|||
|
SmallestBlockAllocated = actualLength;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
EnterCriticalSection(&MemoryVarsCritSec);
|
|||
|
TotalActualMemoryAllocated += actualLength;
|
|||
|
TotalBlockMemoryAllocated += blockLength;
|
|||
|
TotalRealMemoryAllocated += requestedLength;
|
|||
|
ActualMemoryAllocated += actualLength;
|
|||
|
BlockLengthAllocated += blockLength;
|
|||
|
RealLengthAllocated += requestedLength;
|
|||
|
LeaveCriticalSection(&MemoryVarsCritSec);
|
|||
|
|
|||
|
if (bNoDebugAlloc || (hLocal == NULL)) {
|
|||
|
if ((hLocal != NULL) && !(Flags & LMEM_ZEROINIT) && bFillMemoryOnAlloc) {
|
|||
|
DebugFillMem(hLocal, Size, AllocMemoryFiller);
|
|||
|
}
|
|||
|
//dprintf("%#x\n", hLocal);
|
|||
|
return hLocal;
|
|||
|
}
|
|||
|
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader = (LPDEBUG_MEMORY_HEADER)hLocal;
|
|||
|
|
|||
|
//InitializeListHead(&lpHeader->List);
|
|||
|
lpHeader->ThreadId = GetCurrentThreadId();
|
|||
|
lpHeader->CreatedFile = File;
|
|||
|
lpHeader->CreatedLine = Line;
|
|||
|
lpHeader->AccessedFile = File;
|
|||
|
lpHeader->AccessedLine = Line;
|
|||
|
lpHeader->RequestedLength = Size;
|
|||
|
lpHeader->BlockLength = blockLength;
|
|||
|
lpHeader->ActualLength = actualLength;
|
|||
|
lpHeader->Signature = HEADER_SIGNATURE;
|
|||
|
lpHeader->Flags = Flags;
|
|||
|
lpHeader->TimeDeferred = 0;
|
|||
|
lpHeader->ClashTest = -1;
|
|||
|
lpHeader->LastAccessOperation = MemAllocate;
|
|||
|
|
|||
|
#if DUMP_STACK
|
|||
|
#if ONE_STACK
|
|||
|
|
|||
|
memset(lpHeader->Stack, 0, sizeof(lpHeader->Stack));
|
|||
|
GET_CALL_STACK(lpHeader->Stack);
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
GET_CALLERS_ADDRESS(&lpHeader->CreateStack[0],
|
|||
|
&lpHeader->CreateStack[1]
|
|||
|
);
|
|||
|
|
|||
|
memset(lpHeader->CreateStack, 0, sizeof(lpHeader->CreateStack));
|
|||
|
|
|||
|
GET_CALL_STACK(lpHeader->CreateStack);
|
|||
|
|
|||
|
memcpy(lpHeader->LastAccessStack,
|
|||
|
lpHeader->CreateStack,
|
|||
|
sizeof(lpHeader->LastAccessStack)
|
|||
|
);
|
|||
|
|
|||
|
#endif // ONE_STACK
|
|||
|
#endif // DUMP_STACK
|
|||
|
|
|||
|
UINT i;
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpHeader->Guard); ++i) {
|
|||
|
lpHeader->Guard[i] = GUARD_DWORD_FILL;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// BUGBUG - should be using AllocAlignment - could be > sizeof(DWORD)
|
|||
|
//
|
|||
|
|
|||
|
if (!(Flags & LMEM_ZEROINIT) && bFillMemoryOnAlloc) {
|
|||
|
DebugFillMem(lpHeader + 1, Size, AllocMemoryFiller);
|
|||
|
}
|
|||
|
|
|||
|
UINT bFillLength2 = (Size % sizeof(DWORD)) ? (sizeof(DWORD) - (Size % sizeof(DWORD))) : 0;
|
|||
|
LPBYTE lpbUserPointer = (LPBYTE)(lpHeader + 1) + Size;
|
|||
|
|
|||
|
for (i = 0; i < bFillLength2; ++i) {
|
|||
|
*lpbUserPointer++ = BYTE_ALLOC_FILL_EXTRA;
|
|||
|
}
|
|||
|
|
|||
|
LPDEBUG_MEMORY_FOOTER lpFooter = (LPDEBUG_MEMORY_FOOTER)lpbUserPointer;
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpFooter->Guard); ++i) {
|
|||
|
lpFooter->Guard[i] = GUARD_DWORD_FILL;
|
|||
|
}
|
|||
|
|
|||
|
lpFooter->BlockLength = blockLength;
|
|||
|
lpFooter->Signature = FOOTER_SIGNATURE;
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpFooter->Guard2); ++i) {
|
|||
|
lpFooter->Guard2[i] = GUARD_DWORD_FILL;
|
|||
|
}
|
|||
|
|
|||
|
if (!CheckEntryOnSerializedList(&AllocatedBlockList, &lpHeader->List, FALSE)) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugAllocMem(%d): %#x already on list?\n",
|
|||
|
Size,
|
|||
|
lpHeader
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// put at the tail of list so we can view unfreed blocks in chronological
|
|||
|
// order
|
|||
|
//
|
|||
|
|
|||
|
InsertAtTailOfSerializedList(&AllocatedBlockList, &lpHeader->List);
|
|||
|
|
|||
|
//dprintf("%#x\n", lpHeader + 1);
|
|||
|
return (HLOCAL)(lpHeader + 1);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
HLOCAL
|
|||
|
InternetDebugFreeMem(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN LPSTR File,
|
|||
|
IN DWORD Line
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Frees a block of memory allocated by InternetDebugAllocMem(). Checks that
|
|||
|
the block is on our allocated block list, and that the header and footer
|
|||
|
areas are still intact
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - handle (pointer) of block to free
|
|||
|
|
|||
|
File - from where alloc called
|
|||
|
|
|||
|
Line - in File
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - NULL
|
|||
|
|
|||
|
Failure - hLocal
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (!MemoryPackageInitialized) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//dprintf("InternetDebugFreeMem(%#x)\n", hLocal);
|
|||
|
InterlockedIncrement((LPLONG)&MemoryFrees);
|
|||
|
|
|||
|
if (hLocal == NULL) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugFreeMem(NULL)\n"
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
return InternetFree(hLocal);
|
|||
|
}
|
|||
|
|
|||
|
HLOCAL hLocalOriginal = hLocal;
|
|||
|
SIZE_T actualLength;
|
|||
|
SIZE_T blockLength;
|
|||
|
SIZE_T realLength;
|
|||
|
|
|||
|
if (bNoDebugAlloc) {
|
|||
|
actualLength = InternetSize(hLocal);
|
|||
|
blockLength = actualLength;
|
|||
|
realLength = actualLength;
|
|||
|
} else {
|
|||
|
hLocal = (HLOCAL)((LPDEBUG_MEMORY_HEADER)hLocal - 1);
|
|||
|
actualLength = InternetSize(hLocal);
|
|||
|
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader = (LPDEBUG_MEMORY_HEADER)hLocal;
|
|||
|
|
|||
|
if (CheckEntryOnSerializedList(&AllocatedBlockList, &lpHeader->List, TRUE)) {
|
|||
|
RemoveFromSerializedList(&AllocatedBlockList, &lpHeader->List);
|
|||
|
|
|||
|
if (!((lpHeader->ActualLength == actualLength)
|
|||
|
&& (lpHeader->BlockLength <= actualLength)
|
|||
|
&& !(lpHeader->BlockLength & (sizeof(DWORD) - 1))
|
|||
|
&& (lpHeader->RequestedLength < lpHeader->BlockLength))) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugFreeMem(%#x): block lengths mismatch\n",
|
|||
|
hLocalOriginal
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
}
|
|||
|
if (InternetDebugCheckMemBlock(lpHeader)) {
|
|||
|
blockLength = lpHeader->BlockLength;
|
|||
|
realLength = lpHeader->RequestedLength;
|
|||
|
} else {
|
|||
|
blockLength = 0;
|
|||
|
realLength = 0;
|
|||
|
}
|
|||
|
if (bDeferFree) {
|
|||
|
|
|||
|
#if DUMP_STACK
|
|||
|
#if ONE_STACK
|
|||
|
|
|||
|
memset(lpHeader->Stack, 0, sizeof(lpHeader->Stack));
|
|||
|
GET_CALL_STACK(lpHeader->Stack);
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
GET_CALLERS_ADDRESS(&lpHeader->CreateStack[0],
|
|||
|
&lpHeader->CreateStack[1]
|
|||
|
);
|
|||
|
|
|||
|
memset(lpHeader->CreateStack, 0, sizeof(lpHeader->CreateStack));
|
|||
|
|
|||
|
GET_CALL_STACK(lpHeader->CreateStack);
|
|||
|
|
|||
|
memcpy(lpHeader->LastAccessStack,
|
|||
|
lpHeader->CreateStack,
|
|||
|
sizeof(lpHeader->LastAccessStack)
|
|||
|
);
|
|||
|
|
|||
|
#endif // ONE_STACK
|
|||
|
#endif // DUMP_STACK
|
|||
|
|
|||
|
InsertAtTailOfSerializedList(&DeferredFreeList, &lpHeader->List);
|
|||
|
hLocal = NULL;
|
|||
|
}
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugFreeMem(%#x): can't find %#x\n",
|
|||
|
hLocalOriginal,
|
|||
|
&lpHeader->List
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
FindAndDumpDeferredBlock(hLocal);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (hLocal && bFillMemoryOnFree) {
|
|||
|
DebugFillMem(hLocal, actualLength, FreeMemoryFiller);
|
|||
|
}
|
|||
|
|
|||
|
hLocal = InternetFree(hLocal);
|
|||
|
|
|||
|
if (hLocal == NULL) {
|
|||
|
InterlockedIncrement((LPLONG)&GoodMemoryFrees);
|
|||
|
EnterCriticalSection(&MemoryVarsCritSec);
|
|||
|
TotalActualMemoryFreed += actualLength;
|
|||
|
TotalBlockMemoryFreed += blockLength;
|
|||
|
TotalRealMemoryFreed += realLength;
|
|||
|
ActualMemoryAllocated -= actualLength;
|
|||
|
BlockLengthAllocated -= blockLength;
|
|||
|
RealLengthAllocated -= realLength;
|
|||
|
LeaveCriticalSection(&MemoryVarsCritSec);
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugFreeMem(%#x) failed\n",
|
|||
|
hLocalOriginal
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
hLocal = hLocalOriginal;
|
|||
|
}
|
|||
|
|
|||
|
return hLocal;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
HLOCAL
|
|||
|
InternetDebugReAllocMem(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN UINT Size,
|
|||
|
IN UINT Flags,
|
|||
|
IN LPSTR File,
|
|||
|
IN DWORD Line
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Reallocates a debug memory block allocated by InternetDebugAllocMem()
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - the handle (pointer) of the allocated block
|
|||
|
|
|||
|
Size - requested size of new block; can be larger or smaller than current
|
|||
|
size
|
|||
|
|
|||
|
Flags - controlling flags (normally passed to LocalReAlloc)
|
|||
|
|
|||
|
File - from where alloc called
|
|||
|
|
|||
|
Line - in File
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - pointer to new block. May be same or different than previous
|
|||
|
pointer, depending on flags
|
|||
|
|
|||
|
Failure - NULL
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (!MemoryPackageInitialized) {
|
|||
|
return NULL;
|
|||
|
}
|
|||
|
|
|||
|
//dprintf("InternetDebugReAllocMem(%#x, %d, %#x)\n", hLocal, Size, Flags);
|
|||
|
InterlockedIncrement((LPLONG)&MemoryReAllocations);
|
|||
|
|
|||
|
//
|
|||
|
// we can't handle the following flags
|
|||
|
//
|
|||
|
|
|||
|
INET_ASSERT(!(Flags & LMEM_MODIFY));
|
|||
|
|
|||
|
//
|
|||
|
// can't handle reallocating down to zero
|
|||
|
//
|
|||
|
|
|||
|
if (Size == 0) {
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
HLOCAL hLocalOriginal = hLocal;
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader;
|
|||
|
SIZE_T actualLength;
|
|||
|
SIZE_T blockLength;
|
|||
|
SIZE_T requestedLength;
|
|||
|
SIZE_T oldRequestedLength;
|
|||
|
|
|||
|
if (bNoDebugAlloc) {
|
|||
|
actualLength = InternetSize(hLocal);
|
|||
|
blockLength = actualLength;
|
|||
|
requestedLength = actualLength;
|
|||
|
} else {
|
|||
|
if (Size > LargestBlockRequested) {
|
|||
|
LargestBlockRequested = Size;
|
|||
|
LargestBlockRequestedFile = File;
|
|||
|
LargestBlockRequestedLine = Line;
|
|||
|
} else if (Size < SmallestBlockRequested) {
|
|||
|
SmallestBlockRequested = Size;
|
|||
|
SmallestBlockRequestedFile = File;
|
|||
|
SmallestBlockRequestedLine = Line;
|
|||
|
}
|
|||
|
lpHeader = (LPDEBUG_MEMORY_HEADER)hLocal - 1;
|
|||
|
hLocal = (HLOCAL)lpHeader;
|
|||
|
if (!CheckEntryOnSerializedList(&AllocatedBlockList, &lpHeader->List, TRUE)) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugReAllocMem(%#x): can't find %#x\n",
|
|||
|
hLocalOriginal
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
return hLocalOriginal;
|
|||
|
}
|
|||
|
RemoveFromSerializedList(&AllocatedBlockList, &lpHeader->List);
|
|||
|
InternetDebugCheckMemBlock(lpHeader);
|
|||
|
actualLength = InternetSize((HLOCAL)lpHeader);
|
|||
|
blockLength = lpHeader->BlockLength;
|
|||
|
requestedLength = lpHeader->RequestedLength;
|
|||
|
oldRequestedLength = requestedLength;
|
|||
|
if (!((lpHeader->ActualLength == actualLength)
|
|||
|
&& (lpHeader->BlockLength <= actualLength)
|
|||
|
&& !(lpHeader->BlockLength & (sizeof(DWORD) - 1))
|
|||
|
&& (lpHeader->RequestedLength < lpHeader->BlockLength))) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugReAllocMem(%#x): block lengths mismatch\n",
|
|||
|
hLocalOriginal
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
}
|
|||
|
}
|
|||
|
EnterCriticalSection(&MemoryVarsCritSec);
|
|||
|
ActualMemoryAllocated -= actualLength;
|
|||
|
BlockLengthAllocated -= blockLength;
|
|||
|
RealLengthAllocated -= requestedLength;
|
|||
|
LeaveCriticalSection(&MemoryVarsCritSec);
|
|||
|
requestedLength = Size;
|
|||
|
if (bNoDebugAlloc) {
|
|||
|
blockLength = Size;
|
|||
|
} else {
|
|||
|
blockLength = ROUND_UP_DWORD(Size)
|
|||
|
+ sizeof(DEBUG_MEMORY_HEADER)
|
|||
|
+ sizeof(DEBUG_MEMORY_FOOTER);
|
|||
|
}
|
|||
|
hLocal = InternetReAlloc(hLocal, blockLength, Flags);
|
|||
|
if (hLocal != NULL) {
|
|||
|
InterlockedIncrement((LPLONG)&GoodMemoryReAllocations);
|
|||
|
actualLength = InternetSize(hLocal);
|
|||
|
if (bNoDebugAlloc) {
|
|||
|
blockLength = actualLength;
|
|||
|
} else {
|
|||
|
lpHeader = (LPDEBUG_MEMORY_HEADER)hLocal;
|
|||
|
//InitializeListHead(&lpHeader->List);
|
|||
|
lpHeader->ThreadId = GetCurrentThreadId();
|
|||
|
lpHeader->AccessedFile = File;
|
|||
|
lpHeader->AccessedLine = Line;
|
|||
|
lpHeader->RequestedLength = requestedLength;
|
|||
|
lpHeader->BlockLength = blockLength;
|
|||
|
lpHeader->ActualLength = actualLength;
|
|||
|
lpHeader->Flags = Flags;
|
|||
|
lpHeader->TimeDeferred = 0;
|
|||
|
lpHeader->ClashTest = -1;
|
|||
|
lpHeader->LastAccessOperation = MemReallocate;
|
|||
|
|
|||
|
#if DUMP_STACK
|
|||
|
#if ONE_STACK
|
|||
|
#else
|
|||
|
|
|||
|
//GET_CALLERS_ADDRESS(&lpHeader->LastAccessStack[0],
|
|||
|
// &lpHeader->LastAccessStack[1]
|
|||
|
// );
|
|||
|
|
|||
|
memset(lpHeader->LastAccessStack, 0, sizeof(lpHeader->LastAccessStack));
|
|||
|
|
|||
|
GET_CALL_STACK(lpHeader->LastAccessStack);
|
|||
|
|
|||
|
#endif // ONE_STACK
|
|||
|
#endif // DUMP_STACK
|
|||
|
|
|||
|
LPBYTE extraPointer;
|
|||
|
UINT dwFillLength;
|
|||
|
UINT i;
|
|||
|
|
|||
|
if ((requestedLength > oldRequestedLength)
|
|||
|
&& bFillMemoryOnAlloc && !(Flags & LMEM_ZEROINIT)) {
|
|||
|
|
|||
|
extraPointer = (LPBYTE)(lpHeader + 1) + oldRequestedLength;
|
|||
|
|
|||
|
SIZE_T difference = requestedLength - oldRequestedLength;
|
|||
|
DWORD dwFiller = AllocMemoryFiller;
|
|||
|
SIZE_T syncLength = oldRequestedLength & (sizeof(DWORD) - 1);
|
|||
|
|
|||
|
if (syncLength) {
|
|||
|
syncLength = sizeof(DWORD) - syncLength;
|
|||
|
syncLength = min(syncLength, difference);
|
|||
|
difference -= syncLength;
|
|||
|
for (i = 0; i < syncLength; ++i) {
|
|||
|
*extraPointer++ = ((LPBYTE)&dwFiller)[i];
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//dwFillLength = difference / sizeof(DWORD);
|
|||
|
//difference %= sizeof(DWORD);
|
|||
|
//while (dwFillLength--) {
|
|||
|
// *(LPDWORD)extraPointer = 0;
|
|||
|
// extraPointer += sizeof(DWORD);
|
|||
|
//}
|
|||
|
//while (difference--) {
|
|||
|
// *extraPointer++ = 0;
|
|||
|
//}
|
|||
|
|
|||
|
if (difference) {
|
|||
|
DebugFillMem(extraPointer, difference, dwFiller);
|
|||
|
extraPointer += difference;
|
|||
|
}
|
|||
|
} else {
|
|||
|
extraPointer = (LPBYTE)(lpHeader + 1) + requestedLength;
|
|||
|
}
|
|||
|
|
|||
|
SIZE_T bFillLength = (sizeof(DWORD) - (requestedLength % sizeof(DWORD))) & (sizeof(DWORD) - 1);
|
|||
|
|
|||
|
while (bFillLength--) {
|
|||
|
*extraPointer++ = BYTE_ALLOC_FILL_EXTRA;
|
|||
|
}
|
|||
|
|
|||
|
LPDEBUG_MEMORY_FOOTER lpFooter = (LPDEBUG_MEMORY_FOOTER)extraPointer;
|
|||
|
|
|||
|
INET_ASSERT(lpFooter == (LPDEBUG_MEMORY_FOOTER)
|
|||
|
((LPBYTE)(lpHeader + 1) + ROUND_UP_DWORD(requestedLength)));
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpFooter->Guard); ++i) {
|
|||
|
lpFooter->Guard[i] = GUARD_DWORD_FILL;
|
|||
|
}
|
|||
|
lpFooter->Signature = FOOTER_SIGNATURE;
|
|||
|
lpFooter->BlockLength = blockLength;
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpFooter->Guard2); ++i) {
|
|||
|
lpFooter->Guard2[i] = GUARD_DWORD_FILL;
|
|||
|
}
|
|||
|
InsertAtTailOfSerializedList(&AllocatedBlockList, &lpHeader->List);
|
|||
|
hLocal = (HLOCAL)(lpHeader + 1);
|
|||
|
}
|
|||
|
EnterCriticalSection(&MemoryVarsCritSec);
|
|||
|
ActualMemoryAllocated += actualLength;
|
|||
|
BlockLengthAllocated += blockLength;
|
|||
|
RealLengthAllocated += requestedLength;
|
|||
|
LeaveCriticalSection(&MemoryVarsCritSec);
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("failed to reallocate %u bytes memory. Last error = %d\n",
|
|||
|
Size,
|
|||
|
GetLastError()
|
|||
|
));
|
|||
|
|
|||
|
DEBUG_BREAK(MEMALLOC);
|
|||
|
|
|||
|
}
|
|||
|
return hLocal;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
SIZE_T
|
|||
|
InternetDebugSizeMem(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN LPSTR File,
|
|||
|
IN DWORD Line
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Returns actual allocated block size
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - pointer to allocated block
|
|||
|
|
|||
|
File - from where alloc called
|
|||
|
|
|||
|
Line - in File
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
SIZE_T
|
|||
|
size of allocated block
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (!MemoryPackageInitialized) {
|
|||
|
return 0;
|
|||
|
}
|
|||
|
|
|||
|
//dprintf("InternetDebugSizeMem(%#x)\n", hLocal);
|
|||
|
SIZE_T size = InternetSize(hLocal);
|
|||
|
|
|||
|
if (!bNoDebugAlloc) {
|
|||
|
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader = (LPDEBUG_MEMORY_HEADER)hLocal - 1;
|
|||
|
|
|||
|
INET_ASSERT(lpHeader->Signature == HEADER_SIGNATURE);
|
|||
|
|
|||
|
SIZE_T sizeInHeader = lpHeader->BlockLength
|
|||
|
- (sizeof(DEBUG_MEMORY_HEADER) + sizeof(DEBUG_MEMORY_FOOTER));
|
|||
|
|
|||
|
INET_ASSERT((sizeInHeader <= size)
|
|||
|
&& (size >= sizeof(DEBUG_MEMORY_HEADER) + sizeof(DEBUG_MEMORY_FOOTER))
|
|||
|
&& (lpHeader->ActualLength == size)
|
|||
|
);
|
|||
|
|
|||
|
size = sizeInHeader;
|
|||
|
}
|
|||
|
|
|||
|
return size;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
BOOL
|
|||
|
InternetDebugCheckMemFreed(
|
|||
|
IN BOOL bReport
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Called when we're about to quit. Checks that all allocated memory has been
|
|||
|
cleaned up
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
bReport - TRUE if in-use blocks reported
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
BOOL
|
|||
|
TRUE - all allocated memory freed
|
|||
|
|
|||
|
FALSE - we failed to clean up
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (bReport) {
|
|||
|
if (bReportMemoryUsage) {
|
|||
|
ReportMemoryUsage();
|
|||
|
}
|
|||
|
if (bReportUnfreedBlocks) {
|
|||
|
ReportMemoryBlocks();
|
|||
|
}
|
|||
|
}
|
|||
|
if (ElementsOnSerializedList(&AllocatedBlockList) != 0) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetDebugCheckMemFreed(): %d memory blocks still allocated\n",
|
|||
|
MemoryAllocations - MemoryFrees
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
BOOL
|
|||
|
InternetDebugMemReport(
|
|||
|
IN BOOL bTerminateSymbols,
|
|||
|
IN BOOL bCloseFile
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Dumps in-use blocks to debugger and/or file
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
bTerminateSymbols - TRUE if we are to terminate symbols here
|
|||
|
|
|||
|
bCloseFile - TRUE if we are to close debug log file here
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
BOOL - TRUE if we opened debug log file
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
BOOL bOpened = FALSE;
|
|||
|
|
|||
|
if (bDumpToFile) {
|
|||
|
bOpened = InternetOpenDebugFile();
|
|||
|
if (bOpened) {
|
|||
|
InternetDebugResetControlFlags(DBG_NO_DEBUG);
|
|||
|
InternetDebugSetControlFlags(DBG_TO_FILE | DBG_NO_ASSERT_BREAK);
|
|||
|
}
|
|||
|
}
|
|||
|
ReportMemoryUsage();
|
|||
|
ReportMemoryBlocks();
|
|||
|
if (bUseSymbols && bTerminateSymbols) {
|
|||
|
TermSymLib();
|
|||
|
}
|
|||
|
if (bOpened && bCloseFile) {
|
|||
|
InternetCloseDebugFile();
|
|||
|
}
|
|||
|
return bOpened;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// private functions
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DebugFillMem(
|
|||
|
IN LPVOID Pointer,
|
|||
|
IN SIZE_T Size,
|
|||
|
IN DWORD dwFiller
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Fills memory with repeating debug pattern. Performs DWORD fill then finishes
|
|||
|
off any remaining bytes with character fill (rep movsd/rep movsb (ideally)
|
|||
|
(x86!))
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Pointer - memory to fill
|
|||
|
|
|||
|
Size - of Pointer in bytes
|
|||
|
|
|||
|
dwFiller - DWORD value to use
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
INET_ASSERT(((DWORD_PTR)Pointer & (sizeof(DWORD) - 1)) == 0);
|
|||
|
|
|||
|
SIZE_T dwFillLength = Size / sizeof(DWORD);
|
|||
|
SIZE_T bFillLength = Size % sizeof(DWORD);
|
|||
|
|
|||
|
//
|
|||
|
// assume > 0 DWORDs to fill
|
|||
|
//
|
|||
|
|
|||
|
LPDWORD lpdwPointer = (LPDWORD)Pointer;
|
|||
|
SIZE_T i;
|
|||
|
|
|||
|
for (i = 0; i < dwFillLength; ++i) {
|
|||
|
*lpdwPointer++ = dwFiller;
|
|||
|
}
|
|||
|
|
|||
|
if (bFillLength) {
|
|||
|
|
|||
|
LPBYTE lpbPointer = (LPBYTE)lpdwPointer;
|
|||
|
|
|||
|
for (i = 0; i < bFillLength; ++i) {
|
|||
|
*lpbPointer++ = ((LPBYTE)&dwFiller)[i];
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetAlloc(
|
|||
|
IN UINT Flags,
|
|||
|
IN SIZE_T Size
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Allocator - uses process (local) heap or component (debug) heap based on
|
|||
|
global flag setting
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Flags - LocalAlloc flags
|
|||
|
|
|||
|
Size - of block to allocate
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - pointer to allocated block
|
|||
|
|
|||
|
Failure - NULL
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (bUseLocalAlloc) {
|
|||
|
return LocalAlloc(Flags, Size);
|
|||
|
} else {
|
|||
|
return InternetHeapAlloc(Flags, Size);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetFree(
|
|||
|
IN HLOCAL hLocal
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Deallocator - uses process (local) heap or component (debug) heap based on
|
|||
|
global flag setting
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - pointer to block to deallocate
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - NULL
|
|||
|
|
|||
|
Failure - pointer to still allocated block
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (bUseLocalAlloc) {
|
|||
|
return LocalFree(hLocal);
|
|||
|
} else {
|
|||
|
return InternetHeapFree(hLocal);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetReAlloc(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN SIZE_T Size,
|
|||
|
IN UINT Flags
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Reallocator - uses process (local) heap or component (debug) heap based on
|
|||
|
global flag setting
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - pointer to block to reallocate
|
|||
|
|
|||
|
Flags - LocalAlloc flags
|
|||
|
|
|||
|
Size - of block to allocate
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - pointer to allocated block
|
|||
|
|
|||
|
Failure - NULL
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (bUseLocalAlloc) {
|
|||
|
return LocalReAlloc(hLocal, Size, Flags);
|
|||
|
} else {
|
|||
|
return InternetHeapReAlloc(hLocal, Size, Flags);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
SIZE_T
|
|||
|
InternetSize(
|
|||
|
IN HLOCAL hLocal
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Block sizer - uses process (local) heap or component (debug) heap based on
|
|||
|
global flag setting
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - pointer to block to size
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
SIZE_T
|
|||
|
Success - size of block
|
|||
|
|
|||
|
Failure - 0
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (bUseLocalAlloc) {
|
|||
|
return LocalSize(hLocal);
|
|||
|
} else {
|
|||
|
return InternetHeapSize(hLocal);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetHeapAlloc(
|
|||
|
IN UINT Flags,
|
|||
|
IN SIZE_T Size
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Allocate memory from debug heap
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Flags - passed to LocalAlloc
|
|||
|
|
|||
|
Size - of block to allocate
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
HLOCAL hLocal;
|
|||
|
|
|||
|
if (hDebugHeap != NULL) {
|
|||
|
hLocal = (HLOCAL)HeapAlloc(hDebugHeap,
|
|||
|
(bHeapNoSerialize
|
|||
|
? HEAP_NO_SERIALIZE
|
|||
|
: 0)
|
|||
|
| (bHeapGenerateExceptions
|
|||
|
? HEAP_GENERATE_EXCEPTIONS
|
|||
|
: 0)
|
|||
|
| ((Flags & LMEM_ZEROINIT)
|
|||
|
? HEAP_ZERO_MEMORY
|
|||
|
: 0),
|
|||
|
Size
|
|||
|
);
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetHeapAlloc(): hDebugHeap is NULL\n"
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
hLocal = LocalAlloc(Flags, Size);
|
|||
|
}
|
|||
|
if (hLocal == NULL) {
|
|||
|
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
|||
|
}
|
|||
|
return hLocal;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetHeapReAlloc(
|
|||
|
IN HLOCAL hLocal,
|
|||
|
IN SIZE_T Size,
|
|||
|
IN UINT Flags
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Reallocate memory from debug heap
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - pointer to block to reallocate
|
|||
|
|
|||
|
Size - new size
|
|||
|
|
|||
|
Flags - to LocalReAlloc
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - pointer to new block
|
|||
|
|
|||
|
Failure - NULL
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (hDebugHeap != NULL) {
|
|||
|
hLocal = (HLOCAL)HeapReAlloc(hDebugHeap,
|
|||
|
(bHeapNoSerialize
|
|||
|
? HEAP_NO_SERIALIZE
|
|||
|
: 0)
|
|||
|
| (bHeapGenerateExceptions
|
|||
|
? HEAP_GENERATE_EXCEPTIONS
|
|||
|
: 0)
|
|||
|
| ((Flags & LMEM_MOVEABLE)
|
|||
|
? 0
|
|||
|
: HEAP_REALLOC_IN_PLACE_ONLY)
|
|||
|
| ((Flags & LMEM_ZEROINIT)
|
|||
|
? HEAP_ZERO_MEMORY
|
|||
|
: 0),
|
|||
|
(LPVOID)hLocal,
|
|||
|
Size
|
|||
|
);
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetHeapReAlloc(): hDebugHeap is NULL\n"
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
//
|
|||
|
// block still allocated
|
|||
|
//
|
|||
|
|
|||
|
hLocal = NULL;
|
|||
|
}
|
|||
|
return hLocal;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
HLOCAL
|
|||
|
InternetHeapFree(
|
|||
|
IN HLOCAL hLocal
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Free memory to debug heap
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - to free
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
HLOCAL
|
|||
|
Success - NULL
|
|||
|
|
|||
|
Failure - hLocal
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
BOOL ok;
|
|||
|
|
|||
|
if (hDebugHeap != NULL) {
|
|||
|
ok = HeapFree(hDebugHeap,
|
|||
|
bHeapNoSerialize ? HEAP_NO_SERIALIZE : 0,
|
|||
|
(LPVOID)hLocal
|
|||
|
);
|
|||
|
if (!ok) {
|
|||
|
|
|||
|
DWORD error = GetLastError();
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("HeapFree() returns %s (%d)\n",
|
|||
|
InternetMapError(error),
|
|||
|
error
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
}
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetHeapFree(): hDebugHeap is NULL\n"
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
ok = FALSE;
|
|||
|
}
|
|||
|
return ok ? NULL : hLocal;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
SIZE_T
|
|||
|
InternetHeapSize(
|
|||
|
IN HLOCAL hLocal
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Determines size of block allocated from debug heap
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
hLocal - handle (pointer) of block for which to get size
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
SIZE_T
|
|||
|
Success - size of block
|
|||
|
|
|||
|
Failure - 0
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
SIZE_T size;
|
|||
|
|
|||
|
if (hDebugHeap != NULL) {
|
|||
|
size = HeapSize(hDebugHeap,
|
|||
|
bHeapNoSerialize ? HEAP_NO_SERIALIZE : 0,
|
|||
|
(LPCVOID)hLocal
|
|||
|
);
|
|||
|
} else {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("InternetHeapSize(): hDebugHeap is NULL\n"
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
size = (SIZE_T)-1;
|
|||
|
}
|
|||
|
if (size == (SIZE_T)-1) {
|
|||
|
SetLastError(ERROR_INVALID_HANDLE);
|
|||
|
return 0;
|
|||
|
} else {
|
|||
|
return size;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
InternetDebugCheckMemBlock(
|
|||
|
IN LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Checks the consistency of a debug memory block
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
lpHeader - pointer to what we think is DEBUG_MEMORY_HEADER
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
BOOL result;
|
|||
|
|
|||
|
__try {
|
|||
|
LPDEBUG_MEMORY_FOOTER lpFooter = (LPDEBUG_MEMORY_FOOTER)
|
|||
|
((LPBYTE)lpHeader
|
|||
|
+ (lpHeader->BlockLength - sizeof(DEBUG_MEMORY_FOOTER)));
|
|||
|
|
|||
|
BOOL headerGuardOverrun = FALSE;
|
|||
|
UINT i;
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpHeader->Guard); ++i) {
|
|||
|
if (lpHeader->Guard[i] != GUARD_DWORD_FILL) {
|
|||
|
headerGuardOverrun = TRUE;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
BOOL footerGuardOverrun = FALSE;
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpFooter->Guard); ++i) {
|
|||
|
if (lpFooter->Guard[i] != GUARD_DWORD_FILL) {
|
|||
|
footerGuardOverrun = TRUE;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
BOOL footerGuard2Overrun = FALSE;
|
|||
|
|
|||
|
for (i = 0; i < ARRAY_ELEMENTS(lpFooter->Guard2); ++i) {
|
|||
|
if (lpFooter->Guard2[i] != GUARD_DWORD_FILL) {
|
|||
|
footerGuard2Overrun = TRUE;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
LPBYTE lpExtraMemory = (LPBYTE)(lpHeader + 1) + lpHeader->RequestedLength;
|
|||
|
BOOL extraMemoryOverrun = FALSE;
|
|||
|
SIZE_T byteLength = ROUND_UP_DWORD(lpHeader->RequestedLength) - lpHeader->RequestedLength;
|
|||
|
|
|||
|
for (i = 0; i < byteLength; ++i) {
|
|||
|
if (lpExtraMemory[i] != BYTE_ALLOC_FILL_EXTRA) {
|
|||
|
extraMemoryOverrun = TRUE;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (headerGuardOverrun
|
|||
|
|| footerGuardOverrun
|
|||
|
|| footerGuard2Overrun
|
|||
|
|| extraMemoryOverrun
|
|||
|
|| (lpHeader->Signature != HEADER_SIGNATURE)
|
|||
|
|| (lpFooter->Signature != FOOTER_SIGNATURE)
|
|||
|
|| (lpFooter->BlockLength != lpHeader->BlockLength)) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
ERROR,
|
|||
|
("Bad block: %#x\n",
|
|||
|
lpHeader
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
result = FALSE;
|
|||
|
} else {
|
|||
|
result = TRUE;
|
|||
|
}
|
|||
|
} __except(EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
|
|||
|
DEBUG_PRINT(MEMALLOC,
|
|||
|
FATAL,
|
|||
|
("Invalid block %#x - exception occurred\n",
|
|||
|
lpHeader
|
|||
|
));
|
|||
|
|
|||
|
MEMORY_ASSERT(FALSE);
|
|||
|
|
|||
|
result = FALSE;
|
|||
|
}
|
|||
|
return result;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
InternetDebugMemTest(
|
|||
|
VOID
|
|||
|
)
|
|||
|
{
|
|||
|
//
|
|||
|
// test
|
|||
|
//
|
|||
|
|
|||
|
LPVOID p;
|
|||
|
|
|||
|
p = (LPVOID)ALLOCATE_MEMORY(LMEM_FIXED, 1);
|
|||
|
*((LPBYTE)p + 1) = 'X';
|
|||
|
p = (LPVOID)FREE_MEMORY((HLOCAL)p);
|
|||
|
|
|||
|
INET_ASSERT(p == NULL);
|
|||
|
|
|||
|
p = (LPVOID)ALLOCATE_MEMORY(LMEM_FIXED, 1);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 1111, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 439, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 720, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 256, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 16, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 128, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 32, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 4, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 64, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 63, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 64, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 65, LMEM_MOVEABLE | LMEM_ZEROINIT);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 65, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 64, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 64, LMEM_MOVEABLE);
|
|||
|
p = (LPVOID)FREE_MEMORY((HLOCAL)p);
|
|||
|
|
|||
|
INET_ASSERT(p == NULL);
|
|||
|
|
|||
|
p = (LPVOID)ALLOCATE_MEMORY(LMEM_FIXED, 8);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 8, LMEM_FIXED);
|
|||
|
p = (LPVOID)REALLOCATE_MEMORY(p, 100000, LMEM_FIXED);
|
|||
|
p = (LPVOID)FREE_MEMORY((HLOCAL)p);
|
|||
|
|
|||
|
INET_ASSERT(p == NULL);
|
|||
|
|
|||
|
InternetDebugCheckMemFreed(TRUE);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
ReportMemoryUsage(
|
|||
|
VOID
|
|||
|
)
|
|||
|
{
|
|||
|
//
|
|||
|
// make copies of variables in case debug print functions want to allocate
|
|||
|
// debug memory (!)
|
|||
|
//
|
|||
|
|
|||
|
SIZE_T totalActualMemoryAllocated;
|
|||
|
SIZE_T totalBlockMemoryAllocated;
|
|||
|
SIZE_T totalRealMemoryAllocated;
|
|||
|
SIZE_T totalActualMemoryFreed;
|
|||
|
SIZE_T totalBlockMemoryFreed;
|
|||
|
SIZE_T totalRealMemoryFreed;
|
|||
|
SIZE_T actualMemoryAllocated;
|
|||
|
SIZE_T blockLengthAllocated;
|
|||
|
SIZE_T realLengthAllocated;
|
|||
|
DWORD memoryAllocations;
|
|||
|
DWORD goodMemoryAllocations;
|
|||
|
DWORD memoryReAllocations;
|
|||
|
DWORD goodMemoryReAllocations;
|
|||
|
DWORD memoryFrees;
|
|||
|
DWORD goodMemoryFrees;
|
|||
|
SIZE_T largestBlockRequested;
|
|||
|
SIZE_T largestBlockAllocated;
|
|||
|
SIZE_T smallestBlockRequested;
|
|||
|
SIZE_T smallestBlockAllocated;
|
|||
|
|
|||
|
EnterCriticalSection(&MemoryVarsCritSec);
|
|||
|
|
|||
|
totalActualMemoryAllocated = TotalActualMemoryAllocated;
|
|||
|
totalBlockMemoryAllocated = TotalBlockMemoryAllocated;
|
|||
|
totalRealMemoryAllocated = TotalRealMemoryAllocated;
|
|||
|
totalActualMemoryFreed = TotalActualMemoryFreed;
|
|||
|
totalBlockMemoryFreed = TotalBlockMemoryFreed;
|
|||
|
totalRealMemoryFreed = TotalRealMemoryFreed;
|
|||
|
actualMemoryAllocated = ActualMemoryAllocated;
|
|||
|
blockLengthAllocated = BlockLengthAllocated;
|
|||
|
realLengthAllocated = RealLengthAllocated;
|
|||
|
memoryAllocations = MemoryAllocations;
|
|||
|
goodMemoryAllocations = GoodMemoryAllocations;
|
|||
|
memoryReAllocations = MemoryReAllocations;
|
|||
|
goodMemoryReAllocations = GoodMemoryReAllocations;
|
|||
|
memoryFrees = MemoryFrees;
|
|||
|
goodMemoryFrees = GoodMemoryFrees;
|
|||
|
largestBlockRequested = LargestBlockRequested;
|
|||
|
largestBlockAllocated = LargestBlockAllocated;
|
|||
|
smallestBlockRequested = SmallestBlockRequested;
|
|||
|
smallestBlockAllocated = SmallestBlockAllocated;
|
|||
|
|
|||
|
LeaveCriticalSection(&MemoryVarsCritSec);
|
|||
|
|
|||
|
#ifdef _WIN64
|
|||
|
char numBuf[64];
|
|||
|
#else
|
|||
|
char numBuf[32];
|
|||
|
#endif
|
|||
|
|
|||
|
DEBUG_PUT(("********************************************************************************\n"
|
|||
|
"\n"
|
|||
|
"WinInet Debug Memory Usage:\n"
|
|||
|
"\n"
|
|||
|
"\tInternetDebugMemFlags = %#08x\n"
|
|||
|
"\n",
|
|||
|
InternetDebugMemFlags
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tTotal Memory Allocated. . . , . . . . %s\n", NiceNum(numBuf, totalActualMemoryAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tTotal Block Length Allocated. . . . . %s\n", NiceNum(numBuf, totalBlockMemoryAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tTotal User Length Allocated . . . . . %s\n", NiceNum(numBuf, totalRealMemoryAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tTotal Memory Freed. . . . . . . . . . %s\n", NiceNum(numBuf, totalActualMemoryFreed, 0)));
|
|||
|
DEBUG_PUT(("\tTotal Block Length Freed. . . . . . . %s\n", NiceNum(numBuf, totalBlockMemoryFreed, 0)));
|
|||
|
DEBUG_PUT(("\tTotal User Length Freed . . . . . . . %s\n", NiceNum(numBuf, totalRealMemoryFreed, 0)));
|
|||
|
DEBUG_PUT(("\tMemory Still Allocated. . . . . . . . %s\n", NiceNum(numBuf, actualMemoryAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tBlock Length Still Allocated. . . . . %s\n", NiceNum(numBuf, blockLengthAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tUser Length Still Allocated . . . . . %s\n", NiceNum(numBuf, realLengthAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tAttempted Memory Allocations. . . . . %s\n", NiceNum(numBuf, memoryAllocations, 0)));
|
|||
|
DEBUG_PUT(("\tGood Memory Allocations . . . . . . . %s\n", NiceNum(numBuf, goodMemoryAllocations, 0)));
|
|||
|
DEBUG_PUT(("\tAttempted Memory Reallocations. . . . %s\n", NiceNum(numBuf, memoryReAllocations, 0)));
|
|||
|
DEBUG_PUT(("\tGood Memory Reallocations . . . . . . %s\n", NiceNum(numBuf, goodMemoryReAllocations, 0)));
|
|||
|
DEBUG_PUT(("\tAttempted Memory Frees. . . . . . . . %s\n", NiceNum(numBuf, memoryFrees, 0)));
|
|||
|
DEBUG_PUT(("\tGood Memory Frees . . . . . . . . . . %s\n", NiceNum(numBuf, goodMemoryFrees, 0)));
|
|||
|
DEBUG_PUT(("\tLargest Block Requested . . . . . . . %s\n", NiceNum(numBuf, largestBlockRequested, 0)));
|
|||
|
DEBUG_PUT(("\tLargest Block Allocated . . . . . . . %s\n", NiceNum(numBuf, largestBlockAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tLargest Block Requested From. . . . . %s!%d\n", SourceFilename(LargestBlockRequestedFile), LargestBlockRequestedLine));
|
|||
|
DEBUG_PUT(("\tSmallest Block Requested. . . . . . . %s\n", NiceNum(numBuf, smallestBlockRequested, 0)));
|
|||
|
DEBUG_PUT(("\tSmallest Block Allocated. . . . . . . %s\n", NiceNum(numBuf, smallestBlockAllocated, 0)));
|
|||
|
DEBUG_PUT(("\tSmallest Block Requested From . . . . %s!%d\n", SourceFilename(SmallestBlockRequestedFile), SmallestBlockRequestedLine));
|
|||
|
DEBUG_PUT(("\n"
|
|||
|
"\tBlocks Still Allocated. . . . . . . . %s\n", NiceNum(numBuf, goodMemoryAllocations - goodMemoryFrees, 0)));
|
|||
|
DEBUG_PUT(("\tMemory Still Allocated. . . . . . . . %s\n", NiceNum(numBuf, totalActualMemoryAllocated - totalActualMemoryFreed, 0)));
|
|||
|
DEBUG_PUT(("\n"
|
|||
|
"********************************************************************************\n"
|
|||
|
"\n"));
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
ReportMemoryBlocks(
|
|||
|
VOID
|
|||
|
)
|
|||
|
{
|
|||
|
DEBUG_PUT(("ReportMemoryBlocks\n\n"));
|
|||
|
|
|||
|
DEBUG_PUT(("AllocatedBlockList:\n\n"));
|
|||
|
|
|||
|
DumpMemoryList(&AllocatedBlockList);
|
|||
|
if (bDeferFree) {
|
|||
|
DumpDeferredFreeList();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DumpDeferredFreeList(
|
|||
|
VOID
|
|||
|
)
|
|||
|
{
|
|||
|
DEBUG_PUT(("DeferredFreeList:\n\n"));
|
|||
|
|
|||
|
DumpMemoryList(&DeferredFreeList);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DumpMemoryList(
|
|||
|
IN LPSERIALIZED_LIST lpList
|
|||
|
)
|
|||
|
{
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader;
|
|||
|
int counter = 1;
|
|||
|
|
|||
|
if (bUseSymbols) {
|
|||
|
|
|||
|
//
|
|||
|
// have to load IMAGEHLP.DLL here because we're in DLL_PROCESS_DETACH
|
|||
|
// and if we loaded it earlier, there's a good chance the system has
|
|||
|
// already freed it
|
|||
|
//
|
|||
|
|
|||
|
InitSymLib();
|
|||
|
}
|
|||
|
|
|||
|
LockSerializedList(lpList);
|
|||
|
lpHeader = (LPDEBUG_MEMORY_HEADER)HeadOfSerializedList(lpList);
|
|||
|
while (lpHeader != (LPDEBUG_MEMORY_HEADER)SlSelf(lpList)) {
|
|||
|
|
|||
|
DEBUG_PUT(("Block # %d\n", counter));
|
|||
|
|
|||
|
|
|||
|
if (!DumpBlock(lpHeader)) {
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
DEBUG_PUT(("********************************************************************************\n\n"));
|
|||
|
|
|||
|
lpHeader = (LPDEBUG_MEMORY_HEADER)(lpHeader->List.Flink);
|
|||
|
++counter;
|
|||
|
}
|
|||
|
UnlockSerializedList(lpList);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
FindAndDumpDeferredBlock(
|
|||
|
IN HLOCAL hLocal
|
|||
|
)
|
|||
|
{
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader;
|
|||
|
|
|||
|
LockSerializedList(&DeferredFreeList);
|
|||
|
|
|||
|
lpHeader = (LPDEBUG_MEMORY_HEADER)HeadOfSerializedList(&DeferredFreeList);
|
|||
|
while (lpHeader != (LPDEBUG_MEMORY_HEADER)SlSelf(&DeferredFreeList)) {
|
|||
|
if (hLocal == (HLOCAL)lpHeader) {
|
|||
|
DumpBlock(lpHeader);
|
|||
|
break;
|
|||
|
}
|
|||
|
lpHeader = (LPDEBUG_MEMORY_HEADER)(lpHeader->List.Flink);
|
|||
|
}
|
|||
|
|
|||
|
UnlockSerializedList(&DeferredFreeList);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
DumpBlock(
|
|||
|
IN LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
)
|
|||
|
{
|
|||
|
BOOL ok = DumpDebugMemoryHeader(lpHeader);
|
|||
|
|
|||
|
if (!ok && bStopDumpIfBadBlock) {
|
|||
|
|
|||
|
DEBUG_PUT(("*** stopping block dump: header @ %#x is bad\n", lpHeader));
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
if (bReportUserData) {
|
|||
|
DumpUserData(lpHeader);
|
|||
|
}
|
|||
|
if (bReportMemoryFooters) {
|
|||
|
|
|||
|
LPDEBUG_MEMORY_FOOTER lpFooter;
|
|||
|
|
|||
|
lpFooter = (LPDEBUG_MEMORY_FOOTER)
|
|||
|
((LPBYTE)lpHeader
|
|||
|
+ sizeof(*lpHeader)
|
|||
|
+ ROUND_UP_DWORD(lpHeader->RequestedLength));
|
|||
|
ok = DumpDebugMemoryFooter(lpFooter);
|
|||
|
if (!ok && bStopDumpIfBadBlock) {
|
|||
|
|
|||
|
DEBUG_PUT(("*** stopping block dump: footer @ %#x is bad\n", lpFooter));
|
|||
|
|
|||
|
return FALSE;
|
|||
|
}
|
|||
|
}
|
|||
|
return TRUE;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
DumpDebugMemoryHeader(
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
)
|
|||
|
{
|
|||
|
char numBuf[32];
|
|||
|
BOOL result;
|
|||
|
LPSTR symbol;
|
|||
|
DWORD offset;
|
|||
|
int i;
|
|||
|
char flagsBuf[256];
|
|||
|
|
|||
|
__try {
|
|||
|
DEBUG_PUT(("DEBUG_MEMORY_HEADER @ %#x:\n"
|
|||
|
"\n",
|
|||
|
lpHeader
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tList. . . . . . . . . F=%#x B=%#x\n",
|
|||
|
lpHeader->List.Flink,
|
|||
|
lpHeader->List.Blink
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tThread. . . . . . . . %#x\n",
|
|||
|
lpHeader->ThreadId
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tAllocated From. . . . %s!%d\n",
|
|||
|
SourceFilename(lpHeader->CreatedFile),
|
|||
|
lpHeader->CreatedLine
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tLast Accessed From. . %s!%d\n",
|
|||
|
SourceFilename(lpHeader->AccessedFile),
|
|||
|
lpHeader->AccessedLine
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tRequested Length. . . %s\n",
|
|||
|
NiceNum(numBuf, lpHeader->RequestedLength, 0)
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tBlock Length. . . . . %s\n",
|
|||
|
NiceNum(numBuf, lpHeader->BlockLength, 0)
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tActual Length . . . . %s\n",
|
|||
|
NiceNum(numBuf, lpHeader->ActualLength, 0)
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tSignature . . . . . . %x (%s)\n",
|
|||
|
lpHeader->Signature,
|
|||
|
(lpHeader->Signature == HEADER_SIGNATURE) ? "Good" : "BAD!!!"
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tFlags . . . . . . . . %08x %s\n",
|
|||
|
lpHeader->Flags,
|
|||
|
MapMemoryFlags(lpHeader->Flags, flagsBuf)
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tTime Deferred . . . . %08x\n",
|
|||
|
lpHeader->TimeDeferred
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tClash Test. . . . . . %d\n",
|
|||
|
lpHeader->ClashTest
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tLast Operation. . . . %s\n",
|
|||
|
MapLastAccessOperation(lpHeader->LastAccessOperation)
|
|||
|
));
|
|||
|
|
|||
|
#if DUMP_STACK
|
|||
|
#if ONE_STACK
|
|||
|
|
|||
|
if (lpHeader->Stack[0]) {
|
|||
|
symbol = DbgMemGetDebugSymbol((DWORD)lpHeader->Stack[0], &offset);
|
|||
|
} else {
|
|||
|
symbol = "";
|
|||
|
offset = 0;
|
|||
|
}
|
|||
|
DEBUG_PUT(("\tStack . . . . . . . . %08x %s+%#x\n",
|
|||
|
lpHeader->Stack[0],
|
|||
|
symbol,
|
|||
|
offset
|
|||
|
));
|
|||
|
for (i = 1; i < ARRAY_ELEMENTS(lpHeader->Stack); ++i) {
|
|||
|
//if (!lpHeader->lpHeader->Stack[i]) {
|
|||
|
// break;
|
|||
|
//}
|
|||
|
if (lpHeader->Stack[i]) {
|
|||
|
symbol = DbgMemGetDebugSymbol((DWORD)lpHeader->Stack[i], &offset);
|
|||
|
} else {
|
|||
|
symbol = "";
|
|||
|
offset = 0;
|
|||
|
}
|
|||
|
DEBUG_PUT(("\t. . . . . . . . . . . %08x %s+%#x\n",
|
|||
|
lpHeader->Stack[i],
|
|||
|
symbol,
|
|||
|
offset
|
|||
|
));
|
|||
|
}
|
|||
|
|
|||
|
#else
|
|||
|
|
|||
|
if (lpHeader->LastAccessStack[0]) {
|
|||
|
symbol = DbgMemGetDebugSymbol((DWORD)lpHeader->LastAccessStack[0], &offset);
|
|||
|
} else {
|
|||
|
symbol = "";
|
|||
|
offset = 0;
|
|||
|
}
|
|||
|
DEBUG_PUT(("\tLastAccessStack . . . %08x %s+%#x\n",
|
|||
|
lpHeader->LastAccessStack[0],
|
|||
|
symbol,
|
|||
|
offset
|
|||
|
));
|
|||
|
for (i = 1; i < ARRAY_ELEMENTS(lpHeader->LastAccessStack); ++i) {
|
|||
|
//if (!lpHeader->LastAccessStack[i]) {
|
|||
|
// break;
|
|||
|
//}
|
|||
|
if (lpHeader->LastAccessStack[i]) {
|
|||
|
symbol = DbgMemGetDebugSymbol((DWORD)lpHeader->LastAccessStack[i], &offset);
|
|||
|
} else {
|
|||
|
symbol = "";
|
|||
|
offset = 0;
|
|||
|
}
|
|||
|
DEBUG_PUT(("\t. . . . . . . . . . . %08x %s+%#x\n",
|
|||
|
lpHeader->LastAccessStack[i],
|
|||
|
symbol,
|
|||
|
offset
|
|||
|
));
|
|||
|
}
|
|||
|
if (lpHeader->CreateStack[0]) {
|
|||
|
symbol = DbgMemGetDebugSymbol((DWORD)lpHeader->CreateStack[0], &offset);
|
|||
|
} else {
|
|||
|
symbol = "";
|
|||
|
offset = 0;
|
|||
|
}
|
|||
|
DEBUG_PUT(("\tCreateStack . . . . . %08x %s+%#x\n",
|
|||
|
lpHeader->CreateStack[0],
|
|||
|
symbol,
|
|||
|
offset
|
|||
|
));
|
|||
|
for (i = 1; i < ARRAY_ELEMENTS(lpHeader->CreateStack); ++i) {
|
|||
|
//if (!lpHeader->lpHeader->CreateStack[i]) {
|
|||
|
// break;
|
|||
|
//}
|
|||
|
if (lpHeader->CreateStack[i]) {
|
|||
|
symbol = DbgMemGetDebugSymbol((DWORD)lpHeader->CreateStack[i], &offset);
|
|||
|
} else {
|
|||
|
symbol = "";
|
|||
|
offset = 0;
|
|||
|
}
|
|||
|
DEBUG_PUT(("\t. . . . . . . . . . . %08x %s+%#x\n",
|
|||
|
lpHeader->CreateStack[i],
|
|||
|
symbol,
|
|||
|
offset
|
|||
|
));
|
|||
|
}
|
|||
|
|
|||
|
#endif // ONE_STACK
|
|||
|
#endif // DUMP_STACK
|
|||
|
|
|||
|
DEBUG_PUT(("\tGuard . . . . . . . . %08x\n"
|
|||
|
"\n",
|
|||
|
lpHeader->Guard[0]
|
|||
|
));
|
|||
|
result = TRUE;
|
|||
|
} __except(EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
|
|||
|
//DEBUG_PUT(("DEBUG_MEMORY_HEADER @ %#x is BAD\n", lpHeader));
|
|||
|
|
|||
|
result = FALSE;
|
|||
|
}
|
|||
|
return result;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
BOOL
|
|||
|
DumpDebugMemoryFooter(
|
|||
|
LPDEBUG_MEMORY_FOOTER lpFooter
|
|||
|
)
|
|||
|
{
|
|||
|
char numBuf[32];
|
|||
|
BOOL result;
|
|||
|
|
|||
|
_try {
|
|||
|
DEBUG_PUT(("DEBUG_MEMORY_FOOTER @ %#x:\n"
|
|||
|
"\n",
|
|||
|
lpFooter
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tGuard . . . . . . . . %08x %08x %08x %08x\n",
|
|||
|
lpFooter->Guard[0],
|
|||
|
lpFooter->Guard[1],
|
|||
|
lpFooter->Guard[2],
|
|||
|
lpFooter->Guard[3]
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tSignature . . . . . . %x (%s)\n",
|
|||
|
lpFooter->Signature,
|
|||
|
(lpFooter->Signature == FOOTER_SIGNATURE) ? "Good" : "BAD!!!"
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tBlock Length. . . . . %s\n",
|
|||
|
NiceNum(numBuf, lpFooter->BlockLength, 0)
|
|||
|
));
|
|||
|
DEBUG_PUT(("\tGuard2. . . . . . . . %08x %08x\n"
|
|||
|
"\n",
|
|||
|
lpFooter->Guard2[0],
|
|||
|
lpFooter->Guard2[1]
|
|||
|
));
|
|||
|
result = TRUE;
|
|||
|
} __except(EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
|
|||
|
//DEBUG_PUT(("DEBUG_MEMORY_FOOTER @ %#x is BAD\n", lpFooter));
|
|||
|
|
|||
|
result = FALSE;
|
|||
|
}
|
|||
|
return result;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
VOID
|
|||
|
DumpUserData(
|
|||
|
LPDEBUG_MEMORY_HEADER lpHeader
|
|||
|
)
|
|||
|
{
|
|||
|
static char spaces[] = " "; // 15 * 3 + 2
|
|||
|
SIZE_T userSize = lpHeader->RequestedLength;
|
|||
|
SIZE_T Size = ROUND_UP_DWORD(userSize);
|
|||
|
LPBYTE Address = (LPBYTE)(lpHeader + 1);
|
|||
|
|
|||
|
DEBUG_PUT(("\t%d (%#x) bytes of user data (rounded to %d (%#x)) @ %#x\n\n",
|
|||
|
userSize,
|
|||
|
userSize,
|
|||
|
Size,
|
|||
|
Size,
|
|||
|
Address
|
|||
|
));
|
|||
|
|
|||
|
if (bLimitUserData && (Size > MaxUserDataDumped)) {
|
|||
|
|
|||
|
DEBUG_PUT(("*** User data length %d too large: limited to %d (probably bad block)\n",
|
|||
|
Size,
|
|||
|
MaxUserDataDumped
|
|||
|
));
|
|||
|
|
|||
|
Size = MaxUserDataDumped;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// dump out the data, debug style
|
|||
|
//
|
|||
|
|
|||
|
while (Size) {
|
|||
|
|
|||
|
char buf[128];
|
|||
|
int len;
|
|||
|
int clen;
|
|||
|
|
|||
|
rsprintf(buf, "\t%08x ", Address);
|
|||
|
|
|||
|
clen = (int)min(Size, 16);
|
|||
|
if (bDumpAsDwords) {
|
|||
|
len = clen / 4;
|
|||
|
} else {
|
|||
|
len = clen;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// dump the hex representation of each character - up to 16 per line
|
|||
|
//
|
|||
|
|
|||
|
int i;
|
|||
|
|
|||
|
for (i = 0; i < len; ++i) {
|
|||
|
if (bDumpAsDwords) {
|
|||
|
rsprintf(&buf[11 + i * 9], "%08x ", ((LPDWORD)Address)[i]);
|
|||
|
} else {
|
|||
|
rsprintf(&buf[11 + i * 3],
|
|||
|
((i & 15) == 7) ? "%02.2x-" : "%02.2x ",
|
|||
|
Address[i] & 0xff
|
|||
|
);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// write as many spaces as required to tab to ASCII field
|
|||
|
//
|
|||
|
|
|||
|
int offset;
|
|||
|
|
|||
|
if (bDumpAsDwords) {
|
|||
|
memcpy(&buf[11 + i * 9], spaces, (4 - len) * 9 + 2);
|
|||
|
offset = 49;
|
|||
|
} else {
|
|||
|
memcpy(&buf[11 + i * 3], spaces, (16 - len) * 3 + 2);
|
|||
|
offset = 60;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// dump ASCII representation of each character
|
|||
|
//
|
|||
|
|
|||
|
for (i = 0; i < clen; ++i) {
|
|||
|
|
|||
|
char ch;
|
|||
|
|
|||
|
ch = Address[i];
|
|||
|
buf[offset + i] = ((ch < 32) || (ch > 127)) ? '.' : ch;
|
|||
|
}
|
|||
|
|
|||
|
buf[offset + i++] = '\r';
|
|||
|
buf[offset + i++] = '\n';
|
|||
|
buf[offset + i] = 0;
|
|||
|
|
|||
|
//
|
|||
|
// InternetDebugOut() - no printf expansion (%s in data!), no prefixes
|
|||
|
//
|
|||
|
|
|||
|
InternetDebugOut(buf, FALSE);
|
|||
|
|
|||
|
Address += clen;
|
|||
|
Size -= clen;
|
|||
|
}
|
|||
|
|
|||
|
InternetDebugOut("\r\n", FALSE);
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
LPSTR
|
|||
|
MapLastAccessOperation(
|
|||
|
MEMORY_ACTION Action
|
|||
|
)
|
|||
|
{
|
|||
|
switch (Action) {
|
|||
|
case MemAllocate:
|
|||
|
return "Alloc";
|
|||
|
|
|||
|
case MemReallocate:
|
|||
|
return "Realloc";
|
|||
|
|
|||
|
case MemLock:
|
|||
|
return "Lock";
|
|||
|
|
|||
|
case MemUnlock:
|
|||
|
return "Unlock";
|
|||
|
|
|||
|
case MemFree:
|
|||
|
return "Free";
|
|||
|
|
|||
|
case MemSize:
|
|||
|
return "Size";
|
|||
|
}
|
|||
|
return "?";
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
LPSTR
|
|||
|
MapMemoryFlags(
|
|||
|
DWORD Flags,
|
|||
|
LPSTR Buffer
|
|||
|
)
|
|||
|
{
|
|||
|
LPSTR buf = Buffer;
|
|||
|
int i = 0;
|
|||
|
|
|||
|
*buf++ = '(';
|
|||
|
if (Flags & LMEM_DISCARDABLE) {
|
|||
|
buf += wsprintf(buf, "DISCARDABLE");
|
|||
|
++i;
|
|||
|
}
|
|||
|
if (Flags & LMEM_ZEROINIT) {
|
|||
|
if (i) {
|
|||
|
buf += wsprintf(buf, ", ");
|
|||
|
}
|
|||
|
++i;
|
|||
|
buf += wsprintf(buf, "ZEROINIT");
|
|||
|
}
|
|||
|
if (Flags & LMEM_NODISCARD) {
|
|||
|
if (i) {
|
|||
|
buf += wsprintf(buf, ", ");
|
|||
|
}
|
|||
|
++i;
|
|||
|
buf += wsprintf(buf, "NODISCARD");
|
|||
|
}
|
|||
|
if (Flags & LMEM_NOCOMPACT) {
|
|||
|
if (i) {
|
|||
|
buf += wsprintf(buf, ", ");
|
|||
|
}
|
|||
|
++i;
|
|||
|
buf += wsprintf(buf, "NOCOMPACT");
|
|||
|
}
|
|||
|
if (i) {
|
|||
|
buf += wsprintf(buf, ", ");
|
|||
|
}
|
|||
|
++i;
|
|||
|
buf += wsprintf(buf, (Flags & LMEM_MOVEABLE) ? "MOVEABLE" : "FIXED");
|
|||
|
*buf++ = ')';
|
|||
|
*buf++ = '\0';
|
|||
|
return Buffer;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
PRIVATE
|
|||
|
LPSTR
|
|||
|
DbgMemGetDebugSymbol(
|
|||
|
DWORD Address,
|
|||
|
LPDWORD Offset
|
|||
|
) {
|
|||
|
//if (!bUseSymbols) {
|
|||
|
// return "?";
|
|||
|
//}
|
|||
|
|
|||
|
//
|
|||
|
// RLF 04/14/98 - IMAGEHLP blowing up probably because we are doing this at
|
|||
|
// process detach time. Just return offset and run convsym
|
|||
|
// utility on wininet.log
|
|||
|
//
|
|||
|
|
|||
|
//return GetDebugSymbol(Address, Offset);
|
|||
|
*Offset = Address;
|
|||
|
return "";
|
|||
|
}
|
|||
|
|
|||
|
#endif // defined(USE_DEBUG_MEMORY)
|