|  | /* | 
|  | * Copyright (c) 2017 Intel Corporation | 
|  | * | 
|  | * SPDX-License-Identifier: Apache-2.0 | 
|  | */ | 
|  |  | 
|  | #include <zephyr/kernel.h> | 
|  | #include <string.h> | 
|  | #include <zephyr/sys/math_extras.h> | 
|  | #include <zephyr/sys/util.h> | 
|  |  | 
|  | static void *z_heap_aligned_alloc(struct k_heap *heap, size_t align, size_t size) | 
|  | { | 
|  | void *mem; | 
|  | struct k_heap **heap_ref; | 
|  | size_t __align; | 
|  |  | 
|  | /* | 
|  | * Adjust the size to make room for our heap reference. | 
|  | * Merge a rewind bit with align value (see sys_heap_aligned_alloc()). | 
|  | * This allows for storing the heap pointer right below the aligned | 
|  | * boundary without wasting any memory. | 
|  | */ | 
|  | if (size_add_overflow(size, sizeof(heap_ref), &size)) { | 
|  | return NULL; | 
|  | } | 
|  | __align = align | sizeof(heap_ref); | 
|  |  | 
|  | mem = k_heap_aligned_alloc(heap, __align, size, K_NO_WAIT); | 
|  | if (mem == NULL) { | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | heap_ref = mem; | 
|  | *heap_ref = heap; | 
|  | mem = ++heap_ref; | 
|  | __ASSERT(align == 0 || ((uintptr_t)mem & (align - 1)) == 0, | 
|  | "misaligned memory at %p (align = %zu)", mem, align); | 
|  |  | 
|  | return mem; | 
|  | } | 
|  |  | 
|  | void k_free(void *ptr) | 
|  | { | 
|  | struct k_heap **heap_ref; | 
|  |  | 
|  | if (ptr != NULL) { | 
|  | heap_ref = ptr; | 
|  | --heap_ref; | 
|  | ptr = heap_ref; | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_free, *heap_ref, heap_ref); | 
|  |  | 
|  | k_heap_free(*heap_ref, ptr); | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_free, *heap_ref, heap_ref); | 
|  | } | 
|  | } | 
|  |  | 
|  | #if (K_HEAP_MEM_POOL_SIZE > 0) | 
|  |  | 
|  | K_HEAP_DEFINE(_system_heap, K_HEAP_MEM_POOL_SIZE); | 
|  | #define _SYSTEM_HEAP (&_system_heap) | 
|  |  | 
|  | void *k_aligned_alloc(size_t align, size_t size) | 
|  | { | 
|  | __ASSERT(align / sizeof(void *) >= 1 | 
|  | && (align % sizeof(void *)) == 0, | 
|  | "align must be a multiple of sizeof(void *)"); | 
|  |  | 
|  | __ASSERT((align & (align - 1)) == 0, | 
|  | "align must be a power of 2"); | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_aligned_alloc, _SYSTEM_HEAP); | 
|  |  | 
|  | void *ret = z_heap_aligned_alloc(_SYSTEM_HEAP, align, size); | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_aligned_alloc, _SYSTEM_HEAP, ret); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | void *k_malloc(size_t size) | 
|  | { | 
|  | SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_malloc, _SYSTEM_HEAP); | 
|  |  | 
|  | void *ret = k_aligned_alloc(sizeof(void *), size); | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_malloc, _SYSTEM_HEAP, ret); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | void *k_calloc(size_t nmemb, size_t size) | 
|  | { | 
|  | void *ret; | 
|  | size_t bounds; | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_calloc, _SYSTEM_HEAP); | 
|  |  | 
|  | if (size_mul_overflow(nmemb, size, &bounds)) { | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_calloc, _SYSTEM_HEAP, NULL); | 
|  |  | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ret = k_malloc(bounds); | 
|  | if (ret != NULL) { | 
|  | (void)memset(ret, 0, bounds); | 
|  | } | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_calloc, _SYSTEM_HEAP, ret); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | void *k_realloc(void *ptr, size_t size) | 
|  | { | 
|  | struct k_heap *heap, **heap_ref; | 
|  | void *ret; | 
|  |  | 
|  | if (size == 0) { | 
|  | k_free(ptr); | 
|  | return NULL; | 
|  | } | 
|  | if (ptr == NULL) { | 
|  | return k_malloc(size); | 
|  | } | 
|  | heap_ref = ptr; | 
|  | ptr = --heap_ref; | 
|  | heap = *heap_ref; | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_ENTER(k_heap_sys, k_realloc, heap, ptr); | 
|  |  | 
|  | if (size_add_overflow(size, sizeof(heap_ref), &size)) { | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_realloc, heap, ptr, NULL); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | ret = k_heap_realloc(heap, ptr, size, K_NO_WAIT); | 
|  |  | 
|  | if (ret != NULL) { | 
|  | heap_ref = ret; | 
|  | ret = ++heap_ref; | 
|  | } | 
|  |  | 
|  | SYS_PORT_TRACING_OBJ_FUNC_EXIT(k_heap_sys, k_realloc, heap, ptr, ret); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | void k_thread_system_pool_assign(struct k_thread *thread) | 
|  | { | 
|  | thread->resource_pool = _SYSTEM_HEAP; | 
|  | } | 
|  | #else | 
|  | #define _SYSTEM_HEAP	NULL | 
|  | #endif /* K_HEAP_MEM_POOL_SIZE */ | 
|  |  | 
|  | void *z_thread_aligned_alloc(size_t align, size_t size) | 
|  | { | 
|  | void *ret; | 
|  | struct k_heap *heap; | 
|  |  | 
|  | if (k_is_in_isr()) { | 
|  | heap = _SYSTEM_HEAP; | 
|  | } else { | 
|  | heap = _current->resource_pool; | 
|  | } | 
|  |  | 
|  | if (heap != NULL) { | 
|  | ret = z_heap_aligned_alloc(heap, align, size); | 
|  | } else { | 
|  | ret = NULL; | 
|  | } | 
|  |  | 
|  | return ret; | 
|  | } |