blob: 8d07061421aab593fb582827f0fe3364ad600aff [file] [log] [blame]
/* libnih
*
* alloc.c - multi-reference hierarchial allocator
*
* Copyright © 2009 Scott James Remnant <scott@netsplit.com>.
* Copyright © 2009 Canonical Ltd.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2, as
* published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif /* HAVE_CONFIG_H */
#include <malloc.h>
#include <stdlib.h>
#include <nih/macros.h>
#include <nih/logging.h>
#include <nih/list.h>
#include "alloc.h"
/**
* NihAllocCtx:
* @parents: parents of this context,
* @children: children of this context,
* @destructor: function to be called when freed,
* @size: allocation size.
*
* This structure is placed before all allocations in memory and is used
* to build up an n-ary tree of them. Allocations may have multiple
* parent references and multiple children. Allocations are automatically
* freed if the last parent reference is freed. When an allocation is
* freed, all children are unreferenced and any destructors called.
*
* Members of @parents and @children are both NihAllocRef objects.
**/
typedef struct nih_alloc_ctx {
NihList parents;
NihList children;
NihDestructor destructor;
size_t size;
} NihAllocCtx;
/**
* NihAllocRef:
* @children_entry: list head in parent's children list,
* @parents_entry: list head in child's parents list,
* @parent: pointer to parent context,
* @child: pointer to child context.
*
* This structure is shared by both @parent and @child denoting a reference
* between the two of them. It is placed in @parent's children list through
* @children_entry and @child's parents list through @parents_entry.
**/
typedef struct nih_alloc_ref {
NihList children_entry;
NihList parents_entry;
NihAllocCtx *parent;
NihAllocCtx *child;
} NihAllocRef;
/**
* NIH_ALLOC_SIZE:
*
* Expands to the size of the NihAllocCtx structure plus whetever padding
* is needed to ensure the following pointer is generically aligned.
**/
#define NIH_ALLOC_SIZE (NIH_ALIGN_SIZE * (((sizeof (NihAllocCtx) - 1) \
/ NIH_ALIGN_SIZE) + 1))
/**
* NIH_ALLOC_CTX:
* @ptr: pointer to block of memory.
*
* Obtain the location of the NihAllocCtx structure given a pointer to the
* block of memory beyond it.
*
* Returns: pointer to NihAllocCtx structure or NULL if @ptr is NULL.
**/
#define NIH_ALLOC_CTX(ptr) (ptr ? (void *)(ptr) - NIH_ALLOC_SIZE : NULL)
/**
* NIH_ALLOC_PTR:
* @ctx: pointer to NihAllocCtx structure.
*
* Obtain the location of the block of memory given a pointer to the
* NihAllocCtx structure in front of it.
*
* Returns: pointer to block of memory.
**/
#define NIH_ALLOC_PTR(ctx) ((void *)(ctx) + NIH_ALLOC_SIZE)
/**
* NIH_ALLOC_FINALISED:
*
* Flag placed in the destructor field of a context to indicate the
* destructor has been called and the object is pending being freed.
**/
#define NIH_ALLOC_FINALISED ((void *)-1)
/* Prototypes for static functions */
static inline int nih_alloc_context_free (NihAllocCtx *ctx);
static inline NihAllocRef *nih_alloc_ref_new (NihAllocCtx *parent,
NihAllocCtx *child)
__attribute__ ((malloc));
static inline void nih_alloc_ref_free (NihAllocRef *ref);
static inline NihAllocRef *nih_alloc_ref_lookup (NihAllocCtx *parent,
NihAllocCtx *child);
/* Point to the functions we actually call for allocation. */
void *(*__nih_malloc) (size_t size) = malloc;
void *(*__nih_realloc) (void *ptr, size_t size) = realloc;
void (*__nih_free) (void *ptr) = free;
/**
* nih_alloc:
* @parent: parent object for new object,
* @size: size of requested object.
*
* Allocates an object in memory of at least @size bytes and returns a
* pointer to it.
*
* If @parent is not NULL, it should be a pointer to another object which
* will be used as a parent for the returned object otherwise the special
* NULL parent will be used instead. When all parents of the returned
* object are freed, the returned object will also be freed.
*
* If you have clean-up that you would like to run, you can assign a
* destructor using the nih_alloc_set_destructor() function.
*
* Returns: newly allocated object or NULL if insufficient memory.
**/
void *
nih_alloc (const void *parent,
size_t size)
{
NihAllocCtx *ctx;
ctx = __nih_malloc (NIH_ALLOC_SIZE + size);
if (! ctx)
return NULL;
nih_list_init (&ctx->parents);
nih_list_init (&ctx->children);
ctx->destructor = NULL;
ctx->size = size;
nih_alloc_ref_new (NIH_ALLOC_CTX (parent), ctx);
return NIH_ALLOC_PTR (ctx);
}
/**
* nih_realloc:
* @ptr: object to reallocate,
* @parent: parent object of new object,
* @size: size of new object.
*
* Adjusts the size of the object @ptr to be at least @size bytes, which
* may be larger or smaller than the existing object, and returns the
* new pointer.
*
* If @ptr is NULL, this simply calls nih_alloc() and passes both @parent
* and @size to it, returning the returned object.
*
* If @ptr is not NULL, @parent is ignored; though it is usual to pass a
* parent of @ptr for style reasons.
*
* Returns: reallocated object or NULL if insufficient memory.
**/
void *
nih_realloc (void * ptr,
const void *parent,
size_t size)
{
NihAllocCtx *ctx;
NihList * first_parent = NULL;
NihList * first_child = NULL;
if (! ptr)
return nih_alloc (parent, size);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
/* This is somewhat more difficult than alloc or free because we
* have two lists of pointers to worry about. Fortunately the
* properties of NihList help us a lot here.
*
* The problem is that references between us and our parents,
* and references between us and our children, all contain list
* pointers that are potentially invalid once relloc has been
* called.
*
* We could strip it all down before calling realloc then rebuild
* it afterwards, but that's expensive and could be error-prone in
* the case where the allocator fails.
*
* The solution is to rely on a property of nih_list_add(). The
* entry passed (to be added) is cut out of its containing list
* without dereferencing the return pointers, this means we can
* cut the bad pointers out simply by calling nih_list_add()
* to put the new entry back in the same position.
*
* Of course, this only works in the non-empty list case as trying
* to cut an entry out of an empty list would dereference those
* invalid pointers. Happily all we need to do for the empty
* list case is call nih_list_init() again.
*
* So we just remember the first parent and first child reference,
* or NULL if the list is empty.
*/
if (! NIH_LIST_EMPTY (&ctx->parents))
first_parent = ctx->parents.next;
if (! NIH_LIST_EMPTY (&ctx->children))
first_child = ctx->children.next;
/* Now do the actual realloc(), if this fails then we can just
* return NULL since we've not actually changed anything.
*/
ctx = __nih_realloc (ctx, NIH_ALLOC_SIZE + size);
if (! ctx)
return NULL;
ctx->size = size;
/* Now update our parents and children lists, or reinitialise,
* as noted above this ensures that all the pointers are correct
*/
if (first_parent) {
nih_list_add_after (first_parent, &ctx->parents);
} else {
nih_list_init (&ctx->parents);
}
if (first_child) {
nih_list_add_after (first_child, &ctx->children);
} else {
nih_list_init (&ctx->children);
}
/* We still have to fix up the parent and child pointers, but
* that's easy.
*/
NIH_LIST_FOREACH (&ctx->parents, iter) {
NihAllocRef *ref = NIH_LIST_ITER (iter, NihAllocRef,
parents_entry);
ref->child = ctx;
}
NIH_LIST_FOREACH (&ctx->children, iter) {
NihAllocRef *ref = NIH_LIST_ITER (iter, NihAllocRef,
children_entry);
ref->parent = ctx;
}
return NIH_ALLOC_PTR (ctx);
}
/**
* nih_free:
* @ptr: object to free.
*
* Returns the object @ptr to the allocator so the memory consumed may be
* re-used by something else.
*
* All parent references are discarded and the destructor for @ptr is called.
* Then all children are recursively unreferenced. Those that have no
* remaining parent references will also have their destructors called and
* their children unreferenced, etc. Once all destructors have been called,
* the objects themselves are freed.
*
* If you call nih_free() on an object with parent references, you should
* make sure that any pointers to the object are reset.
*
* If you are unsure whether or not there are references you should call
* nih_discard() which will discard the special NULL reference only if it
* exists, only freeing the object if no other references remain.
*
* Otherwise to remove a particular parent reference you should call
* nih_unref().
*
* Returns: return value from @ptr's destructor, or 0.
**/
int
nih_free (void *ptr)
{
NihAllocCtx *ctx;
nih_assert (ptr != NULL);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
/* Cast off our parents first, without recursing. This ensures
* we always have zero references before we call the destructor,
* and has the somewhat neat property of breaking any reference
* loops.
*/
NIH_LIST_FOREACH_SAFE (&ctx->parents, iter) {
NihAllocRef *ref = NIH_LIST_ITER (iter, NihAllocRef,
parents_entry);
nih_alloc_ref_free (ref);
}
return nih_alloc_context_free (ctx);
}
/**
* nih_discard:
* @ptr: object to discard.
*
* Discards the special NULL parent reference from @ptr if present; if
* no other references have been taken @ptr will be freed and the value
* from the destructor returned otherwise this function takes no
* further action.
*
* You would use nih_discard() when you allocated @ptr without any parent
* but have passed it to functions that may have taken a reference to it
* in the meantime. Compare with nih_free() which acts even if there are
* parent references, and nih_unref() which only removes a single parent
* reference that is known to exist.
*
* Returns: return value from @ptr's destructor, or 0.
**/
int
nih_discard (void *ptr)
{
NihAllocCtx *ctx;
NihAllocRef *ref;
nih_assert (ptr != NULL);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
ref = nih_alloc_ref_lookup (NULL, ctx);
if (! ref)
return 0;
nih_alloc_ref_free (ref);
if (NIH_LIST_EMPTY (&ctx->parents))
return nih_alloc_context_free (ctx);
return 0;
}
/**
* _nih_discard_local:
* @ptr: address of local object to be discarded.
*
* This function should never be called directly, it is used as part of the
* implementation of nih_local and simply calls nih_discard() with the
* local variable itself if non-NULL.
**/
void
_nih_discard_local (void *ptraddr)
{
/* Can't just take void ** as a parameter, since that will upset
* gcc typechecking, and we want to be able to be used on any
* pointer type.
*/
void **ptr = (void **)ptraddr;
if (*ptr)
nih_discard (*ptr);
}
/**
* nih_alloc_context_free:
* @ctx: context to free.
*
* This is the internal function called by nih_free(), nih_discard() and
* nih_unref() to actually free an allocated context and its attached
* objects.
*
* All parent references must have been discarded prior to calling this
* function.
*
* The destructor for @ctx is called, and then all children are recursively
* unreferenced. Those that have no remaining parent references will also
* have their destructors called and their children unreferenced, etc.
* Once all destructors have been called, the objects themselves are freed.
*
* Returns: return value from @ptr's destructor, or 0.
**/
static inline int
nih_alloc_context_free (NihAllocCtx *ctx)
{
int ret = 0;
nih_assert (ctx != NULL);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
nih_assert (NIH_LIST_EMPTY (&ctx->parents));
/* We have no parents, call our destructor before doing anything
* to our children. Save the return value, since this is what
* we return.
*/
if (ctx->destructor)
ret = ctx->destructor (NIH_ALLOC_PTR (ctx));
ctx->destructor = NIH_ALLOC_FINALISED;
/* Recursively finalise all of our children. */
NIH_LIST_FOREACH_SAFE (&ctx->children, iter) {
NihAllocRef *ref = NIH_LIST_ITER (iter, NihAllocRef,
children_entry);
/* Disassociate the child from its parent.
* If that was not the last parent, the child should not
* be freed, so destroy the rest of the reference and move
* on.
*/
nih_list_destroy (&ref->parents_entry);
if (! NIH_LIST_EMPTY (&ref->child->parents)) {
nih_list_destroy (&ref->children_entry);
free (ref);
continue;
}
/* Child is to be destroyed and has no links back to its
* parents. We call the destructor now.
*/
if (ref->child->destructor)
ref->child->destructor (NIH_ALLOC_PTR (ref->child));
ref->child->destructor = NIH_ALLOC_FINALISED;
/* Reparent all of its own children to us so that they too
* will be finalised if the last reference is removed.
*
* In order to do this depth-first while preserving order,
* we insert the items before our cursor; and then put the
* cursor back at the head of them.
*/
NIH_LIST_FOREACH_SAFE (&ref->child->children, citer) {
NihAllocRef *cref = NIH_LIST_ITER (citer, NihAllocRef,
children_entry);
nih_list_add (&_iter, &cref->children_entry);
}
nih_list_add_after (iter, &_iter);
}
/* We now have a single list of children all of which have no
* references back to us as their parent, and all of had their
* destructors called.
*
* Now we free them.
*/
NIH_LIST_FOREACH_SAFE (&ctx->children, iter) {
NihAllocRef *ref = NIH_LIST_ITER (iter, NihAllocRef,
children_entry);
__nih_free (ref->child);
nih_list_destroy (&ref->children_entry);
free (ref);
}
/* And now we can free ourselves. */
__nih_free (ctx);
return ret;
}
/**
* nih_alloc_real_set_destructor:
* @ptr: pointer to object,
* @destructor: destructor function to set.
*
* Sets the destructor of the allocated object @ptr to @destructor, which
* may be NULL to unset an existing destructor. Normally you would use
* the nih_alloc_set_destructor() macro which expands to this function
* but casts @destructor to the correct type, since almost all destructors
* will be defined with their argument to be the type of the object
* rather than void *.
*
* The destructor will be called before the object is freed, either
* explicitly by nih_free() or nih_discard(), or because the last parent
* has unreferenced the object.
*
* When the destructor is called, the parent references to the object will
* have already been discarded but all children references will be intact
* and none of the children will have been freed. There is no need to use
* a destructor to unreference or free children, that is automatic.
*
* The pointer @ptr passed to the destructor is that of the object being
* freed, and the destructor may return a value which will be the return
* value of nih_free() or nih_discard() if used directly on the object.
*
* Since objects may also be freed by unreferencing, and the value is not
* returned in this case, it should only be used for informational or
* debugging purposes.
**/
void
nih_alloc_real_set_destructor (const void * ptr,
NihDestructor destructor)
{
NihAllocCtx *ctx;
nih_assert (ptr != NULL);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
ctx->destructor = destructor;
}
/**
* nih_ref:
* @ptr: object to reference,
* @parent: new parent object.
*
* Adds a reference to the object @ptr from @parent, adding to any other
* objects referencing @ptr. @parent may be the special NULL parent.
*
* The reference can be broken using nih_unref().
*
* @ptr will only be automatically freed when the last parent unreferences
* it. It may still be manually freed with nih_free(), though this doesn't
* sort out any pointers.
*
* This function is generally used when accepting an object that you wish
* to hold a reference to, which is cheaper than making a copy. The caller
* must be careful to only use nih_discard() or nih_unref() to drop its own
* reference.
**/
void
nih_ref (const void *ptr,
const void *parent)
{
nih_assert (ptr != NULL);
nih_alloc_ref_new (NIH_ALLOC_CTX (parent), NIH_ALLOC_CTX (ptr));
}
/**
* nih_alloc_ref_new:
* @parent: parent context,
* @child: child context.
*
* This is the internal function used by nih_ref() and nih_alloc() to
* create a new reference between the @parent and @child contexts.
*
* Returns: new reference, already linked to both objects.
**/
static inline NihAllocRef *
nih_alloc_ref_new (NihAllocCtx *parent,
NihAllocCtx *child)
{
NihAllocRef *ref;
nih_assert ((parent == NULL)
|| (parent->destructor != NIH_ALLOC_FINALISED));
nih_assert (child != NULL);
nih_assert (child->destructor != NIH_ALLOC_FINALISED);
ref = NIH_MUST (malloc (sizeof (NihAllocRef)));
nih_list_init (&ref->children_entry);
nih_list_init (&ref->parents_entry);
ref->parent = parent;
ref->child = child;
if (parent)
nih_list_add_after (&parent->children, &ref->children_entry);
nih_list_add_after (&child->parents, &ref->parents_entry);
return ref;
}
/**
* nih_unref:
* @ptr: object to unreference,
* @parent: parent object to remove.
*
* Removes the reference to the object @ptr from @parent, if this is the
* last reference to @ptr then @ptr will be automatically freed. @parent
* may be the special NULL parent.
*
* You never need to call this in your own destructors since children
* are unreferenced automatically, however this function is useful if you
* only hold a reference to an object for a short period and wish to
* discard it.
**/
void
nih_unref (void * ptr,
const void *parent)
{
NihAllocCtx *ctx;
NihAllocRef *ref;
nih_assert (ptr != NULL);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
ref = nih_alloc_ref_lookup (NIH_ALLOC_CTX (parent), ctx);
nih_assert (ref != NULL);
nih_alloc_ref_free (ref);
if (NIH_LIST_EMPTY (&ctx->parents))
nih_alloc_context_free (ctx);
}
/**
* nih_alloc_ref_free:
* @ref: reference to free.
*
* This is the internal function used by nih_free() and nih_unref() to
* remove the reference @ref from its parent and child contexts. It does
* not free the child context, even if this is the last reference.
*
* This function is notably not called by nih_alloc_context_unref() since
* that manipulates the references to perform finalisation.
**/
static inline void
nih_alloc_ref_free (NihAllocRef *ref)
{
nih_assert (ref != NULL);
nih_list_destroy (&ref->children_entry);
nih_list_destroy (&ref->parents_entry);
free (ref);
}
/**
* nih_alloc_parent:
* @ptr: object to query,
* @parent: parent object to look for.
*
* @parent may be the special NULL parent.
*
* Returns: TRUE if @parent has a reference to @ptr, FALSE otherwise.
**/
int
nih_alloc_parent (const void *ptr,
const void *parent)
{
NihAllocCtx *ctx;
NihAllocRef *ref;
nih_assert (ptr != NULL);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
ref = nih_alloc_ref_lookup (NIH_ALLOC_CTX (parent), ctx);
return ref ? TRUE : FALSE;
}
/**
* nih_alloc_ref_lookup:
* @parent: parent context,
* @child: child context.
*
* This is the internal function used by nih_unref() and nih_alloc_parent()
* to lookup a reference between the @parent and @child contexts. @parent
* may be the special NULL parent.
*
* Returns: NihAllocRef structure or NULL if no reference exists.
**/
static inline NihAllocRef *
nih_alloc_ref_lookup (NihAllocCtx *parent,
NihAllocCtx *child)
{
nih_assert ((parent == NULL)
|| (parent->destructor != NIH_ALLOC_FINALISED));
nih_assert (child != NULL);
nih_assert (child->destructor != NIH_ALLOC_FINALISED);
NIH_LIST_FOREACH (&child->parents, iter) {
NihAllocRef *ref = NIH_LIST_ITER (iter, NihAllocRef,
parents_entry);
if (ref->parent == parent)
return ref;
}
return NULL;
}
/**
* nih_alloc_size:
* @ptr: pointer to object.
*
* Returns: the size of the allocated object, which may be larger than
* originally requested.
**/
size_t
nih_alloc_size (const void *ptr)
{
NihAllocCtx *ctx;
nih_assert (ptr != NULL);
ctx = NIH_ALLOC_CTX (ptr);
nih_assert (ctx->destructor != NIH_ALLOC_FINALISED);
return ctx->size;
}