Refactor the malloc() etc code to reduce code duplication.

Introduce a t_malloc_fatal flag that makes unintentional allocation
failures fatal.  This reduces the need for error handling in tests.

Enable that flag in t_main().  Test programs that don't want it can
override it in t_prepare().
This commit is contained in:
Dag-Erling Smørgrav 2014-08-03 00:58:13 +00:00 committed by des
parent 5875ade2ed
commit 64a2da2b84
3 changed files with 75 additions and 34 deletions

6
t/t.h
View file

@ -111,4 +111,10 @@ t_compare_num(ptr, void *);
extern const uint8_t t_zero[256]; extern const uint8_t t_zero[256];
extern const uint8_t t_seq8[256]; extern const uint8_t t_seq8[256];
/*
* Debugging allocator
*/
extern int t_malloc_fail;
extern int t_malloc_fatal;
#endif #endif

View file

@ -155,6 +155,9 @@ main(int argc, char *argv[])
char *desc; char *desc;
int opt; int opt;
/* make all unintentional allocation failures fatal */
t_malloc_fatal = 1;
/* make stdout line-buffered to preserve ordering */ /* make stdout line-buffered to preserve ordering */
setvbuf(stdout, NULL, _IOLBF, 0); setvbuf(stdout, NULL, _IOLBF, 0);

View file

@ -108,11 +108,14 @@ static struct mapping *mappings;
/* if non-zero, all allocations fail */ /* if non-zero, all allocations fail */
int t_malloc_fail; int t_malloc_fail;
/* if non-zero, unintentional allocation failures are fatal */
int t_malloc_fatal;
/* /*
* Return a pointer to inaccessible memory. * Return a pointer to inaccessible memory.
*/ */
static void * static void *
malloc_null(void) t_malloc_null(void)
{ {
struct bucket *b; struct bucket *b;
@ -128,8 +131,12 @@ malloc_null(void)
return (b->base); return (b->base);
} }
/*
* Allocate a direct mapping. Round up the size to the nearest multiple
* of 8192, call mmap() with the correct arguments, and verify the result.
*/
static void * static void *
malloc_mapped(size_t size) t_malloc_mapped(size_t size)
{ {
struct mapping *m; struct mapping *m;
@ -150,8 +157,13 @@ malloc_mapped(size_t size)
return (m->base); return (m->base);
} }
/*
* Allocate from a bucket. Round up the size to the nearest power of two,
* select the appropriate bucket, and return the first free or unused
* block.
*/
static void * static void *
malloc_bucket(size_t size) t_malloc_bucket(size_t size)
{ {
unsigned int shift; unsigned int shift;
struct bucket *b; struct bucket *b;
@ -196,62 +208,72 @@ malloc_bucket(size_t size)
return (p); return (p);
} }
/*
* Core malloc() logic: select the correct backend based on the requested
* allocation size and call it.
*/
void *
t_malloc(size_t size)
{
/* select and call the right backend */
if (size == 0)
return (t_malloc_null());
else if (size > (1 << BUCKET_MAX_SHIFT))
return (t_malloc_mapped(size));
else
return (t_malloc_bucket(size));
}
/*
* Allocate an object of the requested size. According to the standard,
* the content of the allocated memory is undefined; we fill it with
* easily recognizable garbage.
*/
void * void *
malloc(size_t size) malloc(size_t size)
{ {
void *p; void *p;
/* asked to fail */
if (t_malloc_fail) { if (t_malloc_fail) {
errno = ENOMEM; errno = ENOMEM;
return (NULL); return (NULL);
} }
p = t_malloc(size);
/* select and call the right backend */ if (p == NULL && t_malloc_fatal)
if (size == 0) abort();
p = malloc_null();
else if (size > (1 << BUCKET_MAX_SHIFT))
p = malloc_mapped(size);
else
p = malloc_bucket(size);
if (p == NULL)
return (NULL);
/* fill with garbage */
memset(p, BUCKET_FILL_ALLOC, size); memset(p, BUCKET_FILL_ALLOC, size);
/* XXX fill the slop with garbage */
/* done! */
return (p); return (p);
} }
/*
* Allocate an array of n objects of the requested size and initialize it
* to zero.
*/
void * void *
calloc(size_t n, size_t size) calloc(size_t n, size_t size)
{ {
void *p; void *p;
/* asked to fail */
if (t_malloc_fail) { if (t_malloc_fail) {
errno = ENOMEM; errno = ENOMEM;
return (NULL); return (NULL);
} }
p = t_malloc(n * size);
/* select and call the right backend */ if (p == NULL && t_malloc_fatal)
if (size == 0) abort();
p = malloc_null();
else if (size > (1 << BUCKET_MAX_SHIFT))
p = malloc_mapped(n * size);
else
p = malloc_bucket(n * size);
if (p == NULL)
return (NULL);
/* fill with zeroes */
memset(p, 0, n * size); memset(p, 0, n * size);
/* XXX fill the slop with garbage */
/* done! */
return (p); return (p);
} }
/*
* Grow or shrink an allocated object, preserving its contents up to the
* smaller of the object's original and new size. According to the
* standard, the object may be either grown or shrunk in place or replaced
* with a new one. We always allocate a new object and free the old one.
*/
void * void *
realloc(void *o, size_t size) realloc(void *o, size_t size)
{ {
@ -290,16 +312,26 @@ realloc(void *o, size_t size)
abort(); abort();
found: found:
if ((p = malloc(size)) == NULL) if ((p = t_malloc(size)) == NULL) {
if (t_malloc_fatal)
abort();
return (NULL); return (NULL);
}
if (size > osize) if (size > osize)
memcpy(p, o, osize); memcpy(p, o, osize);
else else
memcpy(p, o, size); memcpy(p, o, size);
/* XXX fill the slop with garbage */
free(o); free(o);
return (p); return (p);
} }
/*
* Free an allocated object. According to the standard, the content of
* the memory previously occupied by the object is undefined. We fill it
* with easily recognizable garbage to facilitate debugging use-after-free
* bugs.
*/
void void
free(void *p) free(void *p)
{ {