2003-11-01 06:35:29 +01:00
|
|
|
/* -*-mode: C; fill-column: 78; c-basic-offset: 4; -*- */
|
|
|
|
/*
|
2006-01-08 02:25:02 +01:00
|
|
|
* Copyright 2001 by Eric House (xwords@eehouse.org). All rights reserved.
|
2003-11-01 06:35:29 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef MEM_DEBUG
|
|
|
|
|
|
|
|
#include "mempool.h"
|
|
|
|
#include "comtypes.h"
|
|
|
|
#include "xwstream.h"
|
|
|
|
|
|
|
|
#ifdef CPLUS
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
typedef struct MemPoolEntry {
|
|
|
|
struct MemPoolEntry* next;
|
2006-09-15 09:32:04 +02:00
|
|
|
const char* fileName;
|
2003-11-01 06:35:29 +01:00
|
|
|
XP_U32 lineNo;
|
|
|
|
XP_U32 size;
|
|
|
|
void* ptr;
|
|
|
|
} MemPoolEntry;
|
|
|
|
|
|
|
|
struct MemPoolCtx {
|
|
|
|
MemPoolEntry* freeList;
|
|
|
|
MemPoolEntry* usedList;
|
|
|
|
|
|
|
|
XP_U16 nFree;
|
|
|
|
XP_U16 nUsed;
|
|
|
|
XP_U16 nAllocs;
|
|
|
|
};
|
|
|
|
|
|
|
|
/*--------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
MemPoolCtx*
|
2007-04-14 18:27:19 +02:00
|
|
|
mpool_make( void )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
MemPoolCtx* result = (MemPoolCtx*)XP_PLATMALLOC( sizeof(*result) );
|
|
|
|
XP_MEMSET( result, 0, sizeof(*result) );
|
|
|
|
return result;
|
|
|
|
} /* mpool_make */
|
|
|
|
|
|
|
|
static void
|
|
|
|
freeList( MemPoolEntry* entry )
|
|
|
|
{
|
|
|
|
while ( !!entry ) {
|
|
|
|
MemPoolEntry* next = entry->next;
|
|
|
|
|
|
|
|
XP_ASSERT( !entry->ptr );
|
|
|
|
XP_PLATFREE( entry );
|
|
|
|
|
|
|
|
entry = next;
|
|
|
|
}
|
|
|
|
} /* freeList */
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
static char*
|
|
|
|
checkIsText( MemPoolEntry* entry )
|
|
|
|
{
|
|
|
|
unsigned char* txt = (unsigned char*)entry->ptr;
|
|
|
|
XP_U32 len = entry->size;
|
|
|
|
|
|
|
|
while ( len-- ) {
|
|
|
|
unsigned char c = *txt++;
|
|
|
|
if ( c < 32 || c > 127 ) {
|
|
|
|
if ( len == 0 && c == '\0' ) {
|
|
|
|
return (char*)entry->ptr;
|
|
|
|
} else {
|
|
|
|
return (char*)NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (char*)NULL;
|
|
|
|
} /* checkIsText */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void
|
|
|
|
mpool_destroy( MemPoolCtx* mpool )
|
|
|
|
{
|
|
|
|
if ( !!mpool->usedList ) {
|
|
|
|
MemPoolEntry* entry;
|
|
|
|
for ( entry = mpool->usedList; !!entry; entry = entry->next ) {
|
|
|
|
#ifndef FOR_GREMLINS /* I don't want to hear about this right now */
|
|
|
|
XP_WARNF( "0x%lx from ln %ld of %s\n",
|
|
|
|
entry->ptr, entry->lineNo, entry->fileName );
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
char* tryTxt;
|
|
|
|
tryTxt = checkIsText( entry );
|
|
|
|
if ( !!tryTxt ) {
|
|
|
|
XP_WARNF( "--- looks like text: %s\n", tryTxt );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef FOR_GREMLINS
|
|
|
|
XP_ASSERT( !mpool->usedList && mpool->nUsed == 0 );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
freeList( mpool->freeList );
|
|
|
|
XP_PLATFREE( mpool );
|
|
|
|
} /* mpool_destroy */
|
|
|
|
|
|
|
|
void*
|
2006-09-15 09:32:04 +02:00
|
|
|
mpool_alloc( MemPoolCtx* mpool, XP_U32 size, const char* file, XP_U32 lineNo )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
MemPoolEntry* entry;
|
|
|
|
|
|
|
|
if ( mpool->nFree > 0 ) {
|
|
|
|
entry = mpool->freeList;
|
|
|
|
mpool->freeList = entry->next;
|
|
|
|
--mpool->nFree;
|
|
|
|
} else {
|
|
|
|
entry = (MemPoolEntry*)XP_PLATMALLOC( sizeof(*entry) );
|
|
|
|
}
|
|
|
|
|
|
|
|
entry->next = mpool->usedList;
|
|
|
|
mpool->usedList = entry;
|
|
|
|
|
|
|
|
entry->fileName = file;
|
|
|
|
entry->lineNo = lineNo;
|
|
|
|
entry->size = size;
|
|
|
|
entry->ptr = XP_PLATMALLOC( size );
|
|
|
|
XP_ASSERT( !!entry->ptr );
|
|
|
|
|
|
|
|
++mpool->nUsed;
|
|
|
|
++mpool->nAllocs;
|
|
|
|
|
|
|
|
return entry->ptr;
|
|
|
|
} /* mpool_alloc */
|
|
|
|
|
|
|
|
static MemPoolEntry*
|
|
|
|
findEntryFor( MemPoolCtx* mpool, void* ptr, MemPoolEntry** prevP )
|
|
|
|
{
|
|
|
|
MemPoolEntry* entry;
|
|
|
|
MemPoolEntry* prev;
|
|
|
|
|
|
|
|
for ( prev = (MemPoolEntry*)NULL, entry = mpool->usedList; !!entry;
|
|
|
|
prev = entry, entry = prev->next ) {
|
|
|
|
|
|
|
|
if ( entry->ptr == ptr ) {
|
|
|
|
|
|
|
|
if ( !!prevP ) {
|
|
|
|
*prevP = prev;
|
|
|
|
}
|
|
|
|
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
XP_ASSERT(0);
|
|
|
|
return (MemPoolEntry*)NULL;
|
|
|
|
} /* findEntryFor */
|
|
|
|
|
|
|
|
void*
|
|
|
|
mpool_realloc( MemPoolCtx* mpool, void* ptr, XP_U32 newsize )
|
|
|
|
{
|
|
|
|
MemPoolEntry* entry = findEntryFor( mpool, ptr, (MemPoolEntry**)NULL );
|
|
|
|
|
|
|
|
entry->ptr = XP_PLATREALLOC( entry->ptr, newsize );
|
|
|
|
XP_ASSERT( !!entry->ptr );
|
|
|
|
return entry->ptr;
|
|
|
|
} /* mpool_realloc */
|
|
|
|
|
|
|
|
void
|
2006-08-16 15:44:44 +02:00
|
|
|
mpool_free( MemPoolCtx* mpool, void* ptr )
|
2003-11-01 06:35:29 +01:00
|
|
|
{
|
|
|
|
MemPoolEntry* entry;
|
|
|
|
MemPoolEntry* prev;
|
|
|
|
|
|
|
|
entry = findEntryFor( mpool, ptr, &prev );
|
|
|
|
|
|
|
|
if ( !!entry ) {
|
|
|
|
|
|
|
|
if ( !!prev ) {
|
|
|
|
prev->next = entry->next;
|
|
|
|
} else {
|
|
|
|
mpool->usedList = entry->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_MEMSET( entry->ptr, 0x00, entry->size );
|
|
|
|
XP_PLATFREE( entry->ptr );
|
|
|
|
entry->ptr = NULL;
|
|
|
|
|
|
|
|
entry->next = mpool->freeList;
|
|
|
|
mpool->freeList = entry;
|
|
|
|
|
|
|
|
++mpool->nFree;
|
|
|
|
--mpool->nUsed;
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
XP_ASSERT( 0 );
|
|
|
|
} /* mpool_free */
|
|
|
|
|
|
|
|
void
|
|
|
|
mpool_stats( MemPoolCtx* mpool, XWStreamCtxt* stream )
|
|
|
|
{
|
|
|
|
XP_UCHAR buf[128];
|
|
|
|
MemPoolEntry* entry;
|
|
|
|
|
|
|
|
XP_SNPRINTF( buf, sizeof(buf), (XP_UCHAR*)"Number of blocks in use: %d\n"
|
|
|
|
"Number of free blocks: %d\n"
|
|
|
|
"Total number of blocks allocated: %d\n",
|
|
|
|
mpool->nUsed, mpool->nFree, mpool->nAllocs );
|
2006-04-25 15:31:15 +02:00
|
|
|
stream_putString( stream, buf );
|
2003-11-01 06:35:29 +01:00
|
|
|
|
|
|
|
for ( entry = mpool->usedList; !!entry; entry = entry->next ) {
|
|
|
|
XP_SNPRINTF( buf, sizeof(buf),
|
|
|
|
(XP_UCHAR*)"%ld byte block allocated %s: line %ld\n",
|
|
|
|
entry->size, entry->fileName, entry->lineNo );
|
2006-04-25 15:31:15 +02:00
|
|
|
stream_putString( stream, buf );
|
2003-11-01 06:35:29 +01:00
|
|
|
}
|
|
|
|
} /* mpool_stats */
|
|
|
|
|
|
|
|
XP_U16
|
|
|
|
mpool_getNUsed( MemPoolCtx* mpool )
|
|
|
|
{
|
|
|
|
return mpool->nUsed;
|
|
|
|
} /* mpool_getNUsed */
|
|
|
|
|
|
|
|
#ifdef CPLUS
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* MEM_DEBUG */
|