2001-08-21 16:56:08 +04:00
/*
2001-10-03 15:06:31 +04:00
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
2001-08-21 16:56:08 +04:00
*
2001-10-31 15:47:01 +03:00
* This file is released under the LGPL .
2001-08-21 16:56:08 +04:00
*/
2002-11-18 17:01:16 +03:00
# include "lib.h"
# include "lvm-types.h"
2002-12-20 02:25:55 +03:00
# include "dbg_malloc.h"
2002-11-18 17:01:16 +03:00
2001-08-21 16:56:08 +04:00
# include <stdarg.h>
struct memblock {
2002-04-24 22:20:51 +04:00
struct memblock * prev , * next ; /* All allocated blocks are linked */
size_t length ; /* Size of the requested block */
int id ; /* Index of the block */
const char * file ; /* File that allocated */
int line ; /* Line that allocated */
void * magic ; /* Address of this block */
2001-08-21 16:56:08 +04:00
} ;
static struct {
unsigned int blocks , mblocks ;
unsigned int bytes , mbytes ;
2002-04-24 22:20:51 +04:00
} _mem_stats = {
0 , 0 , 0 , 0 } ;
2001-08-21 16:56:08 +04:00
static struct memblock * _head = 0 ;
static struct memblock * _tail = 0 ;
void * malloc_aux ( size_t s , const char * file , int line )
{
struct memblock * nb ;
size_t tsize = s + sizeof ( * nb ) + sizeof ( unsigned long ) ;
2003-11-14 02:54:02 +03:00
if ( s > 50000000 ) {
log_error ( " Huge memory allocation (size % " PRIuPTR
" ) rejected - bug? " , s ) ;
return 0 ;
}
2001-08-21 16:56:08 +04:00
if ( ! ( nb = malloc ( tsize ) ) ) {
2001-11-16 14:37:45 +03:00
log_error ( " couldn't allocate any memory, size = % " PRIuPTR , s ) ;
2001-08-21 16:56:08 +04:00
return 0 ;
}
/* set up the file and line info */
nb - > file = file ;
nb - > line = line ;
# ifdef BOUNDS_CHECK
bounds_check ( ) ;
# endif
/* setup fields */
nb - > magic = nb + 1 ;
nb - > length = s ;
nb - > id = + + _mem_stats . blocks ;
nb - > next = 0 ;
nb - > prev = _tail ;
/* link to tail of the list */
if ( ! _head )
_head = _tail = nb ;
else {
_tail - > next = nb ;
_tail = nb ;
}
/* stomp a pretty pattern across the new memory
and fill in the boundary bytes */
{
char * ptr = ( char * ) ( nb + 1 ) ;
2002-12-20 02:25:55 +03:00
size_t i ;
2001-08-21 16:56:08 +04:00
for ( i = 0 ; i < s ; i + + )
* ptr + + = i & 0x1 ? ( char ) 0xba : ( char ) 0xbe ;
for ( i = 0 ; i < sizeof ( unsigned long ) ; i + + )
* ptr + + = ( char ) nb - > id ;
}
if ( _mem_stats . blocks > _mem_stats . mblocks )
_mem_stats . mblocks = _mem_stats . blocks ;
_mem_stats . bytes + = s ;
if ( _mem_stats . bytes > _mem_stats . mbytes )
_mem_stats . mbytes = _mem_stats . bytes ;
return nb + 1 ;
}
void free_aux ( void * p )
{
char * ptr ;
2002-12-20 02:25:55 +03:00
size_t i ;
2001-08-21 16:56:08 +04:00
struct memblock * mb = ( ( struct memblock * ) p ) - 1 ;
if ( ! p )
return ;
# ifdef BOUNDS_CHECK
bounds_check ( ) ;
# endif
/* sanity check */
assert ( mb - > magic = = p ) ;
/* check data at the far boundary */
ptr = ( ( char * ) mb ) + sizeof ( struct memblock ) + mb - > length ;
for ( i = 0 ; i < sizeof ( unsigned long ) ; i + + )
2002-04-24 22:20:51 +04:00
if ( * ptr + + ! = ( char ) mb - > id )
2001-08-21 16:56:08 +04:00
assert ( ! " Damage at far end of block " ) ;
/* have we freed this before ? */
assert ( mb - > id ! = 0 ) ;
mb - > id = 0 ;
/* stomp a different pattern across the memory */
ptr = ( ( char * ) mb ) + sizeof ( struct memblock ) ;
for ( i = 0 ; i < mb - > length ; i + + )
* ptr + + = i & 1 ? ( char ) 0xde : ( char ) 0xad ;
/* unlink */
if ( mb - > prev )
mb - > prev - > next = mb - > next ;
else
_head = mb - > next ;
if ( mb - > next )
mb - > next - > prev = mb - > prev ;
else
_tail = mb - > prev ;
assert ( _mem_stats . blocks ) ;
_mem_stats . blocks - - ;
_mem_stats . bytes - = mb - > length ;
/* free the memory */
free ( mb ) ;
}
void * realloc_aux ( void * p , unsigned int s , const char * file , int line )
{
void * r ;
struct memblock * mb = ( ( struct memblock * ) p ) - 1 ;
r = malloc_aux ( s , file , line ) ;
if ( p ) {
memcpy ( r , p , mb - > length ) ;
free_aux ( p ) ;
}
return r ;
}
# ifdef DEBUG_MEM
int dump_memory ( void )
{
unsigned long tot = 0 ;
struct memblock * mb ;
2004-03-27 00:09:44 +03:00
char str [ 32 ] ;
size_t c ;
2001-08-21 16:56:08 +04:00
if ( _head )
2001-10-16 20:25:28 +04:00
log_very_verbose ( " You have a memory leak: " ) ;
2001-08-21 16:56:08 +04:00
for ( mb = _head ; mb ; mb = mb - > next ) {
2004-03-27 00:09:44 +03:00
for ( c = 0 ; c < sizeof ( str ) - 1 ; c + + ) {
if ( c > = mb - > length )
str [ c ] = ' ' ;
else if ( * ( char * ) ( mb - > magic + c ) < ' ' )
str [ c ] = ' ? ' ;
else
str [ c ] = * ( char * ) ( mb - > magic + c ) ;
}
str [ sizeof ( str ) - 1 ] = ' \0 ' ;
2002-04-24 22:20:51 +04:00
print_log ( _LOG_INFO , mb - > file , mb - > line ,
2004-03-27 00:09:44 +03:00
" block %d at %p, size % " PRIdPTR " \t [%s] " ,
mb - > id , mb - > magic , mb - > length , str ) ;
2001-08-21 16:56:08 +04:00
tot + = mb - > length ;
}
if ( _head )
2001-10-16 20:25:28 +04:00
log_very_verbose ( " %ld bytes leaked in total " , tot ) ;
2001-08-21 16:56:08 +04:00
return 1 ;
}
void bounds_check ( void )
{
struct memblock * mb = _head ;
while ( mb ) {
2002-12-20 02:25:55 +03:00
size_t i ;
2001-08-21 16:56:08 +04:00
char * ptr = ( ( char * ) ( mb + 1 ) ) + mb - > length ;
for ( i = 0 ; i < sizeof ( unsigned long ) ; i + + )
if ( * ptr + + ! = ( char ) mb - > id )
assert ( ! " Memory smash " ) ;
mb = mb - > next ;
}
}
# endif
/*
* Local variables :
* c - file - style : " linux "
* End :
*/