src/cx/common.h

Sat, 04 Jan 2025 12:31:28 +0100

author
Mike Becker <universe@uap-core.de>
date
Sat, 04 Jan 2025 12:31:28 +0100
changeset 1084
0bcd71d2615a
parent 1076
bb4da3255de3
child 1093
bcbf6bf582fa
permissions
-rw-r--r--

change cx_array_reserve() and cx_array_copy() to accept width in bytes instead of bits

/*
 * 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 common.h
 *
 * \brief Common definitions and feature checks.
 *
 * \author Mike Becker
 * \author Olaf Wintermann
 * \copyright 2-Clause BSD License
 *
 * \mainpage UAP Common Extensions
 * Library with common and useful functions, macros and data structures.
 * <p>
 * Latest available source:<br>
 * <a href="https://sourceforge.net/projects/ucx/files/">https://sourceforge.net/projects/ucx/files/</a>
 * </p>
 *
 * <p>
 * Repositories:<br>
 * <a href="https://sourceforge.net/p/ucx/code">https://sourceforge.net/p/ucx/code</a>
 * -&nbsp;or&nbsp;-
 * <a href="https://develop.uap-core.de/hg/ucx">https://develop.uap-core.de/hg/ucx</a>
 * </p>
 *
 * <h2>LICENCE</h2>
 *
 * 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.
 */

#ifndef UCX_COMMON_H
#define UCX_COMMON_H

/** Major UCX version as integer constant. */
#define UCX_VERSION_MAJOR   3

/** Minor UCX version as integer constant. */
#define UCX_VERSION_MINOR   1

/** Version constant which ensures to increase monotonically. */
#define UCX_VERSION (((UCX_VERSION_MAJOR)<<16)|UCX_VERSION_MINOR)

// ---------------------------------------------------------------------------
//       Common includes
// ---------------------------------------------------------------------------

#include <stdlib.h>
#include <stddef.h>
#include <stdbool.h>
#include <stdint.h>
#include <sys/types.h>

// ---------------------------------------------------------------------------
//       Architecture Detection
// ---------------------------------------------------------------------------

#if INTPTR_MAX == INT64_MAX
/**
 * The address width in bits on this platform.
 */
#define CX_WORDSIZE 64
#elif INTPTR_MAX == INT32_MAX
/**
 * The address width in bits on this platform.
 */
#define CX_WORDSIZE 32
#else
#error Unknown pointer size or missing size macros!
#endif

// ---------------------------------------------------------------------------
//       Missing Defines
// ---------------------------------------------------------------------------

#ifndef SSIZE_MAX // not defined in glibc since C23 and MSVC
#if CX_WORDSIZE == 64
#define SSIZE_MAX 0x7fffffffffffffffll
#else
#define SSIZE_MAX 0x7fffffffl
#endif
#endif


// ---------------------------------------------------------------------------
//       Attribute definitions
// ---------------------------------------------------------------------------

#ifndef __GNUC__
/**
 * Removes GNU C attributes where they are not supported.
 */
#define __attribute__(x)
#endif

/**
 * All pointer arguments must be non-NULL.
 */
#define cx_attr_nonnull __attribute__((__nonnull__))

/**
 * The specified pointer arguments must be non-NULL.
 */
#define cx_attr_nonnull_arg(...) __attribute__((__nonnull__(__VA_ARGS__)))

/**
 * The returned value is guaranteed to be non-NULL.
 */
#define cx_attr_returns_nonnull __attribute__((__returns_nonnull__))

/**
 * The attributed function always returns freshly allocated memory.
 */
#define cx_attr_malloc __attribute__((__malloc__))

#ifndef __clang__
/**
 * The pointer returned by the attributed function is supposed to be freed
 * by \p freefunc.
 *
 * @param freefunc the function that shall be used to free the memory
 * @param freefunc_arg the index of the pointer argument in \p freefunc
 */
#define cx_attr_dealloc(freefunc, freefunc_arg) \
    __attribute__((__malloc__(freefunc, freefunc_arg)))
#else
/**
 * Not supported in clang.
 */
#define cx_attr_dealloc(...)
#endif // __clang__

/**
 * Shortcut to specify #cxFree() as deallocator.
 */
#define cx_attr_dealloc_ucx cx_attr_dealloc(cxFree, 2)

/**
 * Specifies the parameters from which the allocation size is calculated.
 */
#define cx_attr_allocsize(...) __attribute__((__alloc_size__(__VA_ARGS__)))


#ifdef __clang__
/**
 * No support for \c null_terminated_string_arg in clang or GCC below 14.
 */
#define cx_attr_cstr_arg(idx)
/**
 * No support for access attribute in clang.
 */
#define cx_attr_access(mode, ...)
#else
#if __GNUC__ < 10
/**
 * No support for access attribute in GCC < 10.
 */
#define cx_attr_access(mode, ...)
#else
/**
 * Helper macro to define access macros.
 */
