src/cx/allocator.h

Mon, 18 Nov 2024 22:05:42 +0100

author
Mike Becker <universe@uap-core.de>
date
Mon, 18 Nov 2024 22:05:42 +0100
changeset 988
15b3ca7ee33f
parent 985
68754c7de906
permissions
-rw-r--r--

make ucx C++ compatible again (and add tests for it) - fixes #486

/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright 2021 Mike Becker, Olaf Wintermann All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 *   1. Redistributions of source code must retain the above copyright
 *      notice, this list of conditions and the following disclaimer.
 *
 *   2. Redistributions in binary form must reproduce the above copyright
 *      notice, this list of conditions and the following disclaimer in the
 *      documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */
/**
 * \file allocator.h
 * Interface for custom allocators.
 */

#ifndef UCX_ALLOCATOR_H
#define UCX_ALLOCATOR_H

#include "common.h"

#ifdef __cplusplus
extern "C" {
#endif

/**
 * The class definition for an allocator.
 */
typedef struct {
    /**
     * The allocator's malloc() implementation.
     */
    cx_attr_nonnull
    cx_attr_nodiscard cx_attr_allocsize(2)
    void *(*malloc)(
            void *data,
            size_t n
    );

    /**
     * The allocator's realloc() implementation.
     */
    cx_attr_nodiscard
    cx_attr_allocsize(3)
    void *(*realloc)(
            void *data,
            void *mem,
            size_t n
    );

    /**
     * The allocator's calloc() implementation.
     */
    cx_attr_nodiscard
    cx_attr_allocsize(2, 3)
    void *(*calloc)(
            void *data,
            size_t nelem,
            size_t n
    );

    /**
     * The allocator's free() implementation.
     */
    cx_attr_nonnull_arg(1)
    void (*free)(
            void *data,
            void *mem
    );
} cx_allocator_class;

/**
 * Structure holding the data for an allocator.
 */
struct cx_allocator_s {
    /**
     * A pointer to the instance of the allocator class.
     */
    cx_allocator_class *cl;
    /**
     * A pointer to the data this allocator uses.
     */
    void *data;
};

/**
 * High-Level type alias for the allocator type.
 */
typedef struct cx_allocator_s CxAllocator;

/**
 * A default allocator using standard library malloc() etc.
 */
extern CxAllocator *cxDefaultAllocator;

/**
 * Function pointer type for destructor functions.
 *
 * A destructor function deallocates possible contents and MAY free the memory
 * pointed to by \p memory. Read the documentation of the respective function
 * pointer to learn if a destructor SHALL, MAY, or MUST NOT free the memory in
 * that particular implementation.
 *
 * @param memory a pointer to the object to destruct
  */
typedef void (*cx_destructor_func)(void *memory);

/**
 * Function pointer type for destructor functions.
 *
 * A destructor function deallocates possible contents and MAY free the memory
 * pointed to by \p memory. Read the documentation of the respective function
 * pointer to learn if a destructor SHALL, MAY, or MUST NOT free the memory in
 * that particular implementation.
 *
 * @param data an optional pointer to custom data
 * @param memory a pointer to the object to destruct
  */
typedef void (*cx_destructor_func2)(
        void *data,
        void *memory
);

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 *
 * \par Error handling
 * \c errno will be set by realloc() on failure.
 *
 * @param mem pointer to the pointer to allocated block
 * @param n the new size in bytes
 * @return zero on success, non-zero on failure
 */
cx_attr_nonnull
cx_attr_nodiscard
int cx_reallocate(
        void **mem,
        size_t n
);

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 *
 * The size is calculated by multiplying \p nemb and \p size.
 *
 * \par Error handling
 * \c errno will be set by realloc() on failure or when the multiplication of
 * \p nmemb and \p size overflows.
 *
 * @param mem pointer to the pointer to allocated block
 * @param nmemb the number of elements
 * @param size the size of each element
 * @return zero on success, non-zero on failure
 */
cx_attr_nonnull
cx_attr_nodiscard
int cx_reallocatearray(
        void **mem,
        size_t nmemb,
        size_t size
);

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 *
 * \par Error handling
 * \c errno will be set by realloc() on failure.
 *
 * @param mem pointer to the pointer to allocated block
 * @param n the new size in bytes
 * @return zero on success, non-zero on failure
 */
#define cx_reallocate(mem, n) cx_reallocate((void**)(mem), n)

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 *
 * The size is calculated by multiplying \p nemb and \p size.
 *
 * \par Error handling
 * \c errno will be set by realloc() on failure or when the multiplication of
 * \p nmemb and \p size overflows.
 *
 * @param mem pointer to the pointer to allocated block
 * @param nmemb the number of elements
 * @param size the size of each element
 * @return zero on success, non-zero on failure
 */
#define cx_reallocatearray(mem, nmemb, size) \
    cx_reallocatearray((void**)(mem), nmemb, size)

/**
 * Free a block allocated by this allocator.
 *
 * \note Freeing a block of a different allocator is undefined.
 *
 * @param allocator the allocator
 * @param mem a pointer to the block to free
 */
cx_attr_nonnull_arg(1)
void cxFree(
        const CxAllocator *allocator,
        void *mem
);

/**
 * Allocate \p n bytes of memory.
 *
 * @param allocator the allocator
 * @param n the number of bytes
 * @return a pointer to the allocated memory
 */
cx_attr_nodiscard
cx_attr_nonnull
cx_attr_malloc
cx_attr_dealloc_ucx
cx_attr_allocsize(2)
void *cxMalloc(
        const CxAllocator *allocator,
        size_t n
);

/**
 * Re-allocate the previously allocated block in \p mem, making the new block
 * \p n bytes long.
 * This function may return the same pointer that was passed to it, if moving
 * the memory was not necessary.
 *
 * \note Re-allocating a block allocated by a different allocator is undefined.
 *
 * @param allocator the allocator
 * @param mem pointer to the previously allocated block
 * @param n the new size in bytes
 * @return a pointer to the re-allocated memory
 */
cx_attr_nodiscard
cx_attr_nonnull_arg(1)
cx_attr_dealloc_ucx
cx_attr_allocsize(3)
void *cxRealloc(
        const CxAllocator *allocator,
        void *mem,
        size_t n
);

/**
 * Re-allocate the previously allocated block in \p mem, making the new block
 * \p n bytes long.
 * This function may return the same pointer that was passed to it, if moving
 * the memory was not necessary.
 *
 * The size is calculated by multiplying \p nemb and \p size.
 * If that multiplication overflows, this function returns \c NULL and \c errno
 * will be set.
 *
 * \note Re-allocating a block allocated by a different allocator is undefined.
 *
 * @param allocator the allocator
 * @param mem pointer to the previously allocated block
 * @param nmemb the number of elements
 * @param size the size of each element
 * @return a pointer to the re-allocated memory
 */
cx_attr_nodiscard
cx_attr_nonnull_arg(1)
cx_attr_dealloc_ucx
cx_attr_allocsize(3, 4)
void *cxReallocArray(
        const CxAllocator *allocator,
        void *mem,
        size_t nmemb,
        size_t size
);

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 * This function acts like cxRealloc() using the pointer pointed to by \p mem.
 *
 * \note Re-allocating a block allocated by a different allocator is undefined.
 *
 * \par Error handling
 * \c errno will be set, if the underlying realloc function does so.
 *
 * @param allocator the allocator
 * @param mem pointer to the pointer to allocated block
 * @param n the new size in bytes
 * @return zero on success, non-zero on failure
 */
cx_attr_nodiscard
cx_attr_nonnull
int cxReallocate(
        const CxAllocator *allocator,
        void **mem,
        size_t n
);

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 * This function acts like cxRealloc() using the pointer pointed to by \p mem.
 *
 * \note Re-allocating a block allocated by a different allocator is undefined.
 *
 * \par Error handling
 * \c errno will be set, if the underlying realloc function does so.
 *
 * @param allocator the allocator
 * @param mem pointer to the pointer to allocated block
 * @param n the new size in bytes
 * @return zero on success, non-zero on failure
 */
#define cxReallocate(allocator, mem, n) \
    cxReallocate(allocator, (void**)(mem), n)

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 * This function acts like cxReallocArray() using the pointer pointed to
 * by \p mem.
 *
 * \note Re-allocating a block allocated by a different allocator is undefined.
 *
 * \par Error handling
 * \c errno will be set, if the underlying realloc function does so or the
 * multiplication of \p nmemb and \p size overflows.
 *
 * @param allocator the allocator
 * @param mem pointer to the pointer to allocated block
 * @param nmemb the number of elements
 * @param size the size of each element
 * @return zero on success, non-zero on failure
 */
cx_attr_nodiscard
cx_attr_nonnull
int cxReallocateArray(
        const CxAllocator *allocator,
        void **mem,
        size_t nmemb,
        size_t size
);

/**
 * Re-allocate a previously allocated block and changes the pointer in-place,
 * if necessary.
 * This function acts like cxReallocArray() using the pointer pointed to
 * by \p mem.
 *
 * \note Re-allocating a block allocated by a different allocator is undefined.
 *
 * \par Error handling
 * \c errno will be set, if the underlying realloc function does so or the
 * multiplication of \p nmemb and \p size overflows.
 *
 * @param allocator the allocator
 * @param mem pointer to the pointer to allocated block
 * @param nmemb the number of elements
 * @param size the size of each element
 * @return zero on success, non-zero on failure
 */
#define cxReallocateArray(allocator, mem, nmemb, size) \
        cxReallocateArray(allocator, (void**) (mem), nmemb, size)

/**
 * Allocate \p nelem elements of \p n bytes each, all initialized to zero.
 *
 * @param allocator the allocator
 * @param nelem the number of elements
 * @param n the size of each element in bytes
 * @return a pointer to the allocated memory
 */
cx_attr_nonnull_arg(1)
cx_attr_nodiscard
cx_attr_malloc
cx_attr_dealloc_ucx
cx_attr_allocsize(2, 3)
void *cxCalloc(
        const CxAllocator *allocator,
        size_t nelem,
        size_t n
);

#ifdef __cplusplus
} // extern "C"
#endif

#endif // UCX_ALLOCATOR_H

mercurial