mirror of
https://github.com/LadybirdBrowser/ladybird
synced 2026-04-26 01:35:08 +02:00
Use mimalloc for Ladybird-owned allocations without overriding malloc(). Route kmalloc(), kcalloc(), krealloc(), and kfree() through mimalloc, and put the embedded Rust crates on the same allocator via a shared shim in AK/kmalloc.cpp. This also lets us drop kfree_sized(), since it no longer used its size argument. StringData, Utf16StringData, JS object storage, Rust error strings, and the CoreAudio playback helpers can all free their AK-backed storage with plain kfree(). Sanitizer builds still use the system allocator. LeakSanitizer does not reliably trace references stored in mimalloc-managed AK containers, so static caches and other long-lived roots can look leaked. Pass the old size into the Rust realloc shim so aligned fallback reallocations can move posix_memalign-backed blocks safely. Static builds still need a little linker help. macOS app binaries need the Rust allocator entry points forced in from liblagom-ak.a, while static ELF links can pull in identical allocator shim definitions from multiple Rust staticlibs. Keep the Apple -u flags and allow those duplicate shim symbols for LibJS and LibRegex links on Linux and BSD.
230 lines
5.4 KiB
C++
230 lines
5.4 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, Andreas Kling <andreas@ladybird.org>
|
|
* Copyright (c) 2021, Daniel Bertalan <dani@danielbertalan.dev>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/kmalloc.h>
|
|
|
|
#if defined(AK_OS_SERENITY)
|
|
|
|
# include <AK/Assertions.h>
|
|
|
|
// However deceptively simple these functions look, they must not be inlined.
|
|
// Memory allocated in one translation unit has to be deallocatable in another
|
|
// translation unit, so these functions must be the same everywhere.
|
|
// By making these functions global, this invariant is enforced.
|
|
|
|
void* operator new(size_t size)
|
|
{
|
|
void* ptr = malloc(size);
|
|
VERIFY(ptr);
|
|
return ptr;
|
|
}
|
|
|
|
void* operator new(size_t size, std::nothrow_t const&) noexcept
|
|
{
|
|
return malloc(size);
|
|
}
|
|
|
|
void operator delete(void* ptr) noexcept
|
|
{
|
|
return free(ptr);
|
|
}
|
|
|
|
void operator delete(void* ptr, size_t) noexcept
|
|
{
|
|
return free(ptr);
|
|
}
|
|
|
|
void* operator new[](size_t size)
|
|
{
|
|
void* ptr = malloc(size);
|
|
VERIFY(ptr);
|
|
return ptr;
|
|
}
|
|
|
|
void* operator new[](size_t size, std::nothrow_t const&) noexcept
|
|
{
|
|
return malloc(size);
|
|
}
|
|
|
|
void operator delete[](void* ptr) noexcept
|
|
{
|
|
return free(ptr);
|
|
}
|
|
|
|
void operator delete[](void* ptr, size_t) noexcept
|
|
{
|
|
return free(ptr);
|
|
}
|
|
|
|
// This is usually provided by libstdc++ in most cases, and the kernel has its own definition in
|
|
// Kernel/Heap/kmalloc.cpp. If neither of those apply, the following should suffice to not fail during linking.
|
|
namespace AK_REPLACED_STD_NAMESPACE {
|
|
|
|
nothrow_t const nothrow;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
# include <cstddef>
|
|
# include <cstring>
|
|
|
|
# if __has_feature(address_sanitizer) || defined(__SANITIZE_ADDRESS__)
|
|
// LeakSanitizer does not reliably trace references stored in mimalloc-managed
|
|
// AK containers, so sanitizer builds fall back to the system allocator.
|
|
# define AK_USE_SYSTEM_ALLOCATOR_INSTRUMENTED 1
|
|
# else
|
|
# include <mimalloc.h>
|
|
# endif
|
|
|
|
static bool allocation_needs_explicit_alignment(size_t alignment)
|
|
{
|
|
return alignment > alignof(std::max_align_t);
|
|
}
|
|
|
|
# ifdef AK_USE_SYSTEM_ALLOCATOR_INSTRUMENTED
|
|
|
|
static void* aligned_alloc_with_system_allocator(size_t size, size_t alignment, bool zeroed)
|
|
{
|
|
void* ptr = nullptr;
|
|
auto actual_size = size == 0 ? static_cast<size_t>(1) : size;
|
|
if (auto result = posix_memalign(&ptr, alignment, actual_size); result != 0)
|
|
return nullptr;
|
|
if (zeroed)
|
|
__builtin_memset(ptr, 0, actual_size);
|
|
return ptr;
|
|
}
|
|
|
|
void* ak_kcalloc(size_t count, size_t size)
|
|
{
|
|
return calloc(count, size);
|
|
}
|
|
|
|
void* ak_kmalloc(size_t size)
|
|
{
|
|
return malloc(size);
|
|
}
|
|
|
|
void* ak_krealloc(void* ptr, size_t size)
|
|
{
|
|
return realloc(ptr, size);
|
|
}
|
|
|
|
size_t ak_kmalloc_good_size(size_t size)
|
|
{
|
|
return size;
|
|
}
|
|
|
|
void ak_kfree(void* ptr)
|
|
{
|
|
free(ptr);
|
|
}
|
|
|
|
extern "C" {
|
|
void* ladybird_rust_alloc(size_t size, size_t alignment);
|
|
void* ladybird_rust_alloc_zeroed(size_t size, size_t alignment);
|
|
void ladybird_rust_dealloc(void* ptr, size_t alignment);
|
|
void* ladybird_rust_realloc(void* ptr, size_t old_size, size_t new_size, size_t alignment);
|
|
}
|
|
|
|
extern "C" void* ladybird_rust_alloc(size_t size, size_t alignment)
|
|
{
|
|
if (allocation_needs_explicit_alignment(alignment))
|
|
return aligned_alloc_with_system_allocator(size, alignment, false);
|
|
return malloc(size);
|
|
}
|
|
|
|
extern "C" void* ladybird_rust_alloc_zeroed(size_t size, size_t alignment)
|
|
{
|
|
if (allocation_needs_explicit_alignment(alignment))
|
|
return aligned_alloc_with_system_allocator(size, alignment, true);
|
|
return calloc(1, size);
|
|
}
|
|
|
|
extern "C" void ladybird_rust_dealloc(void* ptr, size_t)
|
|
{
|
|
free(ptr);
|
|
}
|
|
|
|
extern "C" void* ladybird_rust_realloc(void* ptr, size_t old_size, size_t new_size, size_t alignment)
|
|
{
|
|
if (!allocation_needs_explicit_alignment(alignment))
|
|
return realloc(ptr, new_size);
|
|
|
|
auto* new_ptr = aligned_alloc_with_system_allocator(new_size, alignment, false);
|
|
if (!new_ptr)
|
|
return nullptr;
|
|
if (ptr)
|
|
__builtin_memcpy(new_ptr, ptr, old_size < new_size ? old_size : new_size);
|
|
free(ptr);
|
|
return new_ptr;
|
|
}
|
|
|
|
# else
|
|
|
|
void* ak_kcalloc(size_t count, size_t size)
|
|
{
|
|
return mi_calloc(count, size);
|
|
}
|
|
|
|
void* ak_kmalloc(size_t size)
|
|
{
|
|
return mi_malloc(size);
|
|
}
|
|
|
|
void* ak_krealloc(void* ptr, size_t size)
|
|
{
|
|
return mi_realloc(ptr, size);
|
|
}
|
|
|
|
size_t ak_kmalloc_good_size(size_t size)
|
|
{
|
|
return mi_good_size(size);
|
|
}
|
|
|
|
void ak_kfree(void* ptr)
|
|
{
|
|
mi_free(ptr);
|
|
}
|
|
|
|
extern "C" {
|
|
void* ladybird_rust_alloc(size_t size, size_t alignment);
|
|
void* ladybird_rust_alloc_zeroed(size_t size, size_t alignment);
|
|
void ladybird_rust_dealloc(void* ptr, size_t alignment);
|
|
void* ladybird_rust_realloc(void* ptr, size_t old_size, size_t new_size, size_t alignment);
|
|
}
|
|
|
|
extern "C" void* ladybird_rust_alloc(size_t size, size_t alignment)
|
|
{
|
|
if (allocation_needs_explicit_alignment(alignment))
|
|
return mi_malloc_aligned(size, alignment);
|
|
return mi_malloc(size);
|
|
}
|
|
|
|
extern "C" void* ladybird_rust_alloc_zeroed(size_t size, size_t alignment)
|
|
{
|
|
if (allocation_needs_explicit_alignment(alignment))
|
|
return mi_zalloc_aligned(size, alignment);
|
|
return mi_zalloc(size);
|
|
}
|
|
|
|
extern "C" void ladybird_rust_dealloc(void* ptr, size_t)
|
|
{
|
|
mi_free(ptr);
|
|
}
|
|
|
|
extern "C" void* ladybird_rust_realloc(void* ptr, size_t, size_t new_size, size_t alignment)
|
|
{
|
|
if (allocation_needs_explicit_alignment(alignment))
|
|
return mi_realloc_aligned(ptr, new_size, alignment);
|
|
return mi_realloc(ptr, new_size);
|
|
}
|
|
|
|
# endif
|
|
|
|
#endif
|