2020-09-30 16:53:55 +02:00

200 lines
4.3 KiB
C++

// memmgr.c
//
// This file contains definitions for the memory management.
// Implementation details may change so beware of relying on internal details.
//
#include "private.h"
#include "memmgr.h"
#ifdef DEBUG
int cAllocMem = 0; // Amount of memory alloced
int cAlloc = 0; // Count of allocs outstanding
int cAllocMaxMem = 0; // Max amount of memory ever alloced.
#endif
#ifdef DEBUG
int gFailure = 0;
#endif
/******************************Public*Routine******************************\
* ExternAlloc
*
* This guy keeps the size in the first long so we can fake a realloc. Lot's
* of debug checking for heap overwrites.
*
* History:
* 19-Nov-1996 -by- Patrick Haluptzok patrickh
* Wrote it.
\**************************************************************************/
void *ExternAlloc(DWORD cb)
{
long *pl;
DWORD cbAlloc;
#ifdef DEBUG
#ifndef WINCE
//
// If gFailure is 0 nothing happens, if it's non-zero we
// fail 1 in gFailure allocations.
//
if (gFailure)
{
if (((rand() * gFailure) / (RAND_MAX + 1)) == 0)
{
return (void *) NULL;
}
}
#endif
#endif
// Since we can't use realloc on WINCE, we need to save the original size for memcpy
// in our own realloc function.
cbAlloc = cb + 4;
#ifdef DEBUG
cbAlloc += 3; // round it up to DWORD boundary
cbAlloc &= ~3;
cbAlloc += 8; // write size at begining and overwrite detector at begining and end
#endif
pl = (long *) malloc(cbAlloc);
if (pl == (long *) NULL)
return pl;
// Stamp this baby full of invalid bytes so code that relies on 0's in it are sniffed out.
#ifdef DEBUG
memset(pl,0xff,cbAlloc);
#endif
// OK, tuck the object size away at the begining
*(pl++) = cb;
#ifdef DEBUG
*(pl++) = 0xDEADBEEF;
pl[(cbAlloc / 4) - 3] = 0xDEADBEEF;
cAlloc++;
cAllocMem += cb;
if (cAllocMem > cAllocMaxMem)
{
TCHAR szDebug[128];
cAllocMaxMem = cAllocMem;
wsprintf(szDebug, TEXT("cAllocMaxMem = %d \r\n"), cAllocMaxMem);
OutputDebugString(szDebug);
}
#endif
return pl;
}
/******************************Public*Routine******************************\
* ExternRealloc
*
* Well this not good but we want the same exact code on NT and WINCE and
* we can't find a way to use the flags and have Realloc work the same on
* both. Realloc is a very infrequent event so this work for us.
*
* History:
* 19-Nov-1996 -by- Patrick Haluptzok patrickh
* Wrote it.
\**************************************************************************/
void *ExternRealloc(void *pv, DWORD cbNew)
{
void *pvNew = ExternAlloc(cbNew);
if (pv && pvNew)
{
long *pl;
DWORD cb;
pl = (long *) pv;
#ifdef DEBUG
pl--;
#endif
cb = (DWORD) *(--pl);
memcpy(pvNew, pv, min(cbNew, cb));
ExternFree(pv);
}
return pvNew;
}
/******************************Public*Routine******************************\
* ExternFree
*
* Free up the memory, in debug mode check for heap corruption !
*
* History:
* 19-Nov-1996 -by- Patrick Haluptzok patrickh
* Wrote it.
\**************************************************************************/
void ExternFree(void *pv)
{
long *pl;
// We now allow freeing of null pointers
if (pv == (void *) NULL)
return;
pl = (long *) pv;
pl--;
#ifdef DEBUG
{
int cbAlloc;
// Check nothing has been stepped on.
pl--;
cbAlloc = *pl;
cAllocMem -= cbAlloc;
cbAlloc = (cbAlloc + 11) & ~3;
cAlloc--;
}
#endif
free(pl);
}
char *Externstrdup( const char *strSource )
{
int nLen = 0;
char* pszOut = NULL;
// fail immediately on a null pointer
if (NULL == strSource)
return NULL;
// get the length of the ansi string
nLen = strlen(strSource) * sizeof(char);
// fail on a 0 length string
// @todo(petewil) - is this right, or return 0 length string instead?
if (0 == nLen)
return NULL;
// allow room for a trailing null
nLen += sizeof(char);
// allocate space for the string
pszOut = (char*)ExternAlloc(nLen);
if (NULL == pszOut)
return NULL;
// copy the string into the buffer provided
StringCchCopyA(pszOut, nLen, strSource);
return pszOut;
}