2001-02-23 17:55:21 +00:00
/*
2002-01-01 16:50:03 +00:00
* xmlmemory . c : libxml memory allocator wrapper .
2001-02-23 17:55:21 +00:00
*
2001-06-24 12:13:24 +00:00
* daniel @ veillard . com
2001-02-23 17:55:21 +00:00
*/
2002-03-18 19:37:11 +00:00
# define IN_LIBXML
2001-04-21 16:57:29 +00:00
# include "libxml.h"
2001-02-23 17:55:21 +00:00
# include <string.h>
2022-03-02 00:29:17 +01:00
# include <stdlib.h>
# include <ctype.h>
# include <time.h>
2001-02-23 17:55:21 +00:00
2002-02-06 16:06:58 +00:00
/**
* MEM_LIST :
*
2008-07-30 12:58:11 +00:00
* keep track of all allocated blocks for error reporting
2002-02-06 16:06:58 +00:00
* Always build the memory list !
*/
2003-09-29 10:55:05 +00:00
# ifdef DEBUG_MEMORY_LOCATION
2002-02-06 16:06:58 +00:00
# ifndef MEM_LIST
# define MEM_LIST /* keep a list of all the allocated memory blocks */
# endif
2003-09-29 10:55:05 +00:00
# endif
2001-02-23 17:55:21 +00:00
# include <libxml/xmlmemory.h>
# include <libxml/xmlerror.h>
2023-09-20 17:00:50 +02:00
# include <libxml/parser.h>
2003-11-29 10:47:56 +00:00
# include <libxml/threads.h>
2001-02-23 17:55:21 +00:00
2022-11-24 20:52:57 +01:00
# include "private/memory.h"
2022-11-24 20:54:18 +01:00
# include "private/threads.h"
2022-11-24 20:52:57 +01:00
2003-09-29 09:22:39 +00:00
static unsigned long debugMemSize = 0 ;
2004-11-02 14:52:23 +00:00
static unsigned long debugMemBlocks = 0 ;
2003-09-29 09:22:39 +00:00
static unsigned long debugMaxMemSize = 0 ;
2022-11-24 20:54:18 +01:00
static xmlMutex xmlMemMutex ;
2003-09-28 18:58:27 +00:00
2001-03-24 17:00:36 +00:00
void xmlMallocBreakpoint ( void ) ;
/************************************************************************
* *
2012-09-11 13:26:36 +08:00
* Macros , variables and associated types *
2001-03-24 17:00:36 +00:00
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-06-10 13:00:15 +00:00
# if !defined(LIBXML_THREAD_ENABLED) && !defined(LIBXML_THREAD_ALLOC_ENABLED)
2001-02-23 17:55:21 +00:00
# ifdef xmlMalloc
# undef xmlMalloc
# endif
# ifdef xmlRealloc
# undef xmlRealloc
# endif
# ifdef xmlMemStrdup
# undef xmlMemStrdup
# endif
2004-06-10 13:00:15 +00:00
# endif
2001-02-23 17:55:21 +00:00
/*
2020-03-08 17:19:42 +01:00
* Each of the blocks allocated begin with a header containing information
2001-02-23 17:55:21 +00:00
*/
2022-01-25 02:21:05 +01:00
# define MEMTAG 0x5aa5U
2001-02-23 17:55:21 +00:00
# define MALLOC_TYPE 1
# define REALLOC_TYPE 2
# define STRDUP_TYPE 3
2003-04-19 00:07:51 +00:00
# define MALLOC_ATOMIC_TYPE 4
# define REALLOC_ATOMIC_TYPE 5
2001-02-23 17:55:21 +00:00
typedef struct memnod {
unsigned int mh_tag ;
unsigned int mh_type ;
unsigned long mh_number ;
size_t mh_size ;
# ifdef MEM_LIST
struct memnod * mh_next ;
struct memnod * mh_prev ;
# endif
const char * mh_file ;
unsigned int mh_line ;
} MEMHDR ;
# ifdef SUN4
# define ALIGN_SIZE 16
# else
# define ALIGN_SIZE sizeof(double)
# endif
# define HDR_SIZE sizeof(MEMHDR)
# define RESERVE_SIZE (((HDR_SIZE + (ALIGN_SIZE-1)) \
/ ALIGN_SIZE ) * ALIGN_SIZE )
2016-04-05 12:05:25 -07:00
# define MAX_SIZE_T ((size_t)-1)
2001-02-23 17:55:21 +00:00
2017-06-17 14:13:51 +02:00
# define CLIENT_2_HDR(a) ((void *) (((char *) (a)) - RESERVE_SIZE))
2001-02-23 17:55:21 +00:00
# define HDR_2_CLIENT(a) ((void *) (((char *) (a)) + RESERVE_SIZE))
2003-11-29 10:47:56 +00:00
static unsigned int block = 0 ;
static unsigned int xmlMemStopAtBlock = 0 ;
2001-10-11 22:55:55 +00:00
static void * xmlMemTraceBlockAt = NULL ;
2001-02-23 17:55:21 +00:00
# ifdef MEM_LIST
static MEMHDR * memlist = NULL ;
# endif
2002-12-10 15:19:08 +00:00
static void debugmem_tag_error ( void * addr ) ;
2001-02-23 17:55:21 +00:00
# ifdef MEM_LIST
2002-12-10 15:19:08 +00:00
static void debugmem_list_add ( MEMHDR * ) ;
static void debugmem_list_delete ( MEMHDR * ) ;
2001-02-23 17:55:21 +00:00
# endif
# define Mem_Tag_Err(a) debugmem_tag_error(a);
# ifndef TEST_POINT
# define TEST_POINT
# endif
/**
* xmlMallocBreakpoint :
*
* Breakpoint to use in conjunction with xmlMemStopAtBlock . When the block
* number reaches the specified value this function is called . One need to add a breakpoint
* to it to get the context in which the given block is allocated .
*/
void
xmlMallocBreakpoint ( void ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2001-02-23 17:55:21 +00:00
" xmlMallocBreakpoint reached on block %d \n " , xmlMemStopAtBlock ) ;
}
/**
* xmlMallocLoc :
* @ size : an int specifying the size in byte to allocate .
* @ file : the file name or NULL
* @ line : the line number
*
* a malloc ( ) equivalent , with logging of the allocation info .
*
* Returns a pointer to the allocated area or NULL in case of lack of memory .
*/
void *
2001-07-17 21:38:51 +00:00
xmlMallocLoc ( size_t size , const char * file , int line )
2001-02-23 17:55:21 +00:00
{
MEMHDR * p ;
2001-07-30 13:42:13 +00:00
void * ret ;
2008-07-30 12:58:11 +00:00
2022-11-25 12:21:49 +01:00
xmlInitParser ( ) ;
2001-02-23 17:55:21 +00:00
TEST_POINT
2008-07-30 12:58:11 +00:00
2017-06-06 13:21:14 +02:00
if ( size > ( MAX_SIZE_T - RESERVE_SIZE ) ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2017-06-06 13:21:14 +02:00
" xmlMallocLoc : Unsigned overflow \n " ) ;
return ( NULL ) ;
}
2001-02-23 17:55:21 +00:00
p = ( MEMHDR * ) malloc ( RESERVE_SIZE + size ) ;
if ( ! p ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2002-01-01 16:50:03 +00:00
" xmlMallocLoc : Out of free space \n " ) ;
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
2008-07-30 12:58:11 +00:00
}
2001-02-23 17:55:21 +00:00
p - > mh_tag = MEMTAG ;
p - > mh_size = size ;
p - > mh_type = MALLOC_TYPE ;
p - > mh_file = file ;
p - > mh_line = line ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2003-11-29 10:47:56 +00:00
p - > mh_number = + + block ;
2001-02-23 17:55:21 +00:00
debugMemSize + = size ;
2004-11-02 14:52:23 +00:00
debugMemBlocks + + ;
2001-02-23 17:55:21 +00:00
if ( debugMemSize > debugMaxMemSize ) debugMaxMemSize = debugMemSize ;
# ifdef MEM_LIST
debugmem_list_add ( p ) ;
# endif
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
2003-11-29 10:47:56 +00:00
if ( xmlMemStopAtBlock = = p - > mh_number ) xmlMallocBreakpoint ( ) ;
2001-02-23 17:55:21 +00:00
2001-07-30 13:42:13 +00:00
ret = HDR_2_CLIENT ( p ) ;
if ( xmlMemTraceBlockAt = = ret ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2011-05-06 17:40:10 +03:00
" %p : Malloc(%lu) Ok \n " , xmlMemTraceBlockAt ,
( long unsigned ) size ) ;
2001-07-30 13:42:13 +00:00
xmlMallocBreakpoint ( ) ;
}
2001-02-23 17:55:21 +00:00
TEST_POINT
2001-07-30 13:42:13 +00:00
return ( ret ) ;
2001-02-23 17:55:21 +00:00
}
2003-04-19 00:07:51 +00:00
/**
* xmlMallocAtomicLoc :
2016-04-05 12:05:25 -07:00
* @ size : an unsigned int specifying the size in byte to allocate .
2003-04-19 00:07:51 +00:00
* @ file : the file name or NULL
* @ line : the line number
*
* a malloc ( ) equivalent , with logging of the allocation info .
*
* Returns a pointer to the allocated area or NULL in case of lack of memory .
*/
void *
xmlMallocAtomicLoc ( size_t size , const char * file , int line )
{
MEMHDR * p ;
void * ret ;
2008-07-30 12:58:11 +00:00
2022-11-25 12:21:49 +01:00
xmlInitParser ( ) ;
2003-04-19 00:07:51 +00:00
TEST_POINT
2008-07-30 12:58:11 +00:00
2016-04-05 12:05:25 -07:00
if ( size > ( MAX_SIZE_T - RESERVE_SIZE ) ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2017-06-11 17:12:21 +02:00
" xmlMallocAtomicLoc : Unsigned overflow \n " ) ;
2016-04-05 12:05:25 -07:00
return ( NULL ) ;
}
2003-04-19 00:07:51 +00:00
p = ( MEMHDR * ) malloc ( RESERVE_SIZE + size ) ;
if ( ! p ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2016-04-05 12:05:25 -07:00
" xmlMallocAtomicLoc : Out of free space \n " ) ;
2003-04-19 00:07:51 +00:00
return ( NULL ) ;
2008-07-30 12:58:11 +00:00
}
2003-04-19 00:07:51 +00:00
p - > mh_tag = MEMTAG ;
p - > mh_size = size ;
p - > mh_type = MALLOC_ATOMIC_TYPE ;
p - > mh_file = file ;
p - > mh_line = line ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2003-11-29 10:47:56 +00:00
p - > mh_number = + + block ;
2003-04-19 00:07:51 +00:00
debugMemSize + = size ;
2004-11-02 14:52:23 +00:00
debugMemBlocks + + ;
2003-04-19 00:07:51 +00:00
if ( debugMemSize > debugMaxMemSize ) debugMaxMemSize = debugMemSize ;
# ifdef MEM_LIST
debugmem_list_add ( p ) ;
# endif
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2003-04-19 00:07:51 +00:00
2003-11-29 10:47:56 +00:00
if ( xmlMemStopAtBlock = = p - > mh_number ) xmlMallocBreakpoint ( ) ;
2003-04-19 00:07:51 +00:00
ret = HDR_2_CLIENT ( p ) ;
if ( xmlMemTraceBlockAt = = ret ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2011-05-06 17:40:10 +03:00
" %p : Malloc(%lu) Ok \n " , xmlMemTraceBlockAt ,
( long unsigned ) size ) ;
2003-04-19 00:07:51 +00:00
xmlMallocBreakpoint ( ) ;
}
TEST_POINT
return ( ret ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlMemMalloc :
* @ size : an int specifying the size in byte to allocate .
*
* a malloc ( ) equivalent , with logging of the allocation info .
*
* Returns a pointer to the allocated area or NULL in case of lack of memory .
*/
void *
2001-07-17 21:38:51 +00:00
xmlMemMalloc ( size_t size )
2001-02-23 17:55:21 +00:00
{
return ( xmlMallocLoc ( size , " none " , 0 ) ) ;
}
/**
* xmlReallocLoc :
* @ ptr : the initial memory block pointer
* @ size : an int specifying the size in byte to allocate .
* @ file : the file name or NULL
* @ line : the line number
*
* a realloc ( ) equivalent , with logging of the allocation info .
*
* Returns a pointer to the allocated area or NULL in case of lack of memory .
*/
void *
2001-07-17 21:38:51 +00:00
xmlReallocLoc ( void * ptr , size_t size , const char * file , int line )
2001-02-23 17:55:21 +00:00
{
2014-10-10 12:23:09 +02:00
MEMHDR * p , * tmp ;
2001-02-23 17:55:21 +00:00
unsigned long number ;
2003-04-24 16:06:47 +00:00
if ( ptr = = NULL )
2004-01-22 22:20:31 +00:00
return ( xmlMallocLoc ( size , file , line ) ) ;
2022-11-25 12:21:49 +01:00
xmlInitParser ( ) ;
2001-02-23 17:55:21 +00:00
TEST_POINT
p = CLIENT_2_HDR ( ptr ) ;
number = p - > mh_number ;
2005-04-14 17:50:59 +00:00
if ( xmlMemStopAtBlock = = number ) xmlMallocBreakpoint ( ) ;
2001-02-23 17:55:21 +00:00
if ( p - > mh_tag ! = MEMTAG ) {
Mem_Tag_Err ( p ) ;
goto error ;
}
p - > mh_tag = ~ MEMTAG ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2001-02-23 17:55:21 +00:00
debugMemSize - = p - > mh_size ;
2004-11-02 14:52:23 +00:00
debugMemBlocks - - ;
2001-02-23 17:55:21 +00:00
# ifdef MEM_LIST
debugmem_list_delete ( p ) ;
# endif
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
2017-06-06 13:21:14 +02:00
if ( size > ( MAX_SIZE_T - RESERVE_SIZE ) ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2017-06-11 17:12:21 +02:00
" xmlReallocLoc : Unsigned overflow \n " ) ;
2017-06-06 13:21:14 +02:00
return ( NULL ) ;
}
2014-10-10 12:23:09 +02:00
tmp = ( MEMHDR * ) realloc ( p , RESERVE_SIZE + size ) ;
if ( ! tmp ) {
free ( p ) ;
2001-02-23 17:55:21 +00:00
goto error ;
}
2014-10-10 12:23:09 +02:00
p = tmp ;
2001-07-30 13:42:13 +00:00
if ( xmlMemTraceBlockAt = = ptr ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2011-05-06 17:40:10 +03:00
" %p : Realloced(%lu -> %lu) Ok \n " ,
xmlMemTraceBlockAt , ( long unsigned ) p - > mh_size ,
( long unsigned ) size ) ;
2001-07-30 13:42:13 +00:00
xmlMallocBreakpoint ( ) ;
}
2001-02-23 17:55:21 +00:00
p - > mh_tag = MEMTAG ;
p - > mh_number = number ;
p - > mh_type = REALLOC_TYPE ;
p - > mh_size = size ;
p - > mh_file = file ;
p - > mh_line = line ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2001-02-23 17:55:21 +00:00
debugMemSize + = size ;
2004-11-02 14:52:23 +00:00
debugMemBlocks + + ;
2001-02-23 17:55:21 +00:00
if ( debugMemSize > debugMaxMemSize ) debugMaxMemSize = debugMemSize ;
# ifdef MEM_LIST
debugmem_list_add ( p ) ;
# endif
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2001-02-23 17:55:21 +00:00
TEST_POINT
return ( HDR_2_CLIENT ( p ) ) ;
2008-07-30 12:58:11 +00:00
error :
2001-02-23 17:55:21 +00:00
return ( NULL ) ;
}
/**
* xmlMemRealloc :
* @ ptr : the initial memory block pointer
* @ size : an int specifying the size in byte to allocate .
*
* a realloc ( ) equivalent , with logging of the allocation info .
*
* Returns a pointer to the allocated area or NULL in case of lack of memory .
*/
void *
2001-07-17 21:38:51 +00:00
xmlMemRealloc ( void * ptr , size_t size ) {
2001-02-23 17:55:21 +00:00
return ( xmlReallocLoc ( ptr , size , " none " , 0 ) ) ;
}
/**
* xmlMemFree :
* @ ptr : the memory block pointer
*
* a free ( ) equivalent , with error checking .
*/
void
xmlMemFree ( void * ptr )
{
MEMHDR * p ;
2001-03-27 12:47:33 +00:00
char * target ;
2001-02-23 17:55:21 +00:00
2007-10-30 20:24:40 +00:00
if ( ptr = = NULL )
return ;
2001-07-30 13:42:13 +00:00
if ( ptr = = ( void * ) - 1 ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2001-07-30 13:42:13 +00:00
" trying to free pointer from freed area \n " ) ;
goto error ;
}
if ( xmlMemTraceBlockAt = = ptr ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2001-07-30 13:42:13 +00:00
" %p : Freed() \n " , xmlMemTraceBlockAt ) ;
xmlMallocBreakpoint ( ) ;
}
2001-02-23 17:55:21 +00:00
TEST_POINT
2001-03-27 12:47:33 +00:00
target = ( char * ) ptr ;
2001-02-23 17:55:21 +00:00
p = CLIENT_2_HDR ( ptr ) ;
if ( p - > mh_tag ! = MEMTAG ) {
2001-07-30 13:42:13 +00:00
Mem_Tag_Err ( p ) ;
goto error ;
2001-02-23 17:55:21 +00:00
}
2005-04-14 17:50:59 +00:00
if ( xmlMemStopAtBlock = = p - > mh_number ) xmlMallocBreakpoint ( ) ;
2001-02-23 17:55:21 +00:00
p - > mh_tag = ~ MEMTAG ;
2001-03-27 12:47:33 +00:00
memset ( target , - 1 , p - > mh_size ) ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2003-11-29 10:47:56 +00:00
debugMemSize - = p - > mh_size ;
2004-11-02 14:52:23 +00:00
debugMemBlocks - - ;
2001-02-23 17:55:21 +00:00
# ifdef MEM_LIST
debugmem_list_delete ( p ) ;
# endif
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2003-11-29 10:47:56 +00:00
2001-02-23 17:55:21 +00:00
free ( p ) ;
TEST_POINT
return ;
2008-07-30 12:58:11 +00:00
error :
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2017-10-09 13:37:42 +02:00
" xmlMemFree(%p) error \n " , ptr ) ;
2001-07-30 13:42:13 +00:00
xmlMallocBreakpoint ( ) ;
2001-02-23 17:55:21 +00:00
return ;
}
/**
* xmlMemStrdupLoc :
2002-01-22 18:15:52 +00:00
* @ str : the initial string pointer
2001-02-23 17:55:21 +00:00
* @ file : the file name or NULL
* @ line : the line number
*
* a strdup ( ) equivalent , with logging of the allocation info .
*
2002-01-01 16:50:03 +00:00
* Returns a pointer to the new string or NULL if allocation error occurred .
2001-02-23 17:55:21 +00:00
*/
char *
xmlMemStrdupLoc ( const char * str , const char * file , int line )
{
char * s ;
size_t size = strlen ( str ) + 1 ;
MEMHDR * p ;
2022-11-25 12:21:49 +01:00
xmlInitParser ( ) ;
2001-02-23 17:55:21 +00:00
TEST_POINT
2017-06-06 13:21:14 +02:00
if ( size > ( MAX_SIZE_T - RESERVE_SIZE ) ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2017-06-11 17:12:21 +02:00
" xmlMemStrdupLoc : Unsigned overflow \n " ) ;
2017-06-06 13:21:14 +02:00
return ( NULL ) ;
}
2001-02-23 17:55:21 +00:00
p = ( MEMHDR * ) malloc ( RESERVE_SIZE + size ) ;
if ( ! p ) {
goto error ;
}
p - > mh_tag = MEMTAG ;
p - > mh_size = size ;
p - > mh_type = STRDUP_TYPE ;
p - > mh_file = file ;
p - > mh_line = line ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2003-11-29 10:47:56 +00:00
p - > mh_number = + + block ;
2001-02-23 17:55:21 +00:00
debugMemSize + = size ;
2004-11-02 14:52:23 +00:00
debugMemBlocks + + ;
2001-02-23 17:55:21 +00:00
if ( debugMemSize > debugMaxMemSize ) debugMaxMemSize = debugMemSize ;
# ifdef MEM_LIST
debugmem_list_add ( p ) ;
# endif
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
2001-02-23 17:55:21 +00:00
s = ( char * ) HDR_2_CLIENT ( p ) ;
2008-07-30 12:58:11 +00:00
2003-11-29 10:47:56 +00:00
if ( xmlMemStopAtBlock = = p - > mh_number ) xmlMallocBreakpoint ( ) ;
2001-02-23 17:55:21 +00:00
2014-07-26 21:14:53 +08:00
strcpy ( s , str ) ;
2008-07-30 12:58:11 +00:00
2001-02-23 17:55:21 +00:00
TEST_POINT
2001-07-30 13:42:13 +00:00
if ( xmlMemTraceBlockAt = = s ) {
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2001-07-30 13:42:13 +00:00
" %p : Strdup() Ok \n " , xmlMemTraceBlockAt ) ;
xmlMallocBreakpoint ( ) ;
}
2001-02-23 17:55:21 +00:00
return ( s ) ;
error :
return ( NULL ) ;
}
/**
* xmlMemoryStrdup :
2002-12-10 15:19:08 +00:00
* @ str : the initial string pointer
2001-02-23 17:55:21 +00:00
*
* a strdup ( ) equivalent , with logging of the allocation info .
*
2002-01-01 16:50:03 +00:00
* Returns a pointer to the new string or NULL if allocation error occurred .
2001-02-23 17:55:21 +00:00
*/
char *
xmlMemoryStrdup ( const char * str ) {
return ( xmlMemStrdupLoc ( str , " none " , 0 ) ) ;
}
2023-03-14 13:02:36 +01:00
/**
* xmlMemSize :
* @ ptr : pointer to the memory allocation
*
* Returns the size of a memory allocation .
*/
size_t
xmlMemSize ( void * ptr ) {
MEMHDR * p ;
if ( ptr = = NULL )
return ( 0 ) ;
p = CLIENT_2_HDR ( ptr ) ;
if ( p - > mh_tag ! = MEMTAG )
return ( 0 ) ;
return ( p - > mh_size ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlMemUsed :
*
2002-12-11 14:23:49 +00:00
* Provides the amount of memory currently allocated
2001-02-23 17:55:21 +00:00
*
* Returns an int representing the amount of memory allocated .
*/
int
xmlMemUsed ( void ) {
2022-11-25 17:39:01 +01:00
return ( debugMemSize ) ;
2001-02-23 17:55:21 +00:00
}
2004-11-02 14:52:23 +00:00
/**
* xmlMemBlocks :
*
* Provides the number of memory areas currently allocated
*
* Returns an int representing the number of blocks
*/
int
xmlMemBlocks ( void ) {
2015-04-13 16:32:14 +08:00
int res ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2015-04-13 16:32:14 +08:00
res = debugMemBlocks ;
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2015-04-13 16:32:14 +08:00
return ( res ) ;
2004-11-02 14:52:23 +00:00
}
2008-07-30 12:58:11 +00:00
/**
* xmlMemDisplayLast :
* @ fp : a FILE descriptor used as the output file , if NULL , the result is
* written to the file . memorylist
* @ nbBytes : the amount of memory to dump
*
* the last nbBytes of memory allocated and not freed , useful for dumping
* the memory left allocated between two places at runtime .
*/
void
xmlMemDisplayLast ( FILE * fp , long nbBytes )
{
# ifdef MEM_LIST
MEMHDR * p ;
unsigned idx ;
int nb = 0 ;
# endif
FILE * old_fp = fp ;
if ( nbBytes < = 0 )
return ;
if ( fp = = NULL ) {
fp = fopen ( " .memorylist " , " w " ) ;
if ( fp = = NULL )
return ;
}
# ifdef MEM_LIST
fprintf ( fp , " Last %li MEMORY ALLOCATED : %lu, MAX was %lu \n " ,
nbBytes , debugMemSize , debugMaxMemSize ) ;
fprintf ( fp , " BLOCK NUMBER SIZE TYPE \n " ) ;
idx = 0 ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
p = memlist ;
while ( ( p ) & & ( nbBytes > 0 ) ) {
fprintf ( fp , " %-5u %6lu %6lu " , idx + + , p - > mh_number ,
( unsigned long ) p - > mh_size ) ;
switch ( p - > mh_type ) {
case STRDUP_TYPE : fprintf ( fp , " strdup() in " ) ; break ;
case MALLOC_TYPE : fprintf ( fp , " malloc() in " ) ; break ;
case REALLOC_TYPE : fprintf ( fp , " realloc() in " ) ; break ;
case MALLOC_ATOMIC_TYPE : fprintf ( fp , " atomicmalloc() in " ) ; break ;
case REALLOC_ATOMIC_TYPE : fprintf ( fp , " atomicrealloc() in " ) ; break ;
default :
fprintf ( fp , " Unknown memory block, may be corrupted " ) ;
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
if ( old_fp = = NULL )
fclose ( fp ) ;
return ;
}
if ( p - > mh_file ! = NULL ) fprintf ( fp , " %s(%u) " , p - > mh_file , p - > mh_line ) ;
if ( p - > mh_tag ! = MEMTAG )
fprintf ( fp , " INVALID " ) ;
nb + + ;
fprintf ( fp , " \n " ) ;
nbBytes - = ( unsigned long ) p - > mh_size ;
p = p - > mh_next ;
}
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
# else
fprintf ( fp , " Memory list not compiled (MEM_LIST not defined !) \n " ) ;
# endif
if ( old_fp = = NULL )
fclose ( fp ) ;
}
2001-02-23 17:55:21 +00:00
/**
* xmlMemDisplay :
* @ fp : a FILE descriptor used as the output file , if NULL , the result is
* written to the file . memorylist
*
* show in - extenso the memory blocks allocated
*/
void
xmlMemDisplay ( FILE * fp )
{
# ifdef MEM_LIST
MEMHDR * p ;
2002-02-13 21:14:46 +00:00
unsigned idx ;
2001-07-30 13:42:13 +00:00
int nb = 0 ;
2001-02-23 17:55:21 +00:00
time_t currentTime ;
char buf [ 500 ] ;
struct tm * tstruct ;
2005-05-08 11:39:56 +00:00
# endif
FILE * old_fp = fp ;
if ( fp = = NULL ) {
fp = fopen ( " .memorylist " , " w " ) ;
if ( fp = = NULL )
return ;
}
2001-02-23 17:55:21 +00:00
2005-05-08 11:39:56 +00:00
# ifdef MEM_LIST
2001-02-23 17:55:21 +00:00
currentTime = time ( NULL ) ;
tstruct = localtime ( & currentTime ) ;
2001-03-24 17:00:36 +00:00
strftime ( buf , sizeof ( buf ) - 1 , " %I:%M:%S %p " , tstruct ) ;
2001-02-23 17:55:21 +00:00
fprintf ( fp , " %s \n \n " , buf ) ;
2008-07-30 12:58:11 +00:00
2001-02-23 17:55:21 +00:00
fprintf ( fp , " MEMORY ALLOCATED : %lu, MAX was %lu \n " ,
debugMemSize , debugMaxMemSize ) ;
fprintf ( fp , " BLOCK NUMBER SIZE TYPE \n " ) ;
idx = 0 ;
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2001-02-23 17:55:21 +00:00
p = memlist ;
while ( p ) {
2002-02-13 21:14:46 +00:00
fprintf ( fp , " %-5u %6lu %6lu " , idx + + , p - > mh_number ,
( unsigned long ) p - > mh_size ) ;
2001-02-23 17:55:21 +00:00
switch ( p - > mh_type ) {
case STRDUP_TYPE : fprintf ( fp , " strdup() in " ) ; break ;
case MALLOC_TYPE : fprintf ( fp , " malloc() in " ) ; break ;
2004-07-02 12:23:44 +00:00
case REALLOC_TYPE : fprintf ( fp , " realloc() in " ) ; break ;
2003-04-19 00:07:51 +00:00
case MALLOC_ATOMIC_TYPE : fprintf ( fp , " atomicmalloc() in " ) ; break ;
2004-07-02 12:23:44 +00:00
case REALLOC_ATOMIC_TYPE : fprintf ( fp , " atomicrealloc() in " ) ; break ;
default :
2004-09-18 04:52:08 +00:00
fprintf ( fp , " Unknown memory block, may be corrupted " ) ;
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2005-05-08 11:39:56 +00:00
if ( old_fp = = NULL )
fclose ( fp ) ;
2004-07-02 12:23:44 +00:00
return ;
2001-02-23 17:55:21 +00:00
}
2004-09-18 04:52:08 +00:00
if ( p - > mh_file ! = NULL ) fprintf ( fp , " %s(%u) " , p - > mh_file , p - > mh_line ) ;
2001-02-23 17:55:21 +00:00
if ( p - > mh_tag ! = MEMTAG )
fprintf ( fp , " INVALID " ) ;
2001-07-30 13:42:13 +00:00
nb + + ;
2001-02-23 17:55:21 +00:00
fprintf ( fp , " \n " ) ;
p = p - > mh_next ;
}
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2001-02-23 17:55:21 +00:00
# else
fprintf ( fp , " Memory list not compiled (MEM_LIST not defined !) \n " ) ;
# endif
2005-05-08 11:39:56 +00:00
if ( old_fp = = NULL )
fclose ( fp ) ;
2001-02-23 17:55:21 +00:00
}
# ifdef MEM_LIST
2002-12-10 15:19:08 +00:00
static void debugmem_list_add ( MEMHDR * p )
2001-02-23 17:55:21 +00:00
{
p - > mh_next = memlist ;
p - > mh_prev = NULL ;
if ( memlist ) memlist - > mh_prev = p ;
memlist = p ;
}
2002-12-10 15:19:08 +00:00
static void debugmem_list_delete ( MEMHDR * p )
2001-02-23 17:55:21 +00:00
{
if ( p - > mh_next )
p - > mh_next - > mh_prev = p - > mh_prev ;
if ( p - > mh_prev )
p - > mh_prev - > mh_next = p - > mh_next ;
else memlist = p - > mh_next ;
}
# endif
/*
2002-12-10 15:19:08 +00:00
* debugmem_tag_error :
*
* internal error function .
2001-02-23 17:55:21 +00:00
*/
2008-07-30 12:58:11 +00:00
2002-12-10 15:19:08 +00:00
static void debugmem_tag_error ( void * p )
2001-02-23 17:55:21 +00:00
{
2023-12-18 21:32:49 +01:00
fprintf ( stderr ,
2001-02-23 17:55:21 +00:00
" Memory tag error occurs :%p \n \t bye \n " , p ) ;
# ifdef MEM_LIST
if ( stderr )
xmlMemDisplay ( stderr ) ;
# endif
}
2003-09-29 13:20:24 +00:00
# ifdef MEM_LIST
2001-10-11 22:55:55 +00:00
static FILE * xmlMemoryDumpFile = NULL ;
2003-09-29 13:20:24 +00:00
# endif
2001-02-23 17:55:21 +00:00
2003-09-29 09:22:39 +00:00
/**
* xmlMemShow :
* @ fp : a FILE descriptor used as the output file
* @ nr : number of entries to dump
*
* show a show display of the memory allocated , and dump
* the @ nr last allocated areas which were not freed
*/
void
2003-09-29 10:55:05 +00:00
xmlMemShow ( FILE * fp , int nr ATTRIBUTE_UNUSED )
2003-09-29 09:22:39 +00:00
{
# ifdef MEM_LIST
MEMHDR * p ;
# endif
if ( fp ! = NULL )
fprintf ( fp , " MEMORY ALLOCATED : %lu, MAX was %lu \n " ,
debugMemSize , debugMaxMemSize ) ;
# ifdef MEM_LIST
2022-11-24 20:54:18 +01:00
xmlMutexLock ( & xmlMemMutex ) ;
2003-09-29 09:22:39 +00:00
if ( nr > 0 ) {
fprintf ( fp , " NUMBER SIZE TYPE WHERE \n " ) ;
p = memlist ;
while ( ( p ) & & nr > 0 ) {
fprintf ( fp , " %6lu %6lu " , p - > mh_number , ( unsigned long ) p - > mh_size ) ;
switch ( p - > mh_type ) {
case STRDUP_TYPE : fprintf ( fp , " strdup() in " ) ; break ;
case MALLOC_TYPE : fprintf ( fp , " malloc() in " ) ; break ;
case MALLOC_ATOMIC_TYPE : fprintf ( fp , " atomicmalloc() in " ) ; break ;
case REALLOC_TYPE : fprintf ( fp , " realloc() in " ) ; break ;
case REALLOC_ATOMIC_TYPE : fprintf ( fp , " atomicrealloc() in " ) ; break ;
default : fprintf ( fp , " ??? in " ) ; break ;
}
if ( p - > mh_file ! = NULL )
2004-09-18 04:52:08 +00:00
fprintf ( fp , " %s(%u) " , p - > mh_file , p - > mh_line ) ;
2003-09-29 09:22:39 +00:00
if ( p - > mh_tag ! = MEMTAG )
fprintf ( fp , " INVALID " ) ;
fprintf ( fp , " \n " ) ;
nr - - ;
p = p - > mh_next ;
}
}
2022-11-24 20:54:18 +01:00
xmlMutexUnlock ( & xmlMemMutex ) ;
2008-07-30 12:58:11 +00:00
# endif /* MEM_LIST */
2003-09-29 09:22:39 +00:00
}
2001-02-23 17:55:21 +00:00
/**
* xmlMemoryDump :
*
* Dump in - extenso the memory blocks allocated to the file . memorylist
*/
void
xmlMemoryDump ( void )
{
2003-09-29 10:55:05 +00:00
# ifdef MEM_LIST
2001-02-23 17:55:21 +00:00
FILE * dump ;
2002-03-18 18:36:20 +00:00
if ( debugMaxMemSize = = 0 )
return ;
2001-02-23 17:55:21 +00:00
dump = fopen ( " .memdump " , " w " ) ;
2001-11-22 18:20:37 +00:00
if ( dump = = NULL )
xmlMemoryDumpFile = stderr ;
2001-02-23 17:55:21 +00:00
else xmlMemoryDumpFile = dump ;
xmlMemDisplay ( xmlMemoryDumpFile ) ;
if ( dump ! = NULL ) fclose ( dump ) ;
2003-09-29 10:55:05 +00:00
# endif /* MEM_LIST */
2001-02-23 17:55:21 +00:00
}
/****************************************************************
* *
* Initialization Routines *
* *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/**
* xmlInitMemory :
*
2022-11-24 20:52:57 +01:00
* DEPRECATED : Alias for xmlInitParser .
*/
int
xmlInitMemory ( void ) {
xmlInitParser ( ) ;
return ( 0 ) ;
}
/**
* xmlInitMemoryInternal :
2022-03-06 13:55:48 +01:00
*
2001-02-23 17:55:21 +00:00
* Initialize the memory layer .
*
* Returns 0 on success
*/
2022-11-24 20:52:57 +01:00
void
xmlInitMemoryInternal ( void ) {
2004-07-13 15:25:08 +00:00
char * breakpoint ;
2022-11-24 20:54:18 +01:00
xmlInitMutex ( & xmlMemMutex ) ;
2001-02-23 17:55:21 +00:00
breakpoint = getenv ( " XML_MEM_BREAKPOINT " ) ;
if ( breakpoint ! = NULL ) {
2003-11-29 10:47:56 +00:00
sscanf ( breakpoint , " %ud " , & xmlMemStopAtBlock ) ;
2001-02-23 17:55:21 +00:00
}
2001-07-30 13:42:13 +00:00
breakpoint = getenv ( " XML_MEM_TRACE " ) ;
if ( breakpoint ! = NULL ) {
sscanf ( breakpoint , " %p " , & xmlMemTraceBlockAt ) ;
}
2008-07-30 12:58:11 +00:00
2001-02-23 17:55:21 +00:00
}
2003-12-30 08:30:19 +00:00
/**
* xmlCleanupMemory :
*
2022-11-24 20:52:57 +01:00
* DEPRECATED : This function is a no - op . Call xmlCleanupParser
2022-03-06 13:55:48 +01:00
* to free global state but see the warnings there . xmlCleanupParser
* should be only called once at program exit . In most cases , you don ' t
* have call cleanup functions at all .
2022-11-24 20:52:57 +01:00
*/
void
xmlCleanupMemory ( void ) {
}
/**
* xmlCleanupMemoryInternal :
2022-03-06 13:55:48 +01:00
*
2004-12-10 10:26:42 +00:00
* Free up all the memory allocated by the library for its own
* use . This should not be called by user level code .
2003-12-30 08:30:19 +00:00
*/
void
2022-11-24 20:52:57 +01:00
xmlCleanupMemoryInternal ( void ) {
2023-09-21 23:29:18 +02:00
/*
* Don ' t clean up mutex on Windows . Global state destructors can call
* malloc functions after xmlCleanupParser was called . If memory
* debugging is enabled , xmlMemMutex can be used after cleanup .
*
* See python / tests / thread2 . py
*/
# if !defined(LIBXML_THREAD_ENABLED) || !defined(_WIN32)
2022-11-24 20:54:18 +01:00
xmlCleanupMutex ( & xmlMemMutex ) ;
2023-09-21 23:29:18 +02:00
# endif
2003-12-30 08:30:19 +00:00
}
2001-02-23 17:55:21 +00:00
/**
* xmlMemSetup :
* @ freeFunc : the free ( ) function to use
* @ mallocFunc : the malloc ( ) function to use
* @ reallocFunc : the realloc ( ) function to use
* @ strdupFunc : the strdup ( ) function to use
*
* Override the default memory access functions with a new set
* This has to be called before any other libxml routines !
*
* Should this be blocked if there was already some allocations
* done ?
*
* Returns 0 on success
*/
int
xmlMemSetup ( xmlFreeFunc freeFunc , xmlMallocFunc mallocFunc ,
xmlReallocFunc reallocFunc , xmlStrdupFunc strdupFunc ) {
if ( freeFunc = = NULL )
return ( - 1 ) ;
if ( mallocFunc = = NULL )
return ( - 1 ) ;
if ( reallocFunc = = NULL )
return ( - 1 ) ;
if ( strdupFunc = = NULL )
return ( - 1 ) ;
xmlFree = freeFunc ;
xmlMalloc = mallocFunc ;
2003-04-19 00:07:51 +00:00
xmlMallocAtomic = mallocFunc ;
2001-02-23 17:55:21 +00:00
xmlRealloc = reallocFunc ;
xmlMemStrdup = strdupFunc ;
return ( 0 ) ;
}
/**
* xmlMemGet :
2002-12-11 14:23:49 +00:00
* @ freeFunc : place to save the free ( ) function in use
* @ mallocFunc : place to save the malloc ( ) function in use
* @ reallocFunc : place to save the realloc ( ) function in use
* @ strdupFunc : place to save the strdup ( ) function in use
2001-02-23 17:55:21 +00:00
*
2002-12-11 14:23:49 +00:00
* Provides the memory access functions set currently in use
2001-02-23 17:55:21 +00:00
*
* Returns 0 on success
*/
int
xmlMemGet ( xmlFreeFunc * freeFunc , xmlMallocFunc * mallocFunc ,
xmlReallocFunc * reallocFunc , xmlStrdupFunc * strdupFunc ) {
if ( freeFunc ! = NULL ) * freeFunc = xmlFree ;
if ( mallocFunc ! = NULL ) * mallocFunc = xmlMalloc ;
if ( reallocFunc ! = NULL ) * reallocFunc = xmlRealloc ;
if ( strdupFunc ! = NULL ) * strdupFunc = xmlMemStrdup ;
return ( 0 ) ;
}
2003-04-19 00:07:51 +00:00
/**
* xmlGcMemSetup :
* @ freeFunc : the free ( ) function to use
* @ mallocFunc : the malloc ( ) function to use
* @ mallocAtomicFunc : the malloc ( ) function to use for atomic allocations
* @ reallocFunc : the realloc ( ) function to use
* @ strdupFunc : the strdup ( ) function to use
*
* Override the default memory access functions with a new set
* This has to be called before any other libxml routines !
* The mallocAtomicFunc is specialized for atomic block
* allocations ( i . e . of areas useful for garbage collected memory allocators
*
* Should this be blocked if there was already some allocations
* done ?
*
* Returns 0 on success
*/
int
xmlGcMemSetup ( xmlFreeFunc freeFunc , xmlMallocFunc mallocFunc ,
xmlMallocFunc mallocAtomicFunc , xmlReallocFunc reallocFunc ,
xmlStrdupFunc strdupFunc ) {
if ( freeFunc = = NULL )
return ( - 1 ) ;
if ( mallocFunc = = NULL )
return ( - 1 ) ;
if ( mallocAtomicFunc = = NULL )
return ( - 1 ) ;
if ( reallocFunc = = NULL )
return ( - 1 ) ;
if ( strdupFunc = = NULL )
return ( - 1 ) ;
xmlFree = freeFunc ;
xmlMalloc = mallocFunc ;
xmlMallocAtomic = mallocAtomicFunc ;
xmlRealloc = reallocFunc ;
xmlMemStrdup = strdupFunc ;
return ( 0 ) ;
}
/**
* xmlGcMemGet :
* @ freeFunc : place to save the free ( ) function in use
* @ mallocFunc : place to save the malloc ( ) function in use
* @ mallocAtomicFunc : place to save the atomic malloc ( ) function in use
* @ reallocFunc : place to save the realloc ( ) function in use
* @ strdupFunc : place to save the strdup ( ) function in use
*
* Provides the memory access functions set currently in use
* The mallocAtomicFunc is specialized for atomic block
* allocations ( i . e . of areas useful for garbage collected memory allocators
*
* Returns 0 on success
*/
int
xmlGcMemGet ( xmlFreeFunc * freeFunc , xmlMallocFunc * mallocFunc ,
xmlMallocFunc * mallocAtomicFunc , xmlReallocFunc * reallocFunc ,
xmlStrdupFunc * strdupFunc ) {
if ( freeFunc ! = NULL ) * freeFunc = xmlFree ;
if ( mallocFunc ! = NULL ) * mallocFunc = xmlMalloc ;
if ( mallocAtomicFunc ! = NULL ) * mallocAtomicFunc = xmlMallocAtomic ;
if ( reallocFunc ! = NULL ) * reallocFunc = xmlRealloc ;
if ( strdupFunc ! = NULL ) * strdupFunc = xmlMemStrdup ;
return ( 0 ) ;
}