mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
eefed0e0cd
some standard c libraries (e.g. newlib before 4.0.0) don't perform proper overflow check in the multiplication. We just implement calloc here ourselves on top of malloc with proper overflow check in place. This might even safe a handful of ROM bytes.
74 lines
1.5 KiB
C
74 lines
1.5 KiB
C
/*
|
|
* Copyright (C) 2019 Gunar Schorcht
|
|
*
|
|
* 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 <gunar@schorcht.net>
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include "assert.h"
|
|
#include "irq.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 *__wrap_malloc(size_t size)
|
|
{
|
|
assert(!irq_is_in());
|
|
mutex_lock(&_lock);
|
|
void *ptr = __real_malloc(size);
|
|
mutex_unlock(&_lock);
|
|
return ptr;
|
|
}
|
|
|
|
void __wrap_free(void *ptr)
|
|
{
|
|
assert(!irq_is_in());
|
|
mutex_lock(&_lock);
|
|
__real_free(ptr);
|
|
mutex_unlock(&_lock);
|
|
}
|
|
|
|
void *__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;
|
|
}
|
|
|
|
void *res = __wrap_malloc(total_size);
|
|
if (res) {
|
|
memset(res, 0, total_size);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
void *__wrap_realloc(void *ptr, size_t size)
|
|
{
|
|
assert(!irq_is_in());
|
|
mutex_lock(&_lock);
|
|
void *new = __real_realloc(ptr, size);
|
|
mutex_unlock(&_lock);
|
|
return new;
|
|
}
|
|
|
|
/** @} */
|