A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KTestSystem.cc
1#include "KTestSystem.hh"
2
3#include "host/Option.hh"
4#include "host/SceneCreatorDynamic.hh"
5
6#include <game/kart/KartObjectManager.hh>
7#include <game/system/RaceManager.hh>
8
9#include <abstract/File.hh>
10
11// We use an unscoped enum to avoid static_casting in all usecases
12// This is defined in the source due to its lack of scoping
13enum Changelog {
14 Initial = 1,
15 AddedExtVel = 2,
16 AddedIntVel = 3,
17 AddedSpeed = 4,
18 AddedRotation = 5,
19 AddedCheckpoints = 6,
20};
21
22struct TestHeader {
23 u32 signature;
24 u16 byteOrderMark;
25 u16 frameCount;
26 u16 versionMajor;
27 u16 versionMinor;
28 u32 dataOffset;
29};
30
35 constexpr u32 TEST_HEADER_SIGNATURE = 0x54535448; // TSTH
36 constexpr u32 TEST_FOOTER_SIGNATURE = 0x54535446; // TSTF
37 constexpr u16 SUITE_MAJOR_VER = 1;
38 constexpr u16 SUITE_MAX_MINOR_VER = 0;
39
40 auto *sceneCreator = new Host::SceneCreatorDynamic;
41 m_sceneMgr = new EGG::SceneManager(sceneCreator);
42
43 System::RaceConfig::RegisterInitCallback(OnInit, nullptr);
44 Abstract::File::Remove("results.txt");
45
46 u16 numTestCases = m_stream.read_u16();
47 u16 testMajorVer = m_stream.read_u16();
48 u16 testMinorVer = m_stream.read_u16();
49
50 if (testMajorVer != SUITE_MAJOR_VER || testMinorVer > SUITE_MAX_MINOR_VER) {
51 PANIC("Version not supported! Provided file is %d.%d while Kinoko supports up to %d.%d",
52 testMajorVer, testMinorVer, SUITE_MAJOR_VER, SUITE_MAX_MINOR_VER);
53 }
54
55 for (u16 i = 0; i < numTestCases; ++i) {
56 // Validate alignment
57 if (m_stream.read_u32() != TEST_HEADER_SIGNATURE) {
58 PANIC("Invalid binary data for test case!");
59 }
60
61 u16 totalSize = m_stream.read_u16();
62 TestCase testCase;
63
64 u16 nameLen = m_stream.read_u16();
65 testCase.name = m_stream.read_string();
66 if (nameLen != testCase.name.size() + 1) {
67 PANIC("Test case name length mismatch!");
68 }
69
70 u16 rkgPathLen = m_stream.read_u16();
71 testCase.rkgPath = m_stream.read_string();
72 if (rkgPathLen != testCase.rkgPath.size() + 1) {
73 PANIC("Test case RKG Path length mismatch!");
74 }
75
76 u16 krkgPathLen = m_stream.read_u16();
77 testCase.krkgPath = m_stream.read_string();
78 if (krkgPathLen != testCase.krkgPath.size() + 1) {
79 PANIC("Test case KRKG Path length mismatch!");
80 }
81
82 testCase.targetFrame = m_stream.read_u16();
83
84 // Validate alignment
85 if (m_stream.read_u32() != TEST_FOOTER_SIGNATURE) {
86 PANIC("Invalid binary data for test case!");
87 }
88
89 if (totalSize != sizeof(u16) * 4 + nameLen + rkgPathLen + krkgPathLen) {
90 PANIC("Unexpected bytes in test case");
91 }
92
93 m_testCases.push(testCase);
94 }
95
97 m_sceneMgr->changeScene(0);
98}
99
102 m_sceneMgr->calc();
103}
104
109 bool success = true;
110
111 while (true) {
112 success &= runTest();
113
114 if (!popTestCase()) {
115 break;
116 }
117
118 // TODO: Use a system heap! We currently have a dependency on the scene heap
119 m_sceneMgr->destroyScene(m_sceneMgr->currentScene());
121 m_sceneMgr->createScene(2, m_sceneMgr->currentScene());
122 }
123
124 return success;
125}
126
131void KTestSystem::parseOptions(int argc, char **argv) {
132 if (argc < 2) {
133 PANIC("Expected suite argument!");
134 }
135
136 for (int i = 0; i < argc; ++i) {
137 std::optional<Host::EOption> flag = Host::Option::CheckFlag(argv[i]);
138 if (!flag || *flag == Host::EOption::Invalid) {
139 WARN("Expected a flag! Got: %s", argv[i]);
140 continue;
141 }
142
143 switch (*flag) {
144 case Host::EOption::Suite: {
145 ASSERT(i + 1 < argc);
146
147 size_t size;
148 u8 *data = Abstract::File::Load(argv[++i], size);
149
150 if (size == 0) {
151 PANIC("Failed to load suite data!");
152 }
153
154 m_stream = EGG::RamStream(data, size);
155 m_stream.setEndian(std::endian::big);
156 } break;
157 case Host::EOption::Invalid:
158 default:
159 PANIC("Invalid flag!");
160 break;
161 }
162 }
163}
164
165KTestSystem *KTestSystem::CreateInstance() {
166 ASSERT(!s_instance);
167 s_instance = new KTestSystem;
168 return static_cast<KTestSystem *>(s_instance);
169}
170
171void KTestSystem::DestroyInstance() {
172 ASSERT(s_instance);
173 auto *instance = s_instance;
174 s_instance = nullptr;
175 delete instance;
176}
177
178KTestSystem::KTestSystem() = default;
179
180KTestSystem::~KTestSystem() {
181 if (s_instance) {
182 s_instance = nullptr;
183 WARN("KTestSystem instance not explicitly handled!");
184 }
185}
186
189 constexpr u32 KRKG_SIGNATURE = 0x4b524b47; // KRKG
190
191 size_t size;
192 u8 *krkg = Abstract::File::Load(getCurrentTestCase().krkgPath.data(), size);
193 m_stream = EGG::RamStream(krkg, static_cast<u32>(size));
194 m_currentFrame = -1;
195 m_sync = true;
196
197 // Initialize endianness for the RAM stream
198 u16 mark = *reinterpret_cast<u16 *>(krkg + offsetof(TestHeader, byteOrderMark));
199 std::endian endian = parse<u16>(mark) == 0xfeff ? std::endian::big : std::endian::little;
200 m_stream.setEndian(endian);
201
202 ASSERT(m_stream.read_u32() == KRKG_SIGNATURE);
203 m_stream.skip(2);
204 m_frameCount = m_stream.read_u16();
205 m_versionMajor = m_stream.read_u16();
206 m_versionMinor = m_stream.read_u16();
207
208 ASSERT(m_stream.read_u32() == m_stream.index());
209}
210
214 ASSERT(m_testCases.size() > 0);
215 m_testCases.pop();
216 delete[] m_stream.data();
217
218 return !m_testCases.empty();
219}
220
224 // Check if we're out of frames
225 u16 targetFrame = getCurrentTestCase().targetFrame;
226 ASSERT(targetFrame <= m_frameCount);
227 if (++m_currentFrame > targetFrame) {
228 REPORT("Test Case Passed: %s [%d / %d]", getCurrentTestCase().name.c_str(), targetFrame,
229 m_frameCount);
230 return false;
231 }
232
233 // Test the current frame
235 return m_sync;
236}
237
241 EGG::Vector3f pos;
242 EGG::Quatf fullRot;
243 EGG::Vector3f extVel;
244 EGG::Vector3f intVel;
245 f32 speed = 0.0f;
246 f32 acceleration = 0.0f;
247 f32 softSpeedLimit = 0.0f;
248 EGG::Quatf mainRot;
249 EGG::Vector3f angVel2;
250 f32 raceCompletion = 0.0f;
251 u16 checkpointId = 0;
252 u8 jugemId = 0;
253
254 pos.read(m_stream);
255 fullRot.read(m_stream);
256
257 if (m_versionMinor >= Changelog::AddedExtVel) {
258 extVel.read(m_stream);
259 }
260
261 if (m_versionMinor >= Changelog::AddedIntVel) {
262 intVel.read(m_stream);
263 }
264
265 if (m_versionMinor >= Changelog::AddedSpeed) {
266 speed = m_stream.read_f32();
267 acceleration = m_stream.read_f32();
268 softSpeedLimit = m_stream.read_f32();
269 }
270
271 if (m_versionMinor >= Changelog::AddedRotation) {
272 mainRot.read(m_stream);
273 angVel2.read(m_stream);
274 }
275
276 if (m_versionMinor >= Changelog::AddedCheckpoints) {
277 raceCompletion = m_stream.read_f32();
278 checkpointId = m_stream.read_u16();
279 jugemId = m_stream.read_u8();
280 m_stream.skip(1);
281 }
282
283 TestData data;
284 data.pos = pos;
285 data.fullRot = fullRot;
286 data.extVel = extVel;
287 data.intVel = intVel;
288 data.speed = speed;
289 data.acceleration = acceleration;
290 data.softSpeedLimit = softSpeedLimit;
291 data.mainRot = mainRot;
292 data.angVel2 = angVel2;
293 data.raceCompletion = raceCompletion;
294 data.checkpointId = checkpointId;
295 data.jugemId = jugemId;
296 return data;
297}
298
302 auto *object = Kart::KartObjectManager::Instance()->object(0);
303 const auto &pos = object->pos();
304 const auto &fullRot = object->fullRot();
305 const auto &extVel = object->extVel();
306 const auto &intVel = object->intVel();
307 f32 speed = object->speed();
308 f32 acceleration = object->acceleration();
309 f32 softSpeedLimit = object->softSpeedLimit();
310 const auto &mainRot = object->mainRot();
311 const auto &angVel2 = object->angVel2();
312
313 const auto &player = System::RaceManager::Instance()->player();
314 f32 raceCompletion = player.raceCompletion();
315 u16 checkpointId = player.checkpointId();
316 u8 jugemId = player.jugemId();
317
318 switch (m_versionMinor) {
319 case Changelog::AddedCheckpoints:
320 checkDesync(data.raceCompletion, raceCompletion, "raceCompletion");
321 checkDesync(data.checkpointId, checkpointId, "checkpointId");
322 checkDesync(data.jugemId, jugemId, "jugemId");
323 [[fallthrough]];
324 case Changelog::AddedRotation:
325 checkDesync(data.mainRot, mainRot, "mainRot");
326 checkDesync(data.angVel2, angVel2, "angVel2");
327 [[fallthrough]];
328 case Changelog::AddedSpeed:
329 checkDesync(data.speed, speed, "speed");
330 checkDesync(data.acceleration, acceleration, "acceleration");
331 checkDesync(data.softSpeedLimit, softSpeedLimit, "softSpeedLimit");
332 [[fallthrough]];
333 case Changelog::AddedIntVel:
334 checkDesync(data.intVel, intVel, "intVel");
335 [[fallthrough]];
336 case Changelog::AddedExtVel:
337 checkDesync(data.extVel, extVel, "extVel");
338 [[fallthrough]];
339 default:
340 checkDesync(data.pos, pos, "pos");
341 checkDesync(data.fullRot, fullRot, "fullRot");
342 }
343}
344
349 while (calcTest()) {
350 calc();
351 }
352
353 // TODO: Use a system heap! std::string relies on heap allocation
354 // The heap is destroyed after this and there is no further allocation, so it's not re-disabled
355 m_sceneMgr->currentScene()->heap()->enableAllocation();
357 return m_sync;
358}
359
363 std::string outStr(getCurrentTestCase().name.data());
364 outStr += "\n" + std::string(m_sync ? "1" : "0") + "\n";
365 outStr += std::to_string(getCurrentTestCase().targetFrame) + "\n";
366 outStr += std::to_string(m_frameCount) + "\n";
367 Abstract::File::Append("results.txt", outStr.c_str(), outStr.size());
368}
369
374 ASSERT(!m_testCases.empty());
375 return m_testCases.front();
376}
377
381void KTestSystem::OnInit(System::RaceConfig *config, void * /* arg */) {
382 size_t size;
383 u8 *rkg = Abstract::File::Load(Instance()->getCurrentTestCase().rkgPath.data(), size);
384 config->setGhost(rkg);
385 delete[] rkg;
386
387 config->raceScenario().players[0].type = System::RaceConfig::Player::Type::Ghost;
388}
A stream of data stored in memory.
Definition Stream.hh:64
Manages the scene stack and transitions between scenes.
Kinoko system designed to execute tests.
void init() override
Initializes the system.
void writeTestOutput() const
Writes details about the current test to file.
static void OnInit(System::RaceConfig *config, void *arg)
Initializes the race configuration as needed for test cases.
bool popTestCase()
Pops the current test case and frees the KRKG buffer.
bool run() override
Executes a run.
bool runTest()
Runs a single test case, and ends when the test is finished or when a desync is found.
void startNextTestCase()
Starts the next test case.
void parseOptions(int argc, char **argv) override
Parses non-generic command line options.
void testFrame(const TestData &data)
Tests the frame against the provided test data.
bool calcTest()
Checks one frame in the test.
const TestCase & getCurrentTestCase() const
Gets the current test case.
void calc() override
Executes a frame.
TestData findCurrentFrameEntry()
Finds the test data of the current frame.
Initializes the player with parameters specified in the provided ghost file.
Definition RaceConfig.hh:15
A quaternion, used to represent 3D rotation.
Definition Quat.hh:12
A 3D float vector.
Definition Vector.hh:83
void read(Stream &stream)
Initializes a Vector3f by reading 12 bytes from the stream.
Definition Vector.cc:115