mem.c revision 5a505fc4c2e99842052d9409790c7da0b5663bce
/*
* Copyright (C) 2004-2010, 2012-2015 Internet Systems Consortium, Inc. ("ISC")
* Copyright (C) 1997-2003 Internet Software Consortium.
*
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
* AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
* LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
* OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
* PERFORMANCE OF THIS SOFTWARE.
*/
/*! \file */
#include <config.h>
#include <stdio.h>
#include <stdlib.h>
#include <stddef.h>
#include <limits.h>
#include <isc/ondestroy.h>
#ifndef ISC_MEM_DEBUGGING
#define ISC_MEM_DEBUGGING 0
#endif
/*
* Constants.
*/
#define DEF_MAX_SIZE 1100
#define DEF_MEM_TARGET 4096
#define TABLE_INCREMENT 1024
#define DEBUGLIST_COUNT 1024
/*
* Types.
*/
typedef struct isc__mem isc__mem_t;
typedef struct isc__mempool isc__mempool_t;
typedef struct debuglink debuglink_t;
struct debuglink {
const void *ptr[DEBUGLIST_COUNT];
const char *file[DEBUGLIST_COUNT];
unsigned int line[DEBUGLIST_COUNT];
unsigned int count;
};
#else
#define FLARG_PASS
#define FLARG
#endif
struct element {
};
typedef struct {
/*!
* This structure must be ALIGNMENT_SIZE bytes.
*/
union {
char bytes[ALIGNMENT_SIZE];
} u;
} size_info;
struct stats {
unsigned long gets;
unsigned long totalgets;
unsigned long blocks;
unsigned long freefrags;
};
#endif
/* List of all active memory contexts. */
static isc_mutex_t contextslock;
static isc_mutex_t createlock;
/*%
* Total size of lost memory due to a bug of external library.
* Locked by the global lock.
*/
static isc_uint64_t totallost;
struct isc__mem {
unsigned int flags;
void * arg;
unsigned int references;
char name[16];
void * tag;
void * water_arg;
unsigned int poolcnt;
/* ISC_MEMFLAG_INTERNAL */
unsigned char ** basic_table;
unsigned int basic_table_count;
unsigned int basic_table_size;
unsigned char * lowest;
unsigned char * highest;
unsigned int debuglistcnt;
#endif
unsigned int memalloc_failures;
};
struct isc__mempool {
/* always unlocked */
/*%< locked via the memory context's lock */
/*%< optionally locked from here down */
unsigned int maxalloc; /*%< max number of items allowed */
unsigned int allocated; /*%< # of items currently given out */
unsigned int freecount; /*%< # of items on reserved list */
unsigned int freemax; /*%< # of items allowed on free list */
unsigned int fillcount; /*%< # of items to fetch on each fill */
/*%< Stats only. */
unsigned int gets; /*%< # of requests to this pool */
/*%< Debugging only. */
#endif
};
/*
* Private Inline-able.
*/
#if ! ISC_MEM_TRACKLINES
#define ADD_TRACE(a, b, c, d, e)
#define DELETE_TRACE(a, b, c, d, e)
#define ISC_MEMFUNC_SCOPE
#else
#define ADD_TRACE(a, b, c, d, e) \
do { \
if ((isc_mem_debugging & (ISC_MEM_DEBUGTRACE | \
ISC_MEM_DEBUGRECORD)) != 0 && \
b != NULL) \
add_trace_entry(a, b, c, d, e); \
} while (0)
#define DELETE_TRACE(a, b, c, d, e) delete_trace_entry(a, b, c, d, e)
static void
/*%
* The following are intended for internal use (indicated by "isc__"
* prefix) but are not declared as static, allowing direct access
* from unit tests, etc.
*/
void
void
void
void
void *
void
void
void *
void *
void
char *
void
void
void
void
void
const char *
void *
void
void
void
void *
void
void
unsigned int
unsigned int
void
unsigned int
unsigned int
void
unsigned int
void
void
unsigned int
#endif /* ISC_MEM_TRACKLINES */
static struct isc__memmethods {
/*%
* The following are defined just for avoiding unused static functions.
*/
} memmethods = {
{
},
(void *)isc_mem_createx,
(void *)isc_mem_create,
(void *)isc_mem_create2,
(void *)isc_mem_ondestroy,
(void *)isc_mem_stats,
(void *)isc_mem_setquota,
(void *)isc_mem_getquota,
(void *)isc_mem_setname,
(void *)isc_mem_getname,
(void *)isc_mem_gettag
};
static struct isc__mempoolmethods {
/*%
* The following are defined just for avoiding unused static functions.
*/
} mempoolmethods = {
{
},
(void *)isc_mempool_getfreemax,
(void *)isc_mempool_getfreecount,
(void *)isc_mempool_getmaxalloc,
(void *)isc_mempool_getfillcount
};
/*!
* mctx must be locked.
*/
static inline void
unsigned int i;
if ((isc_mem_debugging & ISC_MEM_DEBUGTRACE) != 0)
"add %p size %u "
"file %s line %u mctx %p\n"),
return;
goto next;
for (i = 0; i < DEBUGLIST_COUNT; i++) {
return;
}
}
next:
}
for (i = 1; i < DEBUGLIST_COUNT; i++) {
}
mctx->debuglistcnt++;
}
static inline void
{
unsigned int i;
if ((isc_mem_debugging & ISC_MEM_DEBUGTRACE) != 0)
"del %p size %u "
"file %s line %u mctx %p\n"),
return;
for (i = 0; i < DEBUGLIST_COUNT; i++) {
}
return;
}
}
}
/*
* If we get here, we didn't find the item on the list. We're
* screwed.
*/
}
#endif /* ISC_MEM_TRACKLINES */
static inline size_t
/*
* round down to ALIGNMENT_SIZE
*/
}
static inline size_t
/*!
* Round up the result in order to get a size big
* enough to satisfy the request and be aligned on ALIGNMENT_SIZE
* byte boundaries.
*/
if (size == 0U)
return (ALIGNMENT_SIZE);
}
static inline isc_boolean_t
void *new;
unsigned char **table;
unsigned int table_size;
int i;
/* Require: we hold the context lock. */
/*
* Did we hit the quota for this context?
*/
return (ISC_FALSE);
table_size * sizeof(unsigned char *));
ctx->memalloc_failures++;
return (ISC_FALSE);
}
if (ctx->basic_table_size != 0) {
sizeof(unsigned char *));
}
}
ctx->memalloc_failures++;
return (ISC_FALSE);
}
ctx->basic_table_count++;
for (i = 0; i < (NUM_BASIC_BLOCKS - 1); i++) {
}
/*
* curr is now pointing at the last block in the
* array.
*/
return (ISC_TRUE);
}
static inline isc_boolean_t
int i, frags;
void *new;
/*!
* Try to get more fragments by chopping up a basic block.
*/
if (!more_basic_blocks(ctx)) {
/*
* We can't get more memory from the OS, or we've
* hit the quota for this context.
*/
/*
* XXXRTH "At quota" notification here.
*/
return (ISC_FALSE);
}
}
/*
* Set up a linked-list of blocks of size
* "new_size".
*/
total_size -= new_size;
for (i = 0; i < (frags - 1); i++) {
total_size -= new_size;
}
/*
* Add the remaining fragment of the basic block to a free list.
*/
if (total_size > 0U) {
}
/*
* curr is now pointing at the last block in the
* array.
*/
return (ISC_TRUE);
}
static inline void *
void *ret;
/*
* memget() was called on something beyond our upper limit.
*/
goto done;
}
ctx->memalloc_failures++;
goto done;
}
/*
* If we don't set new_size to size, then the
* ISC_MEM_FILL code might write over bytes we
* don't own.
*/
goto done;
}
/*
* If there are no blocks in the free list for this size, get a chunk
* of memory and then break it up into "new_size"-sized blocks, adding
* them to the free list.
*/
return (NULL);
/*
* The free list uses the "rounded-up" size "new_size".
*/
/*
* The stats[] uses the _actual_ "size" requested by the
* caller, with the caveat (in the code above) that "size" >= the
* max. size (max_size) ends up getting recorded as a call to
* max_size.
*/
done:
#if ISC_MEM_FILL
#endif
return (ret);
}
#if ISC_MEM_FILL && ISC_MEM_CHECKOVERRUN
static inline void
unsigned char *cp;
cp++;
size++;
}
}
#endif
/* coverity[+free : arg-1] */
static inline void
/*
* memput() called on something beyond our upper limit.
*/
#if ISC_MEM_FILL
#endif
return;
}
#if ISC_MEM_FILL
#endif
#endif
/*
* The free list uses the "rounded-up" size "new_size".
*/
/*
* The stats[] uses the _actual_ "size" requested by the
* caller, with the caveat (in the code above) that "size" >= the
* max. size (max_size) ends up getting recorded as a call to
* max_size.
*/
}
/*!
* Perform a malloc, doing memory filling and overrun detection as necessary.
*/
static inline void *
char *ret;
size += 1;
#endif
ctx->memalloc_failures++;
#if ISC_MEM_FILL
#else
# if ISC_MEM_CHECKOVERRUN
# endif
#endif
return (ret);
}
/*!
* Perform a free, doing memory filling and overrun detection as necessary.
*/
/* coverity[+free : arg-1] */
static inline void
#endif
#if ISC_MEM_FILL
#else
#endif
}
/*!
* Update internal counters after a memory get.
*/
static inline void
} else {
}
}
/*!
* Update internal counters after a memory put.
*/
static inline void
} else {
}
}
/*
* Private.
*/
static void *
if (size == 0U)
size = 1;
}
static void
}
static void
initialize_action(void) {
totallost = 0;
}
/*
* Public.
*/
{
}
{
return (ISC_R_NOMEMORY);
if ((flags & ISC_MEMFLAG_NOLOCK) == 0) {
if (result != ISC_R_SUCCESS) {
return (result);
}
}
if (init_max_size == 0U)
else
ctx->debuglistcnt = 0;
#endif
ctx->basic_table_count = 0;
ctx->basic_table_size = 0;
goto error;
}
if ((flags & ISC_MEMFLAG_INTERNAL) != 0) {
if (target_size == 0U)
else
sizeof(element *));
goto error;
}
}
if ((isc_mem_debugging & ISC_MEM_DEBUGRECORD) != 0) {
unsigned int i;
goto error;
}
}
#endif
ctx->memalloc_failures = 0;
LOCK(&contextslock);
return (ISC_R_SUCCESS);
#endif /* ISC_MEM_TRACKLINES */
}
return (result);
}
static void
unsigned int i;
LOCK(&contextslock);
}
} else {
}
}
}
#endif
#endif
}
}
for (i = 0; i < ctx->basic_table_count; i++)
}
}
void
source->references++;
}
void
ctx->references--;
if (ctx->references == 0)
if (want_destroy)
}
/*
* isc_mem_putanddetach() is the equivalent of:
*
* mctx = NULL;
* isc_mem_attach(ptr->mctx, &mctx);
* isc_mem_detach(&ptr->mctx);
* isc_mem_put(mctx, ptr, sizeof(*ptr);
* isc_mem_detach(&mctx);
*/
void
/*
* Must be before mem_putunlocked() as ctxp is usually within
* [ptr..ptr+size).
*/
if ((isc_mem_debugging & ISC_MEM_DEBUGSIZE) != 0) {
if ((isc_mem_debugging & ISC_MEM_DEBUGCTX) != 0)
}
ctx->references--;
if (ctx->references == 0)
if (want_destroy)
return;
}
} else {
}
ctx->references--;
if (ctx->references == 0)
if (want_destroy)
}
void
/*
* This routine provides legacy support for callers who use mctxs
*/
#endif
ctx->references--;
}
return (res);
}
void *
void *ptr;
} else {
}
!ctx->is_overmem) {
}
}
(isc_mem_debugging & ISC_MEM_DEBUGUSAGE) != 0)
}
if (call_water)
return (ptr);
}
void
if ((isc_mem_debugging & ISC_MEM_DEBUGSIZE) != 0) {
if ((isc_mem_debugging & ISC_MEM_DEBUGCTX) != 0)
}
return;
}
} else {
}
/*
* The check against ctx->lo_water == 0 is for the condition
* when the context was pushed over hi_water but then had
* isc_mem_setwater() called with 0 for hi_water and lo_water.
*/
if (ctx->is_overmem &&
}
}
if (call_water)
}
void
if (flag == ISC_MEM_LOWATER)
else if (flag == ISC_MEM_HIWATER)
}
static void
unsigned int i, j;
const char *format;
"Dump of all outstanding "
"memory allocations:\n"));
"\tptr %p size %u file %s line %u\n");
for (j = 0; j < DEBUGLIST_COUNT; j++)
}
}
if (!found)
ISC_MSG_NONE, "\tNone.\n"));
}
}
#endif
/*
* Print the stats[] on the stream "out" with suitable formatting.
*/
void
size_t i;
const struct stats *s;
const isc__mempool_t *pool;
continue;
}
/*
* Note that since a pool can be locked now, these stats might be
* somewhat off if the pool is in active use at the time the stats
* are dumped. The link fields are protected by the isc_mem_t's
* lock, however, so walking this list and extracting integers from
* stats fields is always safe.
*/
"[Pool statistics]\n"));
ISC_MSG_POOLNAME, "name"),
ISC_MSG_POOLSIZE, "size"),
ISC_MSG_POOLMAXALLOC, "maxalloc"),
ISC_MSG_POOLALLOCATED, "allocated"),
ISC_MSG_POOLFREECOUNT, "freecount"),
ISC_MSG_POOLFREEMAX, "freemax"),
ISC_MSG_POOLFILLCOUNT, "fillcount"),
ISC_MSG_POOLGETS, "gets"),
"L");
}
#else
"(not tracked)",
#endif
}
#endif
}
/*
* Replacements for malloc() and free() -- they implicitly remember the
* size of the object allocated (with some additional overhead).
*/
static void *
size += ALIGNMENT_SIZE;
if ((isc_mem_debugging & ISC_MEM_DEBUGCTX) != 0)
size += ALIGNMENT_SIZE;
else
return (NULL);
if ((isc_mem_debugging & ISC_MEM_DEBUGCTX) != 0) {
si++;
}
return (&si[1]);
}
void *
} else {
}
#endif
!ctx->is_overmem) {
}
}
(isc_mem_debugging & ISC_MEM_DEBUGUSAGE) != 0)
}
if (call_water)
return (si);
}
void *
/*
* This function emulates the realloc(3) standard library function:
* - if size > 0, allocate new memory; and if ptr is non NULL, copy
* as much of the old contents to the new buffer and free the old one.
* Note that when allocation fails the original pointer is intact;
* the caller must free it.
* - if size is 0 and ptr is non NULL, simply free the given ptr.
* - this function returns:
* pointer to the newly allocated memory, or
* NULL if allocation fails or doesn't happen.
*/
if (size > 0U) {
if ((isc_mem_debugging & ISC_MEM_DEBUGCTX) != 0) {
}
}
return (new_ptr);
}
void
if ((isc_mem_debugging & ISC_MEM_DEBUGCTX) != 0) {
} else {
}
} else {
}
/*
* The check against ctx->lo_water == 0 is for the condition
* when the context was pushed over hi_water but then had
* isc_mem_setwater() called with 0 for hi_water and lo_water.
*/
if (ctx->is_overmem &&
}
}
if (call_water)
}
/*
* Other useful things.
*/
char *
char *ns;
return (ns);
}
void
}
/*
* Quotas
*/
void
}
return (quota);
}
return (inuse);
}
return (maxinuse);
}
return (total);
}
void
{
void *oldwater_arg;
} else {
}
}
/*
* We don't bother to lock the context because 100% accuracy isn't
* necessary (and even if we locked the context the returned value
* could be different from the actual state when it's used anyway)
*/
return (ctx->is_overmem);
}
void
}
const char *
return ("");
}
void *
}
/*
* Memory pool stuff
*/
/*
* Allocate space for this pool, initialize values, and if all works
* well, attach to the memory context.
*/
return (ISC_R_NOMEMORY);
#endif
return (ISC_R_SUCCESS);
}
void
#else
#endif
}
void
"isc__mempool_destroy(): mempool %s "
"leaked memory",
#endif
/*
* Return any items on the free list
*/
} else {
}
}
/*
* Remove our linked list entry from the memory context.
*/
}
void
}
void *
unsigned int i;
/*
* Don't let the caller go over quota
*/
goto out;
}
/*
* if we have a free list item, return the first here
*/
goto out;
}
/*
* We need to dip into the well. Lock the memory context here and
* fill up our free list.
*/
} else {
}
break;
}
/*
* If we didn't get any items, return NULL.
*/
goto out;
out:
}
#endif /* ISC_MEM_TRACKLINES */
return (item);
}
/* coverity[+free : arg-1] */
void
#endif /* ISC_MEM_TRACKLINES */
/*
* If our free list is full, return this to the mctx directly.
*/
} else {
}
return;
}
/*
* Otherwise, attach it to our free list and bump the counter.
*/
}
/*
* Quotas
*/
void
}
unsigned int
unsigned int freemax;
return (freemax);
}
unsigned int
unsigned int freecount;
return (freecount);
}
void
}
unsigned int
unsigned int maxalloc;
return (maxalloc);
}
unsigned int
unsigned int allocated;
return (allocated);
}
void
}
unsigned int
unsigned int fillcount;
return (fillcount);
}
isc__mem_register(void) {
return (isc_mem_register(isc_mem_create2));
}
void
#else
#endif
}
void
#if !ISC_MEM_TRACKLINES
#else
LOCK(&contextslock);
}
#endif
}
void
#if !ISC_MEM_TRACKLINES
#endif
LOCK(&contextslock);
if (!ISC_LIST_EMPTY(contexts)) {
}
#endif
INSIST(0);
}
}
unsigned int
unsigned int references;
return (references);
}
#if defined(HAVE_LIBXML2) || defined(HAVE_JSON)
typedef struct summarystat {
#endif
#ifdef HAVE_LIBXML2
static int
{
int xmlrc;
}
ctx->basic_table_count * sizeof(char *);
summary->contextsize +=
}
#endif
ctx->mem_target));
} else
return (xmlrc);
}
int
int xmlrc;
LOCK(&contextslock);
if (xmlrc < 0) {
goto error;
}
}
lost));
return (xmlrc);
}
#endif /* HAVE_LIBXML2 */
#ifdef HAVE_JSON
#define CHECKMEM(m) do { \
if (m == NULL) { \
result = ISC_R_NOMEMORY;\
goto error;\
} \
} while(0)
static isc_result_t
char buf[1024];
ctx->basic_table_count * sizeof(char *);
summary->contextsize +=
}
#endif
}
}
return (ISC_R_SUCCESS);
return (result);
}
LOCK(&contextslock);
if (result != ISC_R_SUCCESS) {
goto error;
}
}
return (ISC_R_SUCCESS);
return (result);
}
#endif /* HAVE_JSON */
LOCK(&createlock);
if (mem_createfunc == NULL)
else
UNLOCK(&createlock);
return (result);
}
unsigned int flags)
{
LOCK(&createlock);
UNLOCK(&createlock);
return (result);
}
if (isc_bind9)
LOCK(&createlock);
UNLOCK(&createlock);
return (result);
}
unsigned int flags)
{
if (isc_bind9)
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
void
}
void
{
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
if (isc_bind9)
return (isc__mem_inuse(mctx));
}
if (isc_bind9)
return (isc__mem_maxinuse(mctx));
}
if (isc_bind9)
return (isc__mem_total(mctx));
}
if (isc_bind9)
return (isc__mem_isovermem(mctx));
}
}
void
if (isc_bind9)
else
}
unsigned int
if (isc_bind9)
return (isc__mempool_getallocated(mpctx));
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
}
void *
if (isc_bind9)
}
void
if (isc_bind9)
else
}
void
if (isc_bind9)
else
/*
* XXX: We cannot always ensure *mctxp == NULL here
*/
}
void *
if (isc_bind9)
}
void *
if (isc_bind9)
}
char *
if (isc_bind9)
}
void
if (isc_bind9)
else
}
void *
if (isc_bind9)
}
void
if (isc_bind9)
else
}