mirror of
https://github.com/cryb-to/cryb-to.git
synced 2025-01-18 01:31:09 +00:00
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:
parent
5875ade2ed
commit
64a2da2b84
3 changed files with 75 additions and 34 deletions
6
t/t.h
6
t/t.h
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
100
t/t_malloc.c
100
t/t_malloc.c
|
@ -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)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue