test-mempool-allocfree.c revision a7d1bd5be687bbb125ccfb5c685491414280ddac
/* Copyright (c) 2007-2017 Dovecot authors, see the included COPYING file */
#include "test-lib.h"
#define SENSE 0xAB /* produces 10101011 */
static bool mem_has_bytes(const void *mem, size_t size, uint8_t b)
{
const uint8_t *bytes = mem;
unsigned int i;
for (i = 0; i < size; i++) {
if (bytes[i] != b) {
i_debug("bytes[%u] != %u", i, b);
return FALSE;
}
}
return TRUE;
}
void test_mempool_allocfree(void)
{
pool_t pool;
unsigned int i;
size_t last_alloc = 0;
size_t used = 0;
size_t count = 0;
void *mem = NULL;
test_begin("mempool_allocfree");
pool = pool_allocfree_create("test");
for(i = 0; i <= 1000; i++) {
/* release previous allocation */
if ((i % 3) == 0) {
if (mem != NULL) {
test_assert_idx(mem_has_bytes(mem, last_alloc, SENSE), i);
used -= last_alloc;
count--;
}
last_alloc = 0;
p_free(pool, mem);
/* grow previous allocation */
} else if ((i % 5) == 0) {
if (mem != NULL)
used -= last_alloc;
else
count++;
mem = p_realloc(pool, mem, last_alloc, i*2);
if (last_alloc > 0)
test_assert_idx(mem_has_bytes(mem, last_alloc, SENSE), i);
memset(mem, SENSE, i*2);
last_alloc = i*2;
used += i*2;
/* shrink previous allocation */
} else if ((i % 7) == 0) {
if (mem != NULL)
used -= last_alloc;
else
count++;
mem = p_realloc(pool, mem, last_alloc, i-2);
if (last_alloc > 0)
test_assert_idx(mem_has_bytes(mem, i-2, SENSE), i);
memset(mem, SENSE, i-2);
last_alloc = i-2;
used += i-2;
/* allocate some memory */
} else {
mem = p_malloc(pool, i);
/* fill it with sense marker */
memset(mem, SENSE, i);
used += i;
count++;
last_alloc = i;
}
}
test_assert(pool_allocfree_get_total_used_size(pool) == used);
pool_unref(&pool);
/* make sure realloc works correctly */
pool = pool_allocfree_create("test");
mem = NULL;
for(i = 1; i < 1000; i++) {
mem = p_realloc(pool, mem, i-1, i);
test_assert_idx(mem_has_bytes(mem, i-1, 0xde), i);
memset(mem, 0xde, i);
}
pool_unref(&pool);
test_end();
}
enum fatal_test_state fatal_mempool_allocfree(unsigned int stage)
{
static pool_t pool;
if (pool == NULL && stage != 0)
return FATAL_TEST_FAILURE;
switch(stage) {
case 0: /* forbidden size */
test_begin("fatal_mempool_allocfree");
pool = pool_allocfree_create("fatal");
(void)p_malloc(pool, 0);
return FATAL_TEST_FAILURE;
case 1: /* logically impossible size */
(void)p_malloc(pool, SSIZE_T_MAX + 1ULL);
return FATAL_TEST_FAILURE;
case 2: /* physically impossible size */
(void)p_malloc(pool, SSIZE_T_MAX - 1024);
return FATAL_TEST_FAILURE;
/* Continue with other tests as follows:
case 3:
something_fatal();
return FATAL_TEST_FAILURE;
*/
}
/* Either our tests have finished, or the test suite has got confused. */
pool_unref(&pool);
test_end();
return FATAL_TEST_FINISHED;
}