A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
CourseMap.cc
1#include "CourseMap.hh"
2
3#include "game/system/ResourceManager.hh"
4
5namespace System {
6
8void CourseMap::init() {
9 void *buffer = LoadFile("course.kmp");
10 m_course =
11 new MapdataFileAccessor(reinterpret_cast<const MapdataFileAccessor::SData *>(buffer));
12
13 constexpr u32 CANNON_POINT_SIGNATURE = 0x434e5054;
14 constexpr u32 CHECK_PATH_SIGNATURE = 0x434b5048;
15 constexpr u32 CHECK_POINT_SIGNATURE = 0x434b5054;
16 constexpr u32 GEO_OBJ_SIGNATURE = 0x474f424a;
17 constexpr u32 JUGEM_POINT_SIGNATURE = 0x4a475054;
18 constexpr u32 START_POINT_SIGNATURE = 0x4b545054;
19 constexpr u32 POINT_INFO_SIGNATURE = 0x504f5449;
20 constexpr u32 STAGE_INFO_SIGNATURE = 0x53544749;
21
22 m_startPoint = parseMapdata<MapdataStartPointAccessor>(START_POINT_SIGNATURE);
23 m_checkPath = parseMapdata<MapdataCheckPathAccessor>(CHECK_PATH_SIGNATURE);
24 m_checkPoint = parseMapdata<MapdataCheckPointAccessor>(CHECK_POINT_SIGNATURE);
25 m_geoObj = parseMapdata<MapdataGeoObjAccessor>(GEO_OBJ_SIGNATURE);
26 m_pointInfo = parseMapdata<MapdataPointInfoAccessor>(POINT_INFO_SIGNATURE);
27 m_jugemPoint = parseMapdata<MapdataJugemPointAccessor>(JUGEM_POINT_SIGNATURE);
28 m_cannonPoint = parseMapdata<MapdataCannonPointAccessor>(CANNON_POINT_SIGNATURE);
29 m_stageInfo = parseMapdata<MapdataStageInfoAccessor>(STAGE_INFO_SIGNATURE);
30
31 MapdataStageInfo *stageInfo = getStageInfo();
32 constexpr u8 TRANSLATION_MODE_NARROW = 1;
33 if (stageInfo && stageInfo->translationMode() == TRANSLATION_MODE_NARROW) {
34 m_startTmpAngle = 25.0f;
35 m_startTmp2 = 250.0f;
36 m_startTmp3 = 0.0f;
37 } else {
38 m_startTmpAngle = 30.0f;
39 m_startTmp2 = 400.0f;
40 m_startTmp3 = 100.0f;
41 }
42
43 m_startTmp0 = 800.0f;
44 m_startTmp1 = 1200.0f;
45}
46
48s16 CourseMap::findSector(const EGG::Vector3f &pos, u16 checkpointIdx, f32 &distanceRatio) {
49 clearSectorChecked();
50
51 MapdataCheckPoint *checkpoint = getCheckPoint(checkpointIdx);
52 s16 id = -1;
53
55 checkpoint->checkSectorAndDistanceRatio(pos, distanceRatio);
56 checkpoint->setSearched();
57
58 switch (occupancy) {
59 // The player is fully inside the current checkpoint, so just set to current checkpoint
61 id = checkpoint->id();
62 break;
63
64 // The player is between the sides of the quad, but NOT between this checkpoint and
65 // next; player is likely in the same checkpoint group
67 id = findSectorBetweenSides(pos, checkpoint, distanceRatio);
68 break;
69
70 // The player is not between the sides of the quad (may still be between this checkpoint
71 // and next); player is likely in a different checkpoint group
73 id = findSectorOutsideSector(pos, checkpoint, distanceRatio);
74 break;
75
76 default:
77 break;
78 }
79
80 return id > -1 ? id : findSectorRegional(pos, checkpoint, distanceRatio);
81}
82
84s16 CourseMap::findRecursiveSector(const EGG::Vector3f &pos, s16 depth, bool searchBackwardsFirst,
85 MapdataCheckPoint *checkpoint, f32 &distanceRatio, bool playerIsForwards) const {
86 constexpr s16 MAX_DEPTH = 6;
87
88 if (depth >= 0 && depth > MAX_DEPTH) {
89 return -1;
90 }
91
94
95 if (!checkpoint->searched()) {
96 completion = checkpoint->checkSectorAndDistanceRatio(pos, distanceRatio);
97 checkpoint->setSearched();
98 }
99
100 // If player is inside current checkpoint, stop searching
102 return checkpoint->id();
103 }
104
105 // Search type 0: Search forwards first, then backwards
106 if (!searchBackwardsFirst) {
107 // If "player is forwards" but completion < 0, force completion to 0 and return
108 // current checkpoint (GHOST CHECKPOINT!)
109 if (playerIsForwards && completion == MapdataCheckPoint::SectorOccupancy::BetweenSides &&
110 distanceRatio < 0.0f) {
111 distanceRatio = 0.0f;
112 return checkpoint->id();
113 }
114
115 // Stop if current checkpoint is a KCP
116 if (checkpoint->checkArea() >= 0) {
117 return -1;
118 }
119
120 // If player is between the sides of the quad but NOT between this checkpoint and next, AND
121 // completion > 0, then "player is forwards"
122 bool forward = completion == MapdataCheckPoint::SectorOccupancy::BetweenSides &&
123 distanceRatio > 0.0f;
124
125 // Search forwards, including checkpoints already searched
126 s16 id = searchNextCheckpoint(pos, depth, checkpoint, distanceRatio, forward, false);
127
128 // If that fails, search backwards, excluding checkpoints already searched
129 return id == -1 ?
130 searchPrevCheckpoint(pos, depth, checkpoint, distanceRatio, forward, true) :
131 id;
132 }
133
134 // Search type 1: Search backwards first, then forwards
135
136 // If "player is backwards" flag is true but completion > 1, force completion to 1 and return
137 // current checkpoint (GHOST CHECKPOINT!)
138 if (playerIsForwards && completion == MapdataCheckPoint::SectorOccupancy::BetweenSides &&
139 distanceRatio > 1.0f) {
140 distanceRatio = 1.0f;
141 return checkpoint->id();
142 }
143
144 // Stop if current checkpoint is a KCP (skipped for online players, but they aren't supported)
145 if (checkpoint->checkArea() >= 0) {
146 return -1;
147 }
148
149 // If player is between the sides of the quad but NOT between this checkpoint and next, AND
150 // completion < 0, then set "player is backwards" flag
151 bool forward =
152 completion == MapdataCheckPoint::SectorOccupancy::BetweenSides && distanceRatio < 0.0f;
153
154 // Search backwards, including checkpoints already searched
155 s16 id = searchPrevCheckpoint(pos, depth, checkpoint, distanceRatio, forward, false);
156
157 // If that fails, search forwards, excluding checkpoints already searched
158 return id == -1 ? searchNextCheckpoint(pos, depth, checkpoint, distanceRatio, forward, true) :
159 id;
160}
161
163u16 CourseMap::getCheckPointEntryOffsetMs(u16 i, const EGG::Vector3f &pos,
164 const EGG::Vector3f &prevPos) const {
165 EGG::Vector2f prevPos_ = EGG::Vector2f(prevPos.x, prevPos.z);
166 EGG::Vector2f pos_ = EGG::Vector2f(pos.x, pos.z);
167
168 MapdataCheckPoint *checkPoint = getCheckPoint(i);
169 ASSERT(checkPoint);
170 return checkPoint->getEntryOffsetMs(prevPos_, pos_);
171}
172
173f32 CourseMap::getCheckPointEntryOffsetExact(u16 i, const EGG::Vector3f &pos,
174 const EGG::Vector3f &prevPos) const {
175 EGG::Vector2f prevPos_ = EGG::Vector2f(prevPos.x, prevPos.z);
176 EGG::Vector2f pos_ = EGG::Vector2f(pos.x, pos.z);
177
178 MapdataCheckPoint *checkPoint = getCheckPoint(i);
179 ASSERT(checkPoint);
180 return checkPoint->getEntryOffsetExact(prevPos_, pos_);
181}
182
184CourseMap *CourseMap::CreateInstance() {
185 ASSERT(!s_instance);
186 s_instance = new CourseMap;
187 return s_instance;
188}
189
191void CourseMap::DestroyInstance() {
192 ASSERT(s_instance);
193 auto *instance = s_instance;
194 s_instance = nullptr;
195 delete instance;
196}
197
199CourseMap::CourseMap()
200 : m_course(nullptr), m_startPoint(nullptr), m_stageInfo(nullptr), m_startTmpAngle(0.0f),
201 m_startTmp0(0.0f), m_startTmp1(0.0f), m_startTmp2(0.0f), m_startTmp3(0.0f) {}
202
204CourseMap::~CourseMap() {
205 if (s_instance) {
206 s_instance = nullptr;
207 WARN("CourseMap instance not explicitly handled!");
208 }
209
210 delete m_course;
211 delete m_startPoint;
212 delete m_checkPath;
213 delete m_checkPoint;
214 delete m_pointInfo;
215 delete m_geoObj;
216 delete m_jugemPoint;
217 delete m_cannonPoint;
218 delete m_stageInfo;
219}
220
221s16 CourseMap::findSectorBetweenSides(const EGG::Vector3f &pos, MapdataCheckPoint *checkpoint,
222 f32 &distanceRatio) {
223 s16 id = -1;
224
225 // Search order varies depending on whether player is closer to the next or previous checkpoint.
226 if (distanceRatio > 0.5f) {
227 // Step 1: Starting at current checkpoint, search forwards
228 id = searchNextCheckpoint(pos, 0, checkpoint, distanceRatio, false, false);
229
230 if (id != -1) {
231 return id;
232 }
233
234 // Step 2: If step 1 fails, start at next checkpoint(s) and search backwards
235 for (size_t i = 0; i < checkpoint->nextCount(); ++i) {
236 MapdataCheckPoint *next = checkpoint->nextPoint(i);
237
238 for (size_t j = 0; j < next->prevCount(); ++j) {
239 MapdataCheckPoint *prev = next->prevPoint(j);
240
241 if (prev == checkpoint) {
242 continue;
243 }
244
245 id = findRecursiveSector(pos, 1, true, prev, distanceRatio, false);
246 if (id != -1) {
247 return id;
248 }
249 }
250 }
251
252 // Step 3: If step 2 fails, start at previous checkpoint(s) and search forwards
253 for (size_t i = 0; i < checkpoint->prevCount(); ++i) {
254 MapdataCheckPoint *prev = checkpoint->prevPoint(i);
255
256 for (size_t j = 0; j < prev->nextCount(); ++j) {
257 MapdataCheckPoint *next = prev->nextPoint(j);
258
259 if (next == checkpoint) {
260 continue;
261 }
262
263 id = findRecursiveSector(pos, 1, false, next, distanceRatio, false);
264 if (id != -1) {
265 return id;
266 }
267 }
268 }
269
270 // Step 4: If step 3 fails, start at current checkpoint and search backwards
271 return searchPrevCheckpoint(pos, 0, checkpoint, distanceRatio, false, false);
272 } else {
273 // Step 1: Starting at current checkpoint, search backwards
274 id = searchPrevCheckpoint(pos, 0, checkpoint, distanceRatio, false, false);
275
276 if (id != -1) {
277 return id;
278 }
279
280 // Step 2: If step 1 fails, start at prev checkpoint(s) and search forwards
281 for (size_t i = 0; i < checkpoint->prevCount(); ++i) {
282 MapdataCheckPoint *prev = checkpoint->prevPoint(i);
283
284 for (size_t j = 0; j < prev->nextCount(); ++j) {
285 MapdataCheckPoint *next = prev->nextPoint(j);
286
287 if (next == checkpoint) {
288 continue;
289 }
290
291 id = findRecursiveSector(pos, 1, false, next, distanceRatio, false);
292
293 if (id != -1) {
294 return id;
295 }
296 }
297 }
298
299 // Step 3: If step 2 fails, start at next checkpoint(s) and search backwards
300 for (size_t i = 0; i < checkpoint->nextCount(); ++i) {
301 MapdataCheckPoint *next = checkpoint->nextPoint(i);
302
303 for (size_t j = 0; j < next->prevCount(); ++j) {
304 MapdataCheckPoint *prev = next->prevPoint(j);
305
306 if (prev == checkpoint) {
307 continue;
308 }
309
310 id = findRecursiveSector(pos, 1, true, prev, distanceRatio, false);
311
312 if (id != -1) {
313 return id;
314 }
315 }
316 }
317
318 // Step 4: If step 3 fails, start at current checkpoint and search forwards
319 return searchNextCheckpoint(pos, 0, checkpoint, distanceRatio, false, false);
320 }
321
322 return id;
323}
324
325s16 CourseMap::findSectorOutsideSector(const EGG::Vector3f &pos, MapdataCheckPoint *checkpoint,
326 f32 &distanceRatio) {
327 s16 id = -1;
328
329 // Step 1: Starting at next checkpoint(s), search backwards
330 for (size_t i = 0; i < checkpoint->nextCount(); ++i) {
331 MapdataCheckPoint *next = checkpoint->nextPoint(i);
332
333 for (size_t j = 0; j < next->prevCount(); ++j) {
334 MapdataCheckPoint *prev = next->prevPoint(j);
335
336 if (prev == checkpoint) {
337 continue;
338 }
339
340 id = findRecursiveSector(pos, 1, true, prev, distanceRatio, false);
341
342 if (id != -1) {
343 return id;
344 }
345 }
346 }
347
348 // Step 2: If step 1 fails, start at prev checkpoint(s) and search forwards
349 for (size_t i = 0; i < checkpoint->prevCount(); ++i) {
350 MapdataCheckPoint *prev = checkpoint->prevPoint(i);
351
352 for (size_t j = 0; j < prev->nextCount(); ++j) {
353 MapdataCheckPoint *next = prev->nextPoint(j);
354
355 if (next == checkpoint) {
356 continue;
357 }
358
359 id = findRecursiveSector(pos, 1, false, next, distanceRatio, false);
360
361 if (id != -1) {
362 return id;
363 }
364 }
365 }
366
367 // Step 3: If step 2 fails, start at next checkpoint(s) and search forwards
368 for (size_t i = 0; i < checkpoint->nextCount(); ++i) {
369 id = findRecursiveSector(pos, 1, false, checkpoint->nextPoint(i), distanceRatio, false);
370
371 if (id != -1) {
372 return id;
373 }
374 }
375
376 // Step 4: If step 3 fails, start at prev checkpoint(s) and search backwards
377 for (size_t i = 0; i < checkpoint->prevCount(); ++i) {
378 id = findRecursiveSector(pos, 1, true, checkpoint->prevPoint(i), distanceRatio, false);
379
380 if (id != -1) {
381 return id;
382 }
383 }
384
385 return id;
386}
387
388// If local search fails, remove depth limit and search all "loaded" checkpoints
389s16 CourseMap::findSectorRegional(const EGG::Vector3f &pos, MapdataCheckPoint *checkpoint,
390 f32 &distanceRatio) {
391 s16 id = -1;
392
393 // Step 1: Search all next checkpoints until player or key checkpoint is found
394 for (size_t i = 0; i < checkpoint->nextCount(); ++i) {
395 id = findRecursiveSector(pos, -1, false, checkpoint->nextPoint(i), distanceRatio, false);
396
397 if (id != -1) {
398 return id;
399 }
400 }
401
402 // Step 2: Search all previous checkpoints until player or key checkpoint is found
403 for (size_t i = 0; i < checkpoint->prevCount(); ++i) {
404 id = findRecursiveSector(pos, -1, true, checkpoint->prevPoint(i), distanceRatio, false);
405
406 if (id != -1) {
407 return id;
408 }
409 }
410
411 return id;
412}
413
415s16 CourseMap::searchNextCheckpoint(const EGG::Vector3f &pos, s16 depth,
416 const MapdataCheckPoint *checkpoint, f32 &completion, bool playerIsForwards,
417 bool useCache) const {
418 s16 id = -1;
419 depth = depth >= 0 ? depth + 1 : -1;
420
421 for (size_t i = 0; i < checkpoint->nextCount(); ++i) {
422 MapdataCheckPoint *next = checkpoint->nextPoint(i);
423
424 if (!useCache || !next->searched()) {
425 id = findRecursiveSector(pos, depth, false, next, completion, playerIsForwards);
426
427 if (id != -1) {
428 return id;
429 }
430 }
431 }
432
433 return id;
434}
435
437s16 CourseMap::searchPrevCheckpoint(const EGG::Vector3f &pos, s16 depth,
438 const MapdataCheckPoint *checkpoint, f32 &completion, bool playerIsForwards,
439 bool useCache) const {
440 s16 id = -1;
441 depth = depth >= 0 ? depth + 1 : -1;
442
443 for (size_t i = 0; i < checkpoint->prevCount(); ++i) {
444 MapdataCheckPoint *prev = checkpoint->prevPoint(i);
445
446 if (!useCache || !prev->searched()) {
447 id = findRecursiveSector(pos, depth, true, prev, completion, playerIsForwards);
448
449 if (id != -1) {
450 return id;
451 }
452 }
453 }
454
455 return id;
456}
457
459void CourseMap::clearSectorChecked() {
460 for (size_t i = 0; i < m_checkPoint->size(); ++i) {
461 getCheckPoint(i)->clearSearched();
462 }
463}
464
466void *CourseMap::LoadFile(const char *filename) {
467 return ResourceManager::Instance()->getFile(filename, nullptr, ArchiveId::Course);
468}
469
470CourseMap *CourseMap::s_instance = nullptr;
471
472} // namespace System
Contains course metadata, notably the starting position.
Definition CourseMap.hh:26
f32 getEntryOffsetExact(const EGG::Vector2f &prevPos, const EGG::Vector2f &pos) const
Finds the offset between the two positions that enter the checkpoint.
u16 getEntryOffsetMs(const EGG::Vector2f &prevPos, const EGG::Vector2f &pos) const
Finds the offset between the two positions that enter the checkpoint.
@ OutsideSector
Player is outside the given checkpoint group.
@ InsideSector
Player is inside the given checkpoint group.
High-level handling for generic system operations, such as input reading, race configuration,...
Definition CourseMap.cc:5
A 2D float vector.
Definition Vector.hh:12
A 3D float vector.
Definition Vector.hh:83