1#include "egg/core/ExpHeap.hh"
3using namespace Abstract::Memory;
8Heap::Heap(
MEMiHeapHead *handle) : m_handle(handle), m_children(Disposer::getLinkOffset()) {
10 m_parentHeap =
nullptr;
12 m_flags.makeAllZero();
14 s_heapList.append(
this);
19 s_heapList.remove(
this);
24 Disposer *node =
nullptr;
25 while ((node =
reinterpret_cast<Disposer *
>(m_children.getFirst()))) {
29 ASSERT(!m_children.m_headObject && !m_children.m_tailObject);
32Heap *Heap::becomeAllocatableHeap() {
33 Heap *oldHeap = s_allocatableHeap;
34 s_allocatableHeap =
this;
39Heap *Heap::becomeCurrentHeap() {
40 Heap *oldHeap = s_currentHeap;
46void *Heap::alloc(
size_t size,
int align, Heap *pHeap) {
47 Heap *currentHeap = s_currentHeap;
49 if (s_allocatableHeap) {
50 if (currentHeap && !pHeap) {
54 if (pHeap != s_allocatableHeap) {
55 WARN(
"HEAP ALLOC FAIL (%p, %s): Allocatable heap is %p (%s)", pHeap, pHeap->getName(),
56 s_allocatableHeap, s_allocatableHeap->getName());
63 return pHeap->alloc(size, align);
67 void *block = currentHeap->alloc(size, align);
70 u32 heapFreeSize = currentHeap->getAllocatableSize(0x4);
71 s32 heapSize = GetAddrNum(currentHeap->getEndAddress()) -
72 GetAddrNum(currentHeap->getStartAddress());
74 constexpr f32 BYTES_TO_MBYTES = 1024.0f * 1024.0f;
75 f32 heapSizeMB =
static_cast<f32
>(heapSize) / BYTES_TO_MBYTES;
76 f32 heapFreeSizeMB =
static_cast<f32
>(heapFreeSize) / BYTES_TO_MBYTES;
77 f32 sizeMB =
static_cast<f32
>(size) / BYTES_TO_MBYTES;
79 WARN(
"HEAP ALLOC FAIL (%p, %s):\nTotal bytes: %d (%.1fMBytes)\nFree bytes: %d "
80 "(%.1fMBytes)\nAlloc bytes: %d "
81 "(%.1fMBytes)\nAlign: %d",
82 currentHeap, currentHeap->getName(),
static_cast<f64
>(heapSize),
83 static_cast<f64
>(heapSizeMB), heapFreeSize,
static_cast<f64
>(heapFreeSizeMB),
84 size,
static_cast<f64
>(sizeMB), align);
90 WARN(
"HEAP ALLOC FAIL: Cannot allocate %d from heap %p", size, pHeap);
95void Heap::free(
void *block, Heap *pHeap) {
97 MEMiHeapHead *handle = MEMiHeapHead::findContainHeap(block);
102 pHeap = findHeap(handle);
112 Heap *node =
nullptr;
113 while ((node =
reinterpret_cast<Heap *
>(s_heapList.getNext(node)))) {
114 if (node->m_handle == handle) {
123Heap *Heap::findContainHeap(
const void *block) {
124 MEMiHeapHead *handle = MEMiHeapHead::findContainHeap(block);
125 return handle ? findHeap(handle) : nullptr;
131void *
operator new(
size_t size)
noexcept {
132 return EGG::Heap::alloc(size, 4,
nullptr);
136void *
operator new(
size_t size,
int align)
noexcept {
137 return EGG::Heap::alloc(size, align,
nullptr);
141void *
operator new(
size_t size,
EGG::Heap *heap,
int align)
noexcept {
142 return EGG::Heap::alloc(size, align, heap);
146void *
operator new[](
size_t size)
noexcept {
147 return EGG::Heap::alloc(size, 4,
nullptr);
151void *
operator new[](
size_t size,
int align)
noexcept {
152 return EGG::Heap::alloc(size, align,
nullptr);
156void *
operator new[](
size_t size,
EGG::Heap *heap,
int align)
noexcept {
157 return EGG::Heap::alloc(size, align, heap);
161void operator delete(
void *block)
noexcept {
162 EGG::Heap::free(block,
nullptr);
165void operator delete(
void *block,
size_t )
noexcept {
166 EGG::Heap::free(block,
nullptr);
170void operator delete[](
void *block)
noexcept {
171 EGG::Heap::free(block,
nullptr);
174void operator delete[](
void *block,
size_t )
noexcept {
175 EGG::Heap::free(block,
nullptr);
180EGG::Heap *EGG::Heap::s_currentHeap =
nullptr;
181EGG::Heap *EGG::Heap::s_allocatableHeap =
nullptr;
A low-level representation of a memory heap for managing dynamic memory allocation....
A high-level representation of a memory heap for managing dynamic memory allocation....
Intrusive doubly-linked list. Links are placed within the corresponding object.