A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
GameScene.cc
1#include "GameScene.hh"
2
3#include "game/system/KPadDirector.hh"
4#include "game/system/ResourceManager.hh"
5
6#include <egg/core/SceneManager.hh>
7
8namespace Scene {
9
11GameScene::GameScene() {
12 m_heap->setName("DefaultGameSceneHeap");
13 m_nextSceneId = -1;
14
15#ifdef BUILD_DEBUG
16 m_totalMemoryUsed = 0;
17 EGG::ExpHeap *heap = EGG::Heap::dynamicCastToExp(m_heap);
18 ASSERT(heap);
19
20 heap->calcGroupSize(&m_groupSizeRecord);
21 for (u16 groupID = 0; groupID < m_groupSizeRecord.size(); ++groupID) {
22 m_totalMemoryUsed += m_groupSizeRecord.getGroupSize(groupID);
23 }
24#endif // BUILD_DEBUG
25}
26
28GameScene::~GameScene() {
29 m_resources.clear();
30
31#ifdef BUILD_DEBUG
32 EGG::ExpHeap *heap = EGG::Heap::dynamicCastToExp(m_heap);
33 ASSERT(heap);
34
35 heap->calcGroupSize(&m_groupSizeRecord);
36 size_t sum = 0;
37 for (u16 groupID = 0; groupID < m_groupSizeRecord.size(); ++groupID) {
38 sum += m_groupSizeRecord.getGroupSize(groupID);
39 }
40
41 if (sum > m_totalMemoryUsed) {
42 WARN("MEMORY LEAK DETECTED: %zu bytes", sum - m_totalMemoryUsed);
43 getMemoryLeakTags();
44 }
45#endif // BUILD_DEBUG
46}
47
49void GameScene::calc() {
50 System::KPadDirector::Instance()->calc();
51 calcEngines();
52}
53
55void GameScene::enter() {
56 configure();
57 initScene();
58}
59
61void GameScene::exit() {
62 deinitScene();
63 unmountResources();
64}
65
67void GameScene::reinit() {
68 exit();
69 if (m_nextSceneId < 0) {
70 onReinit();
71 initScene();
72 } else {
73 m_sceneMgr->changeSiblingScene(m_nextSceneId);
74 }
75}
76
78void GameScene::appendResource(System::MultiDvdArchive *archive, s32 id) {
79 m_resources.push_back(new Resource(archive, id));
80}
81
83GameScene::Resource::Resource(System::MultiDvdArchive *archive, s32 id)
84 : archive(archive), id(id) {}
85
87void GameScene::initScene() {
88 createEngines();
89 System::KPadDirector::Instance()->reset();
90 initEngines();
91#ifdef BUILD_DEBUG
92 checkMemory();
93#endif // BUILD_DEBUG
94}
95
97void GameScene::deinitScene() {
98 if (m_nextSceneId >= 0) {
99 return;
100 }
101
102 destroyEngines();
103 System::KPadDirector::Instance()->clear();
104}
105
107void GameScene::unmountResources() {
108 auto *resourceManager = System::ResourceManager::Instance();
109 for (auto iter = m_resources.begin(); iter != m_resources.end();) {
110 Resource *resource = *iter;
111 resourceManager->unmount(resource->archive);
112 iter = m_resources.erase(iter);
113 delete resource;
114 }
115}
116
117#ifdef BUILD_DEBUG
118void GameScene::checkMemory() {
119 EGG::ExpHeap *heap = EGG::Heap::dynamicCastToExp(m_heap);
120 ASSERT(heap);
121
122 heap->calcGroupSize(&m_groupSizeRecord);
123 size_t defaultSize = m_groupSizeRecord.getGroupSize(static_cast<u16>(GroupID::None));
124
125 // Because we're working with a class that can be inherited (but not doubly-inherited),
126 // it's possible that derived classes can have a different size compared to GameScene
127 // We don't need to know the exact derived class, though, as we have the memory block head
128 // The scene is the first, always group ID 0 allocation to happen in the scene's heap
131 SubOffset(this, sizeof(Abstract::Memory::MEMiExpBlockHead)));
132 size_t initialAllocSize = blockHead->m_size;
133 ASSERT(defaultSize >= initialAllocSize);
134
135 defaultSize -= initialAllocSize;
136 if (defaultSize > 0) {
137 WARN("Default memory usage found! %zu bytes", defaultSize);
138 }
139
140 for (u16 groupID = 1; groupID < m_groupSizeRecord.size(); ++groupID) {
141 size_t size = m_groupSizeRecord.getGroupSize(groupID);
142 if (size == 0) {
143 continue;
144 }
145
146 DEBUG("Group ID %d: Allocated %zu bytes", groupID, size);
147 }
148}
149
150void GameScene::getMemoryLeakTags() {
151 size_t count = getMemoryLeakTagCount();
152 if (count == 0) {
153 return;
154 }
155
156 std::span<u32> tags = std::span<u32>(new u32[count], count);
157 for (auto &tag : tags) {
158 tag = 0; // empty tag
159 }
160
161 EGG::Heap::dynamicCastToExp(m_heap)->dynamicCastHandleToExp()->visitAllocated(ViewTags,
162 GetAddrNum(&tags));
163
164 DEBUG("TAGGED MEMORY BLOCKS:");
165 printf("[%d", tags[0]);
166 for (u32 i = 1; i < tags.size(); ++i) {
167 printf(", %d", tags[i]);
168 }
169 printf("]\n");
170
171 delete[] tags.data();
172}
173
174size_t GameScene::getMemoryLeakTagCount() {
175 size_t count = 0;
176 EGG::Heap::dynamicCastToExp(m_heap)->dynamicCastHandleToExp()->visitAllocated(IncreaseTagCount,
177 GetAddrNum(&count));
178 return count;
179}
180
181void GameScene::ViewTags(void *block, Abstract::Memory::MEMiHeapHead * /*heap*/, uintptr_t param) {
184 SubOffset(block, sizeof(Abstract::Memory::MEMiExpBlockHead)));
185 std::span<u32> *span = reinterpret_cast<std::span<u32> *>(param);
186 for (auto &tag : *span) {
187 if (tag == 0) {
188 tag = blockHead->m_tag;
189 break;
190 }
191 }
192}
193
194void GameScene::IncreaseTagCount(void * /*block*/, Abstract::Memory::MEMiHeapHead * /*heap*/,
195 uintptr_t param) {
196 size_t &count = *reinterpret_cast<size_t *>(param);
197 ++count;
198}
199#endif // BUILD_DEBUG
200
201} // namespace Scene
A low-level representation of a memory heap for managing dynamic memory allocation....
Definition HeapCommon.hh:13
High-level implementation of a memory heap for managing dynamic memory allocation....
Definition ExpHeap.hh:15
std::list< Resource * > m_resources
List of all active resources in the scene.
Definition GameScene.hh:57
Pertains to scene handling.
Definition GameScene.cc:8
Head of the memory block. Contains information about the block and a link in the corresponding used/f...
Definition ExpHeap.hh:38