mempool-alloconly.c revision d7b81a1e17f5b115cb1b36bf6c6f64295e357dc2
/* Copyright (c) 2002-2010 Dovecot authors, see the included COPYING file */
/* @UNSAFE: whole file */
#include "lib.h"
#include "safe-memset.h"
#include "mempool.h"
#include <stdlib.h>
#ifdef HAVE_GC_GC_H
# include <gc.h>
#endif
#define MAX_ALLOC_SIZE SSIZE_T_MAX
struct alloconly_pool {
int refcount;
struct pool_block *block;
#ifdef DEBUG
const char *name;
bool disable_warning;
#endif
bool clean_frees;
};
struct pool_block {
struct pool_block *prev;
/* unsigned char data[]; */
};
#define POOL_BLOCK_DATA(block) \
((unsigned char *) (block) + SIZEOF_POOLBLOCK)
#ifdef DEBUG
# define CLEAR_CHR 0xde
# define SENTRY_COUNT 8
#else
# define SENTRY_COUNT 0
# define CLEAR_CHR 0
#endif
static const struct pool_vfuncs static_alloconly_pool_vfuncs = {
};
static const struct pool static_alloconly_pool = {
.v = &static_alloconly_pool_vfuncs,
.alloconly_pool = TRUE,
};
#ifdef DEBUG
{
for (i = 0; i < used_size; ) {
i_panic("mempool-alloconly: saved alloc size broken");
i += MEM_ALIGN(sizeof(alloc_size));
i += alloc_size;
for (; i < max_pos; i++) {
i_panic("mempool-alloconly: buffer overflow");
}
}
if (i != used_size)
i_panic("mempool-alloconly: used_size wrong");
/* The unused data must be NULs */
if (data[i] != '\0')
i_unreached();
}
}
#endif
{
#ifdef DEBUG
sizeof(size_t)*2;
#endif
/* create a fake alloconly_pool so we can call block_alloc() */
/* now allocate the actual alloconly_pool from the created block */
#ifdef DEBUG
}
/* set base_size so p_clear() doesn't trash alloconly_pool structure. */
#endif
/* the first pool allocations must be from the first block */
}
{
struct alloconly_pool *apool;
return pool;
}
{
void *block;
/* destroy all but the last block */
/* destroy the last block */
#ifdef DEBUG
#else
if (apool->clean_frees) {
}
#endif
#ifndef USE_GC
#endif
}
{
#ifdef DEBUG
#else
return "alloconly";
#endif
}
{
}
{
return;
/* erase the pointer before freeing anything, as the pointer may
exist inside the pool's memory area */
}
{
struct pool_block *block;
/* each block is at least twice the size of the previous one */
#ifdef DEBUG
if (!apool->disable_warning) {
/* i_warning() overwrites unallocated data in data
stack, so make sure everything is allocated before
calling it. */
/*i_warning("Growing pool '%s' with: %"PRIuSIZE_T,
apool->name, size);*/
}
#endif
}
#ifndef USE_GC
#else
#endif
"): Out of memory", size);
}
}
{
void *mem;
#ifndef DEBUG
#else
#endif
/* we need a new block */
}
#ifdef DEBUG
/* write CLEAR_CHRs to sentry */
#endif
return mem;
}
{
/* we can free only the last allocation */
}
}
{
/* see if we want to grow the memory we allocated last */
/* yeah, see if we can grow */
/* just shrink the available size */
return TRUE;
}
}
return FALSE;
}
{
unsigned char *new_mem;
return mem;
/* see if we can directly grow it */
/* slow way - allocate + copy */
}
return mem;
}
{
struct pool_block *block;
#ifdef DEBUG
#endif
/* destroy all blocks but the oldest, which contains the
struct alloconly_pool allocation. */
#ifdef DEBUG
#else
if (apool->clean_frees) {
}
#endif
#ifndef USE_GC
#endif
}
/* clear the first block */
#ifdef DEBUG
#else
#endif
}
{
}
{
struct pool_block *block;
return size;
}
{
struct pool_block *block;
return size;
}