blob: 879729594c30f90bf8d78b7aeee95b5318932fef [file] [log] [blame]
// Copyright 2020 The Pigweed Authors
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy of
// the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
#include <span>
#include "pw_allocator/freelist_heap.h"
#include "pw_boot_armv7m/boot.h"
#include "pw_malloc/malloc.h"
#include "pw_preprocessor/util.h"
namespace {
std::aligned_storage_t<sizeof(pw::allocator::FreeListHeapBuffer<>),
alignof(pw::allocator::FreeListHeapBuffer<>)>
buf;
std::span<std::byte> pw_allocator_freelist_raw_heap;
} // namespace
pw::allocator::FreeListHeapBuffer<>* pw_freelist_heap;
#if __cplusplus
extern "C" {
#endif
// Define the global heap variables.
void pw_MallocInit() {
// pw_boot_heap_low_addr and pw_boot_heap_high_addr specifies the heap region
// from the linker script in "pw_boot_armv7m".
std::span<std::byte> pw_allocator_freelist_raw_heap =
std::span(reinterpret_cast<std::byte*>(&pw_boot_heap_low_addr),
&pw_boot_heap_high_addr - &pw_boot_heap_low_addr);
pw_freelist_heap = new (&buf)
pw::allocator::FreeListHeapBuffer(pw_allocator_freelist_raw_heap);
}
// Wrapper functions for malloc, free, realloc and calloc.
// With linker options "-Wl --wrap=<function name>", linker will link
// "__wrap_<function name>" with "<function_name>", and calling
// "<function name>" will call "__wrap_<function name>" instead
// Linker options are set in a config in "pw_malloc:pw_malloc_config".
void* __wrap_malloc(size_t size) { return pw_freelist_heap->Allocate(size); }
void __wrap_free(void* ptr) { pw_freelist_heap->Free(ptr); }
void* __wrap_realloc(void* ptr, size_t size) {
return pw_freelist_heap->Realloc(ptr, size);
}
void* __wrap_calloc(size_t num, size_t size) {
return pw_freelist_heap->Calloc(num, size);
}
void* __wrap__malloc_r(struct _reent* r, size_t size) {
PW_UNUSED(r);
return pw_freelist_heap->Allocate(size);
}
void __wrap__free_r(struct _reent* r, void* ptr) {
PW_UNUSED(r);
pw_freelist_heap->Free(ptr);
}
void* __wrap__realloc_r(struct _reent* r, void* ptr, size_t size) {
PW_UNUSED(r);
return pw_freelist_heap->Realloc(ptr, size);
}
void* __wrap__calloc_r(struct _reent* r, size_t num, size_t size) {
PW_UNUSED(r);
return pw_freelist_heap->Calloc(num, size);
}
#if __cplusplus
}
#endif