blob: f3eb3c98e641667544ad5bfac4c7b05a35c5df5e [file] [log] [blame]
/*
* Copyright (c) 2016 Nordic Semiconductor ASA
* Copyright (c) 2016 Vinayak Kariappa Chettimada
*
* SPDX-License-Identifier: Apache-2.0
*/
#ifndef MALIGN
/**
* @brief Force compiler to place memory at-least on a x-byte boundary
* @details Compiler extension. Supported by GCC and Clang
*/
#define MALIGN(x) __attribute__((aligned(x)))
#endif
#ifndef MROUND
/**
* @brief Round up to nearest multiple of 4, for unsigned integers
* @details
* The addition of 3 forces x into the next multiple of 4. This is responsible
* for the rounding in the the next step, to be Up.
* For ANDing of ~3: We observe y & (~3) == (y>>2)<<2, and we recognize
* (y>>2) as a floored division, which is almost undone by the left-shift. The
* flooring can't be undone so have achieved a rounding.
*
* Examples:
* MROUND( 0) = 0
* MROUND( 1) = 4
* MROUND( 2) = 4
* MROUND( 3) = 4
* MROUND( 4) = 4
* MROUND( 5) = 8
* MROUND( 8) = 8
* MROUND( 9) = 12
* MROUND(13) = 16
*/
#define MROUND(x) (((uint32_t)(x)+3) & (~((uint32_t)3)))
#endif
/* When memory is free in the mem_pool, first few bytes are reserved for
* maintaining the next pointer and free count.
*/
#define MEM_FREE_MEMBER_NEXT_SIZE (sizeof(void *))
#define MEM_FREE_MEMBER_COUNT_SIZE (sizeof(uint16_t))
#define MEM_FREE_RESERVED_SIZE ((MEM_FREE_MEMBER_NEXT_SIZE) + \
(MEM_FREE_MEMBER_COUNT_SIZE))
/* Define to check if a structure member is safe to be accessed when the memory
* is in the mem_pool free list. I.e. whether a structure member be safely
* accessed after the mem being freed.
*/
#define MEM_FREE_MEMBER_ACCESS_BUILD_ASSERT(type, member) \
BUILD_ASSERT(offsetof(type, member) >= MEM_FREE_RESERVED_SIZE, \
"Possible unsafe member access inside free mem pool")
void mem_init(void *mem_pool, uint16_t mem_size, uint16_t mem_count, void **mem_head);
void *mem_acquire(void **mem_head);
void mem_release(void *mem, void **mem_head);
uint16_t mem_free_count_get(void *mem_head);
void *mem_get(void *mem_pool, uint16_t mem_size, uint16_t index);
uint16_t mem_index_get(void *mem, void *mem_pool, uint16_t mem_size);
void mem_rcopy(uint8_t *dst, uint8_t const *src, uint16_t len);
uint8_t mem_nz(uint8_t *src, uint16_t len);
uint32_t mem_ut(void);