2001-11-28 12:13:00 +03:00
/*
* Copyright ( C ) 2001 Sistina Software ( UK ) Limited .
*
* This file is released under the LGPL .
*/
# include "pool.h"
# include "dbg_malloc.h"
# include "log.h"
# include <assert.h>
struct block {
struct block * next ;
2001-12-12 17:25:20 +03:00
size_t size ;
2002-01-28 12:16:09 +03:00
void * data ;
2001-11-28 12:13:00 +03:00
} ;
struct pool {
2001-12-20 15:27:41 +03:00
int begun ;
struct block * object ;
2001-11-28 12:13:00 +03:00
struct block * blocks ;
struct block * tail ;
} ;
/* by default things come out aligned for doubles */
# define DEFAULT_ALIGNMENT __alignof__ (double)
struct pool * pool_create ( size_t chunk_hint )
{
struct pool * mem = dbg_malloc ( sizeof ( * mem ) ) ;
if ( ! mem ) {
2002-01-10 19:47:25 +03:00
log_error ( " Couldn't create memory pool (size %u) " ,
sizeof ( * mem ) ) ;
2001-11-28 12:13:00 +03:00
return NULL ;
}
2002-01-15 20:37:23 +03:00
mem - > begun = 0 ;
2001-12-12 17:25:20 +03:00
mem - > object = 0 ;
2001-11-28 12:13:00 +03:00
mem - > blocks = mem - > tail = NULL ;
return mem ;
}
static void _free_blocks ( struct block * b )
{
struct block * n ;
while ( b ) {
n = b - > next ;
2002-01-28 12:16:09 +03:00
dbg_free ( b - > data ) ;
2001-11-28 12:13:00 +03:00
dbg_free ( b ) ;
b = n ;
}
}
void pool_destroy ( struct pool * p )
{
_free_blocks ( p - > blocks ) ;
dbg_free ( p ) ;
}
void * pool_alloc ( struct pool * p , size_t s )
{
return pool_alloc_aligned ( p , s , DEFAULT_ALIGNMENT ) ;
}
2001-12-12 17:25:20 +03:00
static void _append_block ( struct pool * p , struct block * b )
{
if ( p - > tail ) {
p - > tail - > next = b ;
p - > tail = b ;
} else
p - > blocks = p - > tail = b ;
}
2001-12-20 15:27:41 +03:00
static struct block * _new_block ( size_t s , unsigned alignment )
2001-11-28 12:13:00 +03:00
{
2002-01-28 12:16:09 +03:00
static char * _oom = " Out of memory " ;
2001-11-28 12:13:00 +03:00
/* FIXME: I'm currently ignoring the alignment arg. */
size_t len = sizeof ( struct block ) + s ;
struct block * b = dbg_malloc ( len ) ;
2002-01-28 12:16:09 +03:00
/*
* Too lazy to implement alignment for debug version , and
* I don ' t think LVM will use anything but default
* align .
*/
assert ( alignment = = DEFAULT_ALIGNMENT ) ;
2001-11-28 12:13:00 +03:00
if ( ! b ) {
2002-01-28 12:16:09 +03:00
log_err ( _oom ) ;
return NULL ;
}
if ( ! ( b - > data = dbg_malloc ( s ) ) ) {
log_err ( _oom ) ;
dbg_free ( b ) ;
2001-11-28 12:13:00 +03:00
return NULL ;
}
b - > next = NULL ;
2001-12-12 17:25:20 +03:00
b - > size = s ;
2001-11-28 12:13:00 +03:00
2001-12-20 15:27:41 +03:00
return b ;
}
void * pool_alloc_aligned ( struct pool * p , size_t s , unsigned alignment )
{
struct block * b = _new_block ( s , alignment ) ;
if ( ! b )
return NULL ;
2001-12-12 17:25:20 +03:00
_append_block ( p , b ) ;
2002-01-28 12:16:09 +03:00
return b - > data ;
2001-11-28 12:13:00 +03:00
}
void pool_empty ( struct pool * p )
{
_free_blocks ( p - > blocks ) ;
p - > blocks = p - > tail = NULL ;
}
void pool_free ( struct pool * p , void * ptr )
{
struct block * b , * prev = NULL ;
for ( b = p - > blocks ; b ; b = b - > next ) {
2002-01-28 12:16:09 +03:00
if ( b - > data = = ptr )
2001-11-28 12:13:00 +03:00
break ;
2001-11-28 17:58:33 +03:00
prev = b ;
2001-11-28 12:13:00 +03:00
}
/*
* If this fires then you tried to free a
* pointer that either wasn ' t from this
* pool , or isn ' t the start of a block .
*/
assert ( b ) ;
_free_blocks ( b ) ;
if ( prev ) {
p - > tail = prev ;
prev - > next = NULL ;
} else
p - > blocks = p - > tail = NULL ;
}
2001-12-20 15:27:41 +03:00
int pool_begin_object ( struct pool * p , size_t init_size )
2001-11-28 12:13:00 +03:00
{
2001-12-20 15:27:41 +03:00
assert ( ! p - > begun ) ;
p - > begun = 1 ;
return 1 ;
2001-11-28 12:13:00 +03:00
}
2001-12-20 15:27:41 +03:00
int pool_grow_object ( struct pool * p , const void * buffer , size_t delta )
2001-11-28 12:13:00 +03:00
{
2001-12-20 15:27:41 +03:00
struct block * new ;
2002-01-15 20:37:23 +03:00
size_t size = delta ;
2001-12-12 17:25:20 +03:00
2001-12-20 15:27:41 +03:00
assert ( p - > begun ) ;
2001-12-12 17:25:20 +03:00
2002-01-15 20:37:23 +03:00
if ( p - > object )
size + = p - > object - > size ;
if ( ! ( new = _new_block ( size , DEFAULT_ALIGNMENT ) ) ) {
2001-12-20 15:27:41 +03:00
log_err ( " Couldn't extend object. " ) ;
return 0 ;
}
2001-12-12 17:25:20 +03:00
2001-12-20 15:27:41 +03:00
if ( p - > object ) {
2002-01-28 12:16:09 +03:00
memcpy ( new - > data , p - > object - > data , p - > object - > size ) ;
2001-12-20 15:27:41 +03:00
dbg_free ( p - > object ) ;
}
p - > object = new ;
2001-12-12 17:25:20 +03:00
2001-12-20 15:27:41 +03:00
return 1 ;
2001-11-28 12:13:00 +03:00
}
void * pool_end_object ( struct pool * p )
{
2001-12-20 15:27:41 +03:00
assert ( p - > begun ) ;
_append_block ( p , p - > object ) ;
p - > begun = 0 ;
p - > object = NULL ;
2002-01-28 12:16:09 +03:00
return p - > tail - > data ;
2001-11-28 12:13:00 +03:00
}
void pool_abandon_object ( struct pool * p )
{
2001-12-20 15:27:41 +03:00
assert ( p - > begun ) ;
dbg_free ( p - > object ) ;
p - > begun = 0 ;
p - > object = NULL ;
2001-11-28 12:13:00 +03:00
}
char * pool_strdup ( struct pool * p , const char * str )
{
char * ret = pool_alloc ( p , strlen ( str ) + 1 ) ;
if ( ret )
strcpy ( ret , str ) ;
return ret ;
}