A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
Heap.cc
1#include "egg/core/ExpHeap.hh"
2
3using namespace Abstract::Memory;
4
5namespace EGG {
6
8Heap::Heap(MEMiHeapHead *handle) : m_handle(handle), m_children(Disposer::getLinkOffset()) {
9 m_block = nullptr;
10 m_parentHeap = nullptr;
11 m_name = "NoName";
12 m_flags.makeAllZero();
13
14 s_heapList.append(this);
15}
16
18Heap::~Heap() {
19 s_heapList.remove(this);
20}
21
23void Heap::dispose() {
24 Disposer *node = nullptr;
25 while ((node = reinterpret_cast<Disposer *>(m_children.getFirst()))) {
26 node->~Disposer();
27 }
28
29 ASSERT(!m_children.m_headObject && !m_children.m_tailObject);
30}
31
32Heap *Heap::becomeAllocatableHeap() {
33 Heap *oldHeap = s_allocatableHeap;
34 s_allocatableHeap = this;
35 return oldHeap;
36}
37
39Heap *Heap::becomeCurrentHeap() {
40 Heap *oldHeap = s_currentHeap;
41 s_currentHeap = this;
42 return oldHeap;
43}
44
46void *Heap::alloc(size_t size, int align, Heap *pHeap) {
47 Heap *currentHeap = s_currentHeap;
48
49 if (s_allocatableHeap) {
50 if (currentHeap && !pHeap) {
51 pHeap = currentHeap;
52 }
53
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());
57
58 return nullptr;
59 }
60 }
61
62 if (pHeap) {
63 return pHeap->alloc(size, align);
64 }
65
66 if (currentHeap) {
67 void *block = currentHeap->alloc(size, align);
68
69 if (!block) {
70 u32 heapFreeSize = currentHeap->getAllocatableSize(0x4);
71 s32 heapSize = GetAddrNum(currentHeap->getEndAddress()) -
72 GetAddrNum(currentHeap->getStartAddress());
73
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;
78
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);
85 }
86
87 return block;
88 }
89
90 WARN("HEAP ALLOC FAIL: Cannot allocate %d from heap %p", size, pHeap);
91 return nullptr;
92}
93
95void Heap::free(void *block, Heap *pHeap) {
96 if (!pHeap) {
97 MEMiHeapHead *handle = MEMiHeapHead::findContainHeap(block);
98 if (!handle) {
99 return;
100 }
101
102 pHeap = findHeap(handle);
103 if (!pHeap) {
104 return;
105 }
106 }
107
108 pHeap->free(block);
109}
110
111Heap *Heap::findHeap(MEMiHeapHead *handle) {
112 Heap *node = nullptr;
113 while ((node = reinterpret_cast<Heap *>(s_heapList.getNext(node)))) {
114 if (node->m_handle == handle) {
115 return node;
116 }
117 }
118
119 return nullptr;
120}
121
123Heap *Heap::findContainHeap(const void *block) {
124 MEMiHeapHead *handle = MEMiHeapHead::findContainHeap(block);
125 return handle ? findHeap(handle) : nullptr;
126}
127
128} // namespace EGG
129
131void *operator new(size_t size) noexcept {
132 return EGG::Heap::alloc(size, 4, nullptr);
133}
134
136void *operator new(size_t size, int align) noexcept {
137 return EGG::Heap::alloc(size, align, nullptr);
138}
139
141void *operator new(size_t size, EGG::Heap *heap, int align) noexcept {
142 return EGG::Heap::alloc(size, align, heap);
143}
144
146void *operator new[](size_t size) noexcept {
147 return EGG::Heap::alloc(size, 4, nullptr);
148}
149
151void *operator new[](size_t size, int align) noexcept {
152 return EGG::Heap::alloc(size, align, nullptr);
153}
154
156void *operator new[](size_t size, EGG::Heap *heap, int align) noexcept {
157 return EGG::Heap::alloc(size, align, heap);
158}
159
161void operator delete(void *block) noexcept {
162 EGG::Heap::free(block, nullptr);
163}
164
165void operator delete(void *block, size_t /* size */) noexcept {
166 EGG::Heap::free(block, nullptr);
167}
168
170void operator delete[](void *block) noexcept {
171 EGG::Heap::free(block, nullptr);
172}
173
174void operator delete[](void *block, size_t /* size */) noexcept {
175 EGG::Heap::free(block, nullptr);
176}
177
178MEMList EGG::Heap::s_heapList = MEMList(EGG::Heap::getOffset());
179
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....
Definition HeapCommon.hh:13
A high-level representation of a memory heap for managing dynamic memory allocation....
Definition Heap.hh:16
EGG core library.
Definition Archive.cc:6
Intrusive doubly-linked list. Links are placed within the corresponding object.
Definition List.hh:14