248 lines
6.6 KiB
C
248 lines
6.6 KiB
C
/*
|
|
* $Id$
|
|
*/
|
|
|
|
/*
|
|
* Harbour Project source code:
|
|
* The Fixed Memory API
|
|
*
|
|
* Copyright 1999 Antonio Linares <alinares@fivetech.com>
|
|
* www - http://www.harbour-project.org
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version, with one exception:
|
|
*
|
|
* The exception is that if you link the Harbour Runtime Library (HRL)
|
|
* and/or the Harbour Virtual Machine (HVM) with other files to produce
|
|
* an executable, this does not by itself cause the resulting executable
|
|
* to be covered by the GNU General Public License. Your use of that
|
|
* executable is in no way restricted on account of linking the HRL
|
|
* and/or HVM code into it.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA (or visit
|
|
* their web site at http://www.gnu.org/).
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* The following parts are Copyright of the individual authors.
|
|
* www - http://www.harbour-project.org
|
|
*
|
|
* Copyright 1999 David G. Holm <dholm@jsd-llc.com>
|
|
* hb_xmemcpy()
|
|
* hb_xmemset()
|
|
*
|
|
* See doc/license.txt for licensing terms.
|
|
*
|
|
*/
|
|
|
|
/* NOTE: If you turn this on, the memory subsystem will collect information
|
|
about several statistical data about memory management, it will show
|
|
these on exit if memory seem to have leaked.
|
|
This should be normally turned off in a final release */
|
|
#define HB_FM_STATISTICS
|
|
|
|
#ifndef __MPW__
|
|
#include <malloc.h>
|
|
#endif
|
|
|
|
#include "extend.h"
|
|
#include "errorapi.h"
|
|
|
|
#ifdef HB_FM_STATISTICS
|
|
static ULONG s_ulMemoryBlocks = 0; /* memory blocks used */
|
|
static ULONG s_ulMemoryMaxBlocks = 0; /* maximum number of used memory blocks */
|
|
static ULONG s_ulMemoryMaxConsumed = 0; /* memory size consumed */
|
|
static ULONG s_ulMemoryConsumed = 0; /* memory max size consumed */
|
|
#endif
|
|
|
|
void * hb_xalloc( ULONG ulSize ) /* allocates fixed memory, returns NULL on failure */
|
|
{
|
|
void * pMem = malloc( ulSize + sizeof( ULONG ) );
|
|
|
|
if( ! pMem )
|
|
{
|
|
return pMem;
|
|
}
|
|
|
|
* ( ( ULONG * ) pMem ) = ulSize; /* we store the block size into it */
|
|
|
|
#ifdef HB_FM_STATISTICS
|
|
s_ulMemoryConsumed += ulSize;
|
|
s_ulMemoryMaxConsumed += ulSize;
|
|
s_ulMemoryBlocks++;
|
|
s_ulMemoryMaxBlocks++;
|
|
#endif
|
|
|
|
return ( char * ) pMem + sizeof( ULONG );
|
|
}
|
|
|
|
void * hb_xgrab( ULONG ulSize ) /* allocates fixed memory, exits on failure */
|
|
{
|
|
void * pMem = malloc( ulSize + sizeof( ULONG ) );
|
|
|
|
if( ! pMem )
|
|
{
|
|
hb_errInternal( 9999, "hb_xgrab can't allocate memory", NULL, NULL );
|
|
}
|
|
|
|
* ( ( ULONG * ) pMem ) = ulSize; /* we store the block size into it */
|
|
|
|
#ifdef HB_FM_STATISTICS
|
|
s_ulMemoryConsumed += ulSize;
|
|
s_ulMemoryMaxConsumed += ulSize;
|
|
s_ulMemoryBlocks++;
|
|
s_ulMemoryMaxBlocks++;
|
|
#endif
|
|
|
|
return ( char * ) pMem + sizeof( ULONG );
|
|
}
|
|
|
|
void * hb_xrealloc( void * pMem, ULONG ulSize ) /* reallocates memory */
|
|
{
|
|
ULONG ulMemSize = * ( ULONG * ) ( ( char * ) pMem - sizeof( ULONG ) );
|
|
void * pResult = realloc( ( char * ) pMem - sizeof( ULONG ), ulSize + sizeof( ULONG ) );
|
|
|
|
if( ! pResult )
|
|
{
|
|
hb_errInternal( 9999, "hb_xrealloc can't reallocate memory", NULL, NULL );
|
|
}
|
|
|
|
* ( ( ULONG * ) pResult ) = ulSize; /* we store the block size into it */
|
|
|
|
#ifdef HB_FM_STATISTICS
|
|
if( ! ulSize )
|
|
s_ulMemoryBlocks--;
|
|
|
|
s_ulMemoryConsumed += ( ulSize - ulMemSize );
|
|
if( ulSize > ulMemSize )
|
|
s_ulMemoryMaxConsumed += ulSize - ulMemSize;
|
|
#endif
|
|
|
|
return ( char * ) pResult + sizeof( ULONG );
|
|
}
|
|
|
|
void hb_xfree( void * pMem ) /* frees fixed memory */
|
|
{
|
|
ULONG ulMemSize = * ( ULONG * ) ( ( char * ) pMem - sizeof( ULONG ) );
|
|
|
|
if( pMem )
|
|
free( ( char * ) pMem - sizeof( ULONG ) );
|
|
else
|
|
hb_errInternal( 9999, "hb_xfree called with a null pointer", NULL, NULL );
|
|
|
|
#ifdef HB_FM_STATISTICS
|
|
s_ulMemoryConsumed -= ulMemSize;
|
|
s_ulMemoryBlocks--;
|
|
#endif
|
|
}
|
|
|
|
ULONG hb_xsize( void * pMem ) /* returns the size of an allocated memory block */
|
|
{
|
|
return * ( ULONG * ) ( ( char * ) pMem - sizeof( ULONG ) );
|
|
}
|
|
|
|
void hb_xinit( void ) /* Initialize fixed memory subsystem */
|
|
{
|
|
;
|
|
}
|
|
|
|
void hb_xexit( void ) /* Deinitialize fixed memory subsystem */
|
|
{
|
|
#ifdef HB_FM_STATISTICS
|
|
if( s_ulMemoryBlocks || hb_cmdargCheck( "INFO" ) )
|
|
{
|
|
char buffer[ 100 ];
|
|
|
|
hb_outerr( hb_consoleGetNewLine(), 0 );
|
|
hb_outerr( "----------------------------------------", 0 );
|
|
hb_outerr( hb_consoleGetNewLine(), 0 );
|
|
sprintf( buffer, "Total memory allocated: %ld bytes (%lu blocks)", s_ulMemoryMaxConsumed, s_ulMemoryMaxBlocks );
|
|
hb_outerr( buffer, 0 );
|
|
|
|
if( s_ulMemoryBlocks )
|
|
{
|
|
hb_outerr( hb_consoleGetNewLine(), 0 );
|
|
sprintf( buffer, "WARNING! Memory allocated but not released: %ld bytes (%ld blocks)", s_ulMemoryConsumed, s_ulMemoryBlocks );
|
|
hb_outerr( buffer, 0 );
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if UINT_MAX != ULONG_MAX
|
|
|
|
/* hb_xmemcpy and hb_xmemset are only needed when
|
|
unsigned int and unsigned long differ in length */
|
|
|
|
void * hb_xmemcpy( void * pDestArg, void * pSourceArg, ULONG ulLen )
|
|
{
|
|
BYTE * pDest = ( BYTE * ) pDestArg;
|
|
BYTE * pSource = ( BYTE * ) pSourceArg;
|
|
ULONG ulRemaining = ulLen;
|
|
int iCopySize;
|
|
|
|
while( ulRemaining )
|
|
{
|
|
/* Overcome the memcpy() size_t limitation */
|
|
if( ulRemaining > UINT_MAX )
|
|
{
|
|
iCopySize = UINT_MAX;
|
|
ulRemaining -= ( ULONG ) iCopySize;
|
|
}
|
|
else
|
|
{
|
|
iCopySize = ( int ) ulRemaining;
|
|
ulRemaining = 0;
|
|
}
|
|
memcpy( pDest, pSource, iCopySize );
|
|
pDest += iCopySize;
|
|
pSource += iCopySize;
|
|
}
|
|
return pDestArg;
|
|
}
|
|
|
|
void * hb_xmemset( void * pDestArg, int iFill, ULONG ulLen )
|
|
{
|
|
BYTE * pDest = ( BYTE * ) pDestArg;
|
|
ULONG ulRemaining = ulLen;
|
|
int iSetSize;
|
|
|
|
while( ulRemaining )
|
|
{
|
|
/* Overcome the memset() size_t limitation */
|
|
if( ulRemaining > UINT_MAX )
|
|
{
|
|
iSetSize = UINT_MAX;
|
|
ulRemaining -= ( ULONG ) iSetSize;
|
|
}
|
|
else
|
|
{
|
|
iSetSize = ( int ) ulRemaining;
|
|
ulRemaining = 0;
|
|
}
|
|
memset( pDest, iFill, iSetSize );
|
|
pDest += iSetSize;
|
|
}
|
|
return pDestArg;
|
|
}
|
|
|
|
#endif
|
|
|
|
HARBOUR HB_MEMORY( void )
|
|
{
|
|
/* TODO: Implement */
|
|
|
|
hb_retni( 9999 );
|
|
}
|
|
|