Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

461 lines
8.7 KiB

//+-------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1997.
//
// File: Alloc.cpp
//
// Contents: Allocation routines
//
// Classes:
//
// Notes:
//
// History: 05-Nov-97 rogerg Created.
//
//--------------------------------------------------------------------------
#include "lib.h"
#ifdef _DEBUG
extern DWORD g_dwDebugLeakDetection;
#endif // _DEBUG
//+-------------------------------------------------------------------
//
// Function: ::operator new
//
// Synopsis: Our operator new implementation
//
// Arguments: [size] -- Size of memory to allocate
//
//
// Notes:
//
//--------------------------------------------------------------------
inline void* __cdecl operator new (size_t size)
{
return(ALLOC(size));
}
//+-------------------------------------------------------------------
//
// Function: ::operator delete
//
// Synopsis: Our operator deleteimplementation
//
// Arguments: lpv-- Pointer to memory to free
//
//
// Notes:
//
//--------------------------------------------------------------------
inline void __cdecl operator delete(void FAR* lpv)
{
FREE(lpv);
}
//
// Allocator for MIDL stubs
//
//+-------------------------------------------------------------------
//
// Function: MIDL_user_allocate
//
// Synopsis:
//
// Arguments: lpv-- Pointer to memory to free
//
//
// Notes:
//
//--------------------------------------------------------------------
extern "C" void __RPC_FAR * __RPC_API
MIDL_user_allocate(
IN size_t len
)
{
return ALLOC(len);
}
//+-------------------------------------------------------------------
//
// Function: MIDL_user_free
//
// Synopsis:
//
// Arguments: ptr-- Pointer to memory to free
//
//
// Notes:
//
//--------------------------------------------------------------------
extern "C" void __RPC_API
MIDL_user_free(
IN void __RPC_FAR * ptr
)
{
FREE(ptr);
}
//+---------------------------------------------------------------------------
//
// function: ALLOC, public
//
// Synopsis: memory allocator
//
// Arguments: [cb] - requested size of memory to alloc.
//
// Returns: Pointer to newly allocated memory, NULL on failure
//
// Modifies:
//
// History: 05-Nov-97 rogerg Created.
//
//----------------------------------------------------------------------------
LPVOID ALLOC(ULONG cb)
{
void *pv;
pv = LocalAlloc(LMEM_FIXED,cb);
#ifdef _DEBUG
if (NULL != pv) // under debug always initialize to -1 to catch unitialize errors.
{
memset(pv,MEMINITVALUE,cb);
// add entry to arena
ADDENTRY(pv,cb);
}
#endif // _DEBUG
return pv;
}
//+---------------------------------------------------------------------------
//
// function: FREE, public
//
// Synopsis: memory destructor
//
// Arguments: [pv] - pointer to memory to be released.
//
// Returns:
//
// Modifies:
//
// History: 05-Nov-97 rogerg Created.
//
//----------------------------------------------------------------------------
void FREE(void* pv)
{
#ifdef _DEBUG
if (NULL != pv) // under debug always initialize to -1 to catch unitialize errors.
{
UINT cb;
Assert(LMEM_INVALID_HANDLE != LocalFlags(pv));
cb = (UINT)LocalSize(pv); // returns zero on failure
memset(pv,MEMFREEVALUE,cb);
}
FREEENTRY(pv);
Assert(pv);
#endif // _DEBUG
LocalFree(pv);
}
//+---------------------------------------------------------------------------
//
// function: REALLOC, public
//
// Synopsis: reallocs memory
//
// Arguments: [pv] - pointer to memory to be released.
// [cb] - size to resize the memory to.
//
// Returns:
//
// Modifies:
//
// History: 22-Jul-98 rogerg Created.
//
//----------------------------------------------------------------------------
LPVOID REALLOC(void *pv,ULONG cb)
{
Assert(pv);
FREEENTRY(pv);
pv = LocalReAlloc(pv,cb,LMEM_MOVEABLE);
ADDENTRY(pv,cb);
return pv;
}
#if MEMLEAKDETECTION
#define ARENA_SIZE 5000
MEMARENA *g_MemChk = NULL;
BOOL g_InitArenaFailed = FALSE;
ULONG g_AllocNumber = 0;
ULONG g_FreeNumber = 0;
ULONG g_ulAssertonAlloc = -1; // set this to the leak number for getting the stack trace.
CRITICAL_SECTION g_ArenaCriticalSection;
//+---------------------------------------------------------------------------
//
// function: InitArena, private
//
// Synopsis: Initializes our memory leak arena
//
// Arguments: .
//
// Returns: TRUE if Arena was successsfully initialized
//
// Modifies:
//
// History: 05-Nov-97 rogerg Created.
//
//----------------------------------------------------------------------------
BOOL InitArena()
{
int i = 0;
Assert(g_dwDebugLeakDetection);
if (!g_dwDebugLeakDetection || g_InitArenaFailed)
{
return FALSE;
}
// if g_MemChk is NULL Allocate space for it,
// else we are already initialized
if (g_MemChk == NULL)
{
g_MemChk = (MEMARENA *) CoTaskMemAlloc(sizeof(MEMARENA) * (ARENA_SIZE + 1)) ;
if (g_MemChk == NULL)
{
AssertSz(0,"Unable to Initialize LeakDetection");
g_InitArenaFailed = TRUE;
return FALSE;
}
g_MemChk[ARENA_SIZE - 1].Free = 'END ';
while(g_MemChk[i].Free != 'END ' )
{
g_MemChk[i].Free = 'FREE';
i++ ;
}
g_AllocNumber = 0;
g_FreeNumber = 0;
InitializeCriticalSection(&g_ArenaCriticalSection);
}
return TRUE;
}
//+---------------------------------------------------------------------------
//
// function: AddEntry, private
//
// Synopsis: Adds an entry to our memory leak arena
//
// Arguments: [lpv] - pointer to add to arena
// [size] - size of pointer to add to the arena
//
// Returns:
//
// Modifies:
//
// History: 05-Nov-97 rogerg Created.
//
//----------------------------------------------------------------------------
void AddEntry( void *lpv, unsigned long size)
{
LPMEMARENA lpMem = g_MemChk;
CCriticalSection critsect(&g_ArenaCriticalSection,GetCurrentThreadId());
int i = 0;
if (!g_dwDebugLeakDetection)
{
return;
}
if (!InitArena())
{
return;
}
critsect.Enter();
// Find first available entry.
while(g_MemChk[i].Free != 'FREE' && g_MemChk[i].Free != 'END ')
{
i++;
}
if ( g_MemChk[i].Free != 'END ' )
{
g_AllocNumber++;
g_MemChk[i].Free = 'Ptr ';
g_MemChk[i].lpv = lpv;
g_MemChk[i].size = size;
g_MemChk[i].Order = g_AllocNumber;
if (g_AllocNumber == g_ulAssertonAlloc)
{
AssertSz(0,"Allocating AllocNumber");
}
}
else
{
AssertSz(0,"Arena is full");
}
critsect.Leave();
return;
}
//+---------------------------------------------------------------------------
//
// function: FreeEntry, private
//
// Synopsis: removes an entry from the memory arena
//
// Arguments: [lpv] - pointer to remove from the arena
//
// Returns:
//
// Modifies:
//
// History: 05-Nov-97 rogerg Created.
//
//----------------------------------------------------------------------------
void FreeEntry ( void *lpv )
{
int i = 0;
BOOL fFoundEntry = FALSE;
CCriticalSection critsect(&g_ArenaCriticalSection,GetCurrentThreadId());
if (!g_dwDebugLeakDetection)
{
return;
}
if (!InitArena())
{
return;
}
critsect.Enter();
g_FreeNumber++;
// Find entry.
while( g_MemChk[i].Free != 'END ' )
{
if ( g_MemChk[i].lpv == lpv )
{
g_MemChk[i].Free = 'FREE';
g_MemChk[i].lpv = NULL;
g_MemChk[i].size = 0;
g_MemChk[i].Order = 0;
fFoundEntry = TRUE;
break;
}
i++;
}
if (!fFoundEntry)
{
char buf[255];
wsprintfA(buf, "??Freeing block not in mem arena %X\n", lpv);
AssertSz(0,buf);
}
critsect.Leave();
}
//+---------------------------------------------------------------------------
//
// function: WalkArena, private
//
// Synopsis: walks arena asserting if finds any leaks.
//
// Arguments:
//
// Returns:
//
// Modifies:
//
// History: 05-Nov-97 rogerg Created.
//
//----------------------------------------------------------------------------
void WalkArena()
{
int i = 0;
if (!g_dwDebugLeakDetection)
{
return;
}
if (!InitArena())
{
return;
}
while( g_MemChk[i].Free != 'END ' )
{
if ( g_MemChk[i].Free != 'FREE' )
{
AssertSz(0,"You Have a Leak");
}
i++;
}
g_MemChk = NULL;
}
#endif // MEMLEAKDETECTION