#define cx_attr_access(mode, ...) __attribute__((__access__(mode, __VA_ARGS__)))
#endif // __GNUC__ < 10
#if __GNUC__ < 14
/**
 * No support for \c null_terminated_string_arg in clang or GCC below 14.
 */
#define cx_attr_cstr_arg(idx)
#else
/**
 * The specified argument is expected to be a zero-terminated string.
 *
 * @param idx the index of the argument
 */
#define cx_attr_cstr_arg(idx) \
    __attribute__((__null_terminated_string_arg__(idx)))
#endif // __GNUC__ < 14
#endif // __clang__


/**
 * Specifies that the function will only read through the given pointer.
 *
 * Takes one or two arguments: the index of the pointer and (optionally) the
 * index of another argument specifying the maximum number of accessed bytes.
 */
#define cx_attr_access_r(...) cx_attr_access(__read_only__, __VA_ARGS__)

/**
 * Specifies that the function will read and write through the given pointer.
 *
 * Takes one or two arguments: the index of the pointer and (optionally) the
 * index of another argument specifying the maximum number of accessed bytes.
 */
#define cx_attr_access_rw(...) cx_attr_access(__read_write__, __VA_ARGS__)

/**
 * Specifies that the function will only write through the given pointer.
 *
 * Takes one or two arguments: the index of the pointer and (optionally) the
 * index of another argument specifying the maximum number of accessed bytes.
 */
#define cx_attr_access_w(...) cx_attr_access(__write_only__, __VA_ARGS__)

#if __STDC_VERSION__ >= 202300L

/**
 * Do not warn about unused variable.
 */
#define cx_attr_unused [[maybe_unused]]

/**
 * Warn about discarded return value.
 */
#define cx_attr_nodiscard [[nodiscard]]

#else // no C23

/**
 * Do not warn about unused variable.
 */
#define cx_attr_unused __attribute__((__unused__))

/**
 * Warn about discarded return value.
 */
#define cx_attr_nodiscard __attribute__((__warn_unused_result__))

#endif // __STDC_VERSION__

// ---------------------------------------------------------------------------
//       Useful function pointers
// ---------------------------------------------------------------------------

/**
 * Function pointer compatible with fwrite-like functions.
 */
typedef size_t (*cx_write_func)(
        const void *,
        size_t,
        size_t,
        void *
) cx_attr_nonnull;

/**
 * Function pointer compatible with fread-like functions.
 */
typedef size_t (*cx_read_func)(
        void *,
        size_t,
        size_t,
        void *
) cx_attr_nonnull;

// ---------------------------------------------------------------------------
//       Utility macros
// ---------------------------------------------------------------------------

/**
 * Determines the number of members in a static C array.
 *
 * @attention never use this to determine the size of a dynamically allocated
 * array.
 *
 * @param arr the array identifier
 * @return the number of elements
 */
#define cx_nmemb(arr) (sizeof(arr)/sizeof((arr)[0]))

// ---------------------------------------------------------------------------
//       szmul implementation
// ---------------------------------------------------------------------------

#if (__GNUC__ >= 5 || defined(__clang__)) && !defined(CX_NO_SZMUL_BUILTIN)
#define CX_SZMUL_BUILTIN

/**
 * Alias for \c __builtin_mul_overflow.
 *
 * Performs a multiplication of size_t values and checks for overflow.
 *
 * @param a first operand
 * @param b second operand
 * @param result a pointer to a size_t, where the result should
 * be stored
 * @return zero, if no overflow occurred and the result is correct, non-zero
 * otherwise
 */
#define cx_szmul(a, b, result) __builtin_mul_overflow(a, b, result)

#else // no GNUC or clang bultin

/**
 * Performs a multiplication of size_t values and checks for overflow.
  *
 * @param a first operand
 * @param b second operand
 * @param result a pointer to a size_t, where the result should
 * be stored
 * @return zero, if no overflow occurred and the result is correct, non-zero
 * otherwise
 */
#define cx_szmul(a, b, result) cx_szmul_impl(a, b, result)

/**
 * Performs a multiplication of size_t values and checks for overflow.
 *
 * This is a custom implementation in case there is no compiler builtin
 * available.
 *
 * @param a first operand
 * @param b second operand
 * @param result a pointer to a size_t where the result should be stored
 * @return zero, if no overflow occurred and the result is correct, non-zero
 * otherwise
 */
#if __cplusplus
extern "C"
#endif
int cx_szmul_impl(size_t a, size_t b, size_t *result);

#endif // cx_szmul


// ---------------------------------------------------------------------------
//       Fixes for MSVC incompatibilities
// ---------------------------------------------------------------------------

#ifdef _MSC_VER
// fix missing ssize_t definition
#include <BaseTsd.h>
typedef SSIZE_T ssize_t;

// fix missing _Thread_local support
#define _Thread_local __declspec(thread)
#endif // _MSC_VER

#endif // UCX_COMMON_H

mercurial