1 /* 2 * Copyright (c) 2016 Nordic Semiconductor ASA 3 * Copyright (c) 2016 Vinayak Kariappa Chettimada 4 * 5 * SPDX-License-Identifier: Apache-2.0 6 */ 7 8 #ifndef MALIGN 9 /** 10 * @brief Force compiler to place memory at-least on a x-byte boundary 11 * @details Compiler extension. Supported by GCC and Clang 12 */ 13 #define MALIGN(x) __attribute__((aligned(x))) 14 #endif 15 16 #ifndef MROUND 17 /** 18 * @brief Round up to nearest multiple of 4, for unsigned integers 19 * @details 20 * The addition of 3 forces x into the next multiple of 4. This is responsible 21 * for the rounding in the next step, to be Up. 22 * For ANDing of ~3: We observe y & (~3) == (y>>2)<<2, and we recognize 23 * (y>>2) as a floored division, which is almost undone by the left-shift. The 24 * flooring can't be undone so have achieved a rounding. 25 * 26 * Examples: 27 * MROUND( 0) = 0 28 * MROUND( 1) = 4 29 * MROUND( 2) = 4 30 * MROUND( 3) = 4 31 * MROUND( 4) = 4 32 * MROUND( 5) = 8 33 * MROUND( 8) = 8 34 * MROUND( 9) = 12 35 * MROUND(13) = 16 36 */ 37 #define MROUND(x) (((uint32_t)(x)+3) & (~((uint32_t)3))) 38 #endif 39 40 /* When memory is free in the mem_pool, first few bytes are reserved for 41 * maintaining the next pointer and free count. 42 */ 43 #define MEM_FREE_MEMBER_NEXT_SIZE (sizeof(void *)) 44 #define MEM_FREE_MEMBER_COUNT_SIZE (sizeof(uint16_t)) 45 #define MEM_FREE_RESERVED_SIZE ((MEM_FREE_MEMBER_NEXT_SIZE) + \ 46 (MEM_FREE_MEMBER_COUNT_SIZE)) 47 48 /* Define to check if a structure member is safe to be accessed when the memory 49 * is in the mem_pool free list. I.e. whether a structure member be safely 50 * accessed after the mem being freed. 51 */ 52 #define MEM_FREE_MEMBER_ACCESS_BUILD_ASSERT(type, member) \ 53 BUILD_ASSERT(offsetof(type, member) >= MEM_FREE_RESERVED_SIZE, \ 54 "Possible unsafe member access inside free mem pool") 55 56 void mem_init(void *mem_pool, uint16_t mem_size, uint16_t mem_count, void **mem_head); 57 void *mem_acquire(void **mem_head); 58 void mem_release(void *mem, void **mem_head); 59 60 uint16_t mem_free_count_get(void *mem_head); 61 void *mem_get(const void *mem_pool, uint16_t mem_size, uint16_t index); 62 uint16_t mem_index_get(const void *mem, const void *mem_pool, uint16_t mem_size); 63 64 void mem_rcopy(uint8_t *dst, uint8_t const *src, uint16_t len); 65 uint8_t mem_nz(uint8_t *src, uint16_t len); 66 67 int mem_ut(void); 68