mirror of
https://github.com/GrapheneOS/hardened_malloc.git
synced 2024-10-01 01:36:01 -04:00
d8817417cc
C11 noreturn isn't available in C++.
150 lines
3.6 KiB
C++
150 lines
3.6 KiB
C++
#include <bits/functexcept.h>
|
|
#include <new>
|
|
|
|
#include "h_malloc.h"
|
|
#include "util.h"
|
|
|
|
COLD static void *handle_out_of_memory(size_t size, bool nothrow) {
|
|
void *ptr = nullptr;
|
|
|
|
do {
|
|
std::new_handler handler = std::get_new_handler();
|
|
if (handler == nullptr) {
|
|
break;
|
|
}
|
|
|
|
try {
|
|
handler();
|
|
} catch (const std::bad_alloc &) {
|
|
break;
|
|
}
|
|
|
|
ptr = h_malloc(size);
|
|
} while (ptr == nullptr);
|
|
|
|
if (ptr == nullptr && !nothrow) {
|
|
std::__throw_bad_alloc();
|
|
}
|
|
return ptr;
|
|
}
|
|
|
|
static inline void *new_impl(size_t size, bool nothrow) {
|
|
void *ptr = h_malloc(size);
|
|
if (likely(ptr != nullptr)) {
|
|
return ptr;
|
|
}
|
|
return handle_out_of_memory(size, nothrow);
|
|
}
|
|
|
|
EXPORT void *operator new(size_t size) {
|
|
return new_impl(size, false);
|
|
}
|
|
|
|
EXPORT void *operator new[](size_t size) {
|
|
return new_impl(size, false);
|
|
}
|
|
|
|
EXPORT void *operator new(size_t size, const std::nothrow_t &) noexcept {
|
|
return new_impl(size, true);
|
|
}
|
|
|
|
EXPORT void *operator new[](size_t size, const std::nothrow_t &) noexcept {
|
|
return new_impl(size, true);
|
|
}
|
|
|
|
EXPORT void operator delete(void *ptr) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete[](void *ptr) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete(void *ptr, const std::nothrow_t &) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete[](void *ptr, const std::nothrow_t &) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete(void *ptr, size_t size) noexcept {
|
|
h_free_sized(ptr, size);
|
|
}
|
|
|
|
EXPORT void operator delete[](void *ptr, size_t size) noexcept {
|
|
h_free_sized(ptr, size);
|
|
}
|
|
|
|
COLD static void *handle_out_of_memory(size_t size, size_t alignment, bool nothrow) {
|
|
void *ptr = nullptr;
|
|
|
|
do {
|
|
std::new_handler handler = std::get_new_handler();
|
|
if (handler == nullptr) {
|
|
break;
|
|
}
|
|
|
|
try {
|
|
handler();
|
|
} catch (const std::bad_alloc &) {
|
|
break;
|
|
}
|
|
|
|
ptr = h_aligned_alloc(alignment, size);
|
|
} while (ptr == nullptr);
|
|
|
|
if (ptr == nullptr && !nothrow) {
|
|
std::__throw_bad_alloc();
|
|
}
|
|
return ptr;
|
|
}
|
|
|
|
static inline void *new_impl(size_t size, size_t alignment, bool nothrow) {
|
|
void *ptr = h_aligned_alloc(alignment, size);
|
|
if (likely(ptr != nullptr)) {
|
|
return ptr;
|
|
}
|
|
return handle_out_of_memory(size, alignment, nothrow);
|
|
}
|
|
|
|
EXPORT void *operator new(size_t size, std::align_val_t alignment) {
|
|
return new_impl(size, static_cast<size_t>(alignment), false);
|
|
}
|
|
|
|
EXPORT void *operator new[](size_t size, std::align_val_t alignment) {
|
|
return new_impl(size, static_cast<size_t>(alignment), false);
|
|
}
|
|
|
|
EXPORT void *operator new(size_t size, std::align_val_t alignment, const std::nothrow_t &) noexcept {
|
|
return new_impl(size, static_cast<size_t>(alignment), true);
|
|
}
|
|
|
|
EXPORT void *operator new[](size_t size, std::align_val_t alignment, const std::nothrow_t &) noexcept {
|
|
return new_impl(size, static_cast<size_t>(alignment), true);
|
|
}
|
|
|
|
EXPORT void operator delete(void *ptr, std::align_val_t) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete[](void *ptr, std::align_val_t) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete(void *ptr, std::align_val_t, const std::nothrow_t &) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete[](void *ptr, std::align_val_t, const std::nothrow_t &) noexcept {
|
|
h_free(ptr);
|
|
}
|
|
|
|
EXPORT void operator delete(void *ptr, size_t size, std::align_val_t) noexcept {
|
|
h_free_sized(ptr, size);
|
|
}
|
|
|
|
EXPORT void operator delete[](void *ptr, size_t size, std::align_val_t) noexcept {
|
|
h_free_sized(ptr, size);
|
|
}
|