/* * Copyright (C) 2019 Gunar Schorcht * 2022 Otto-von-Guericke-Universität Magdeburg * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @{ * * @file * @brief Implements various POSIX syscalls * @author Gunar Schorcht * @author Marian Buschsieweke */ #include #include #include "architecture.h" #include "assert.h" #include "cpu.h" #include "irq.h" #include "kernel_defines.h" #include "malloc_monitor_internal.h" #include "mutex.h" extern void *__real_malloc(size_t size); extern void __real_free(void *ptr); extern void *__real_realloc(void *ptr, size_t size); static mutex_t _lock; void __attribute__((used)) *__wrap_malloc(size_t size) { assert(!irq_is_in()); mutex_lock(&_lock); void *ptr = __real_malloc(size); if (IS_USED(MODULE_MALLOC_MONITOR)) { malloc_monitor_add(ptr, size, cpu_get_caller_pc(), "m"); } mutex_unlock(&_lock); return ptr; } void __attribute__((used)) __wrap_free(void *ptr) { assert(!irq_is_in()); mutex_lock(&_lock); __real_free(ptr); if (IS_USED(MODULE_MALLOC_MONITOR)) { malloc_monitor_rm(ptr, cpu_get_caller_pc()); } mutex_unlock(&_lock); } void * __attribute__((used)) __wrap_calloc(size_t nmemb, size_t size) { /* some c libs don't perform proper overflow check (e.g. newlib < 4.0.0). Hence, we * just implement calloc on top of malloc ourselves. In addition to ensuring proper * overflow checks, this likely saves a bit of ROM */ size_t total_size; if (__builtin_mul_overflow(nmemb, size, &total_size)) { return NULL; } mutex_lock(&_lock); void *res = __real_malloc(total_size); if (IS_USED(MODULE_MALLOC_MONITOR)) { malloc_monitor_add(res, total_size, cpu_get_caller_pc(), "c"); } mutex_unlock(&_lock); if (res) { memset(res, 0, total_size); } return res; } void * __attribute__((used))__wrap_realloc(void *ptr, size_t size) { assert(!irq_is_in()); mutex_lock(&_lock); void *new = __real_realloc(ptr, size); if (IS_USED(MODULE_MALLOC_MONITOR)) { malloc_monitor_mv(ptr, new, size, cpu_get_caller_pc()); } mutex_unlock(&_lock); return new; } /** @} */