A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
BoxColManager.cc
1#include "BoxColManager.hh"
2
3#include "game/field/obj/ObjectCollidable.hh"
4#include "game/field/obj/ObjectDrivable.hh"
5
6#include <numeric>
7
8namespace Field {
9
11BoxColUnit::BoxColUnit() : m_pos(nullptr), m_radius(0.0f), m_range(0.0f), m_userData(nullptr) {}
12
14BoxColUnit::~BoxColUnit() = default;
15
17void BoxColUnit::init(f32 radius, f32 maxSpeed, const EGG::Vector3f *pos, const BoxColFlag &flag,
18 void *userData) {
19 m_pos = pos;
20 m_radius = radius;
21 m_range = radius + maxSpeed;
22 m_flag = flag;
23 m_flag.setBit(eBoxColFlag::Active);
24 m_userData = userData;
25 m_xMax = pos->x + m_range;
26 m_xMin = pos->x - m_range;
27}
28
30void BoxColUnit::makeInactive() {
31 m_flag.resetBit(eBoxColFlag::Active);
32}
33
35void BoxColUnit::resize(f32 radius, f32 maxSpeed) {
36 m_radius = radius;
37 m_range = radius + maxSpeed;
39}
40
42void BoxColUnit::reinsert() {
43 BoxColManager::Instance()->reinsertUnit(this);
44}
45
47void BoxColUnit::search(const BoxColFlag &flag) {
48 BoxColManager::Instance()->search(this, flag);
49}
50
54 constexpr f32 SPATIAL_BOUND = 999999.9f;
55
56 static const EGG::Vector3f upperBound(SPATIAL_BOUND, SPATIAL_BOUND, SPATIAL_BOUND);
57 static const EGG::Vector3f lowerBound(-SPATIAL_BOUND, -SPATIAL_BOUND, -SPATIAL_BOUND);
58
59 std::iota(m_unitIDs.begin(), m_unitIDs.end(), 1);
60
61 m_unitCount = 0;
62 m_nextUnitID = 0;
63
64 clear();
65
66 BoxColFlag flags;
67 insert(1.0f, 0.0f, &upperBound, flags, nullptr)->m_flag.setBit(eBoxColFlag::Intangible);
68 insert(1.0f, 0.0f, &lowerBound, flags, nullptr)->m_flag.setBit(eBoxColFlag::Intangible);
69}
70
72BoxColManager::~BoxColManager() {
73 if (s_instance) {
74 s_instance = nullptr;
75 WARN("BoxColManager instance not explicitly handled!");
76 }
77}
78
80void BoxColManager::clear() {
81 m_nextObjectID = MAX_UNIT_COUNT;
82 m_nextDrivableID = MAX_UNIT_COUNT;
83 m_maxID = 0;
84 m_cacheQueryUnit = nullptr;
85 m_cacheRadius = -1.0f;
86 m_cacheFlag.makeAllZero();
87}
88
93 clear();
94
95 // Update the AABB if necessary
96 int activeUnitIdx = 0;
97 for (auto &unit : m_unitPool) {
98 // Unit is not active, so it's not factored into mUnitCount, skip it
99 if (unit.m_flag.offBit(eBoxColFlag::Active)) {
100 continue;
101 }
102
103 // Unit is active, but we don't need to recalc its AABB, move to the next unit
105 unit.m_xMax = unit.m_pos->x + unit.m_range;
106 unit.m_xMin = unit.m_pos->x - unit.m_range;
107 m_highPoints[unit.m_highPointIdx].z = unit.m_pos->z + unit.m_range;
108 m_lowPoints[unit.m_lowPointIdx].z = unit.m_pos->z - unit.m_range;
109
110 unit.m_flag.resetBit(eBoxColFlag::TempRecalcAABB);
111 }
112
113 // We're done with all of the active units, so we avoid iterating over the remaining
114 // inactive units
115 if (++activeUnitIdx >= m_unitCount) {
116 break;
117 }
118 }
119
120 // The loops use insertion sort.
121 // We assume the units in the spatial index only change in small portions at a time per frame.
122 // The time complexity is closer to O(n) as a result rather than O(n^2).
123
124 // Reorganize the high points
125 for (size_t i = 1; i < static_cast<size_t>(m_unitCount); ++i) {
126 for (size_t j = i; m_highPoints[j - 1].z > m_highPoints[j].z && j >= 1; --j) {
127 BoxColHighPoint &upper = m_highPoints[j];
128 BoxColHighPoint &lower = m_highPoints[j - 1];
129
130 std::swap(upper, lower);
131
132 BoxColLowPoint &upperLow = m_lowPoints[upper.lowPoint];
133 BoxColLowPoint &lowerLow = m_lowPoints[lower.lowPoint];
134 ++upperLow.highPoint;
135 --lowerLow.highPoint;
136
137 ++m_unitPool[upperLow.unitID].m_highPointIdx;
138 --m_unitPool[lowerLow.unitID].m_highPointIdx;
139
140 u8 &nextMinLowPoint = upper.minLowPoint;
141 if (nextMinLowPoint == lower.lowPoint) {
142 do {
143 ++nextMinLowPoint;
144 } while (m_lowPoints[nextMinLowPoint].highPoint < j);
145 }
146
147 lower.minLowPoint = std::min(lower.minLowPoint, upper.lowPoint);
148 }
149 }
150
151 // Reorganize the low points
152 for (size_t i = 1; i < static_cast<size_t>(m_unitCount); ++i) {
153 for (size_t j = i; m_lowPoints[j - 1].z > m_lowPoints[j].z && j >= 1; --j) {
154 BoxColLowPoint &upper = m_lowPoints[j];
155 BoxColLowPoint &lower = m_lowPoints[j - 1];
156
157 std::swap(upper, lower);
158
159 ++m_highPoints[upper.highPoint].lowPoint;
160 --m_highPoints[lower.highPoint].lowPoint;
161
162 ++m_unitPool[upper.unitID].m_lowPointIdx;
163 --m_unitPool[lower.unitID].m_lowPointIdx;
164
165 if (lower.highPoint > upper.highPoint) {
166 int k = upper.highPoint;
167
168 while (k > lower.highPoint && m_highPoints[k].minLowPoint == j - 1) {
169 ++m_highPoints[k--].minLowPoint;
170 }
171 } else {
172 int k = lower.highPoint;
173
174 while (k > upper.highPoint && m_highPoints[k].minLowPoint == j) {
175 --m_highPoints[k--].minLowPoint;
176 }
177 }
178 }
179 }
180}
181
183ObjectCollidable *BoxColManager::getNextObject() {
184 return reinterpret_cast<ObjectCollidable *>(getNextImpl(m_nextObjectID, eBoxColFlag::Object));
185}
186
188ObjectDrivable *BoxColManager::getNextDrivable() {
189 return reinterpret_cast<ObjectDrivable *>(getNextImpl(m_nextDrivableID, eBoxColFlag::Drivable));
190}
191
193void BoxColManager::resetIterators() {
194 m_nextObjectID = -1;
195 iterate(m_nextObjectID, eBoxColFlag::Object);
196
197 m_nextDrivableID = -1;
198 iterate(m_nextDrivableID, eBoxColFlag::Drivable);
199}
200
202BoxColUnit *BoxColManager::insertDriver(f32 radius, f32 maxSpeed, const EGG::Vector3f *pos,
203 bool alwaysRecalc, Kart::KartObject *kartObject) {
204 BoxColFlag flag = BoxColFlag(eBoxColFlag::Driver);
205
206 if (alwaysRecalc) {
207 flag.setBit(eBoxColFlag::PermRecalcAABB);
208 }
209
210 return insert(radius, maxSpeed, pos, flag, kartObject);
211}
212
214BoxColUnit *BoxColManager::insertObject(f32 radius, f32 maxSpeed, const EGG::Vector3f *pos,
215 bool alwaysRecalc, void *userData) {
216 BoxColFlag flag = BoxColFlag(eBoxColFlag::Object);
217
218 if (alwaysRecalc) {
219 flag.setBit(eBoxColFlag::PermRecalcAABB);
220 }
221
222 return insert(radius, maxSpeed, pos, flag, userData);
223}
224
226BoxColUnit *BoxColManager::insertDrivable(f32 radius, f32 maxSpeed, const EGG::Vector3f *pos,
227 bool alwaysRecalc, void *userData) {
228 BoxColFlag flag = BoxColFlag(eBoxColFlag::Drivable);
229
230 if (alwaysRecalc) {
231 flag.setBit(eBoxColFlag::PermRecalcAABB);
232 }
233
234 return insert(radius, maxSpeed, pos, flag, userData);
235}
236
238void BoxColManager::reinsertUnit(BoxColUnit *unit) {
239 f32 radius = unit->m_radius;
240 f32 maxSpeed = unit->m_range - radius;
241 const EGG::Vector3f *pos = unit->m_pos;
242 BoxColFlag flag = unit->m_flag;
243 void *userData = unit->m_userData;
244
245 remove(unit);
246 insert(radius, maxSpeed, pos, BoxColFlag(), userData)->m_flag = flag;
247}
248
250void BoxColManager::search(BoxColUnit *unit, const BoxColFlag &flag) {
251 searchImpl(unit, flag);
252 resetIterators();
253}
254
256void BoxColManager::search(f32 radius, const EGG::Vector3f &pos, const BoxColFlag &flag) {
257 searchImpl(radius, pos, flag);
258 resetIterators();
259}
260
262bool BoxColManager::isSphereInSpatialCache(f32 radius, const EGG::Vector3f &pos,
263 const BoxColFlag &flag) const {
264 if (m_cacheRadius == -1.0f) {
265 return false;
266 }
267
268 if (!m_cacheFlag.onAll(flag)) {
269 return false;
270 }
271
272 f32 radiusDiff = m_cacheRadius - radius;
273 EGG::Vector3f posDiff = pos - m_cachePoint;
274
275 return EGG::Mathf::abs(posDiff.x) <= radiusDiff && EGG::Mathf::abs(posDiff.z) <= radiusDiff;
276}
277
279BoxColManager *BoxColManager::CreateInstance() {
280 ASSERT(!s_instance);
281 s_instance = new BoxColManager;
282 return s_instance;
283}
284
286void BoxColManager::DestroyInstance() {
287 ASSERT(s_instance);
288 auto *instance = s_instance;
289 s_instance = nullptr;
290 delete instance;
291}
292
293BoxColManager *BoxColManager::Instance() {
294 return s_instance;
295}
296
298void *BoxColManager::getNextImpl(s32 &id, const BoxColFlag &flag) {
299 if (id == MAX_UNIT_COUNT) {
300 return nullptr;
301 }
302
303 BoxColUnit *unit = m_units[id];
304 iterate(id, flag);
305
306 return unit->m_userData;
307}
308
310void BoxColManager::iterate(s32 &iter, const BoxColFlag &flag) {
311 while (++iter < m_maxID) {
312 if (m_units[iter]->m_flag.on(flag)) {
313 return;
314 }
315 }
316
317 iter = MAX_UNIT_COUNT;
318}
319
321BoxColUnit *BoxColManager::insert(f32 radius, f32 maxSpeed, const EGG::Vector3f *pos,
322 const BoxColFlag &flag, void *userData) {
323 if (m_unitCount >= static_cast<s32>(MAX_UNIT_COUNT)) {
324 return nullptr;
325 }
326
327 s32 unitID = m_nextUnitID;
328 BoxColUnit &unit = m_unitPool[unitID];
329 unit.init(radius, maxSpeed, pos, flag, userData);
330 m_nextUnitID = m_unitIDs[unitID];
331 f32 range = radius + maxSpeed;
332 f32 zHigh = pos->z + range;
333 f32 zLow = pos->z - range;
334
335 if (m_unitCount == 0) {
336 m_highPoints[0].lowPoint = 0;
337 m_highPoints[0].minLowPoint = 0;
338 m_lowPoints[0].unitID = unitID;
339 m_highPoints[0].z = zHigh;
340 m_lowPoints[0].highPoint = 0;
341 m_lowPoints[0].unitID = unitID;
342 m_lowPoints[0].z = zLow;
343 m_unitPool[0].m_highPointIdx = 0;
344 m_unitPool[0].m_lowPointIdx = 0;
345 m_unitCount = 1;
346
347 return &unit;
348 }
349
350 // Binary search
351 int highPointIdx = 0;
352 int lowPointIdx = 0;
353 int i = m_unitCount;
354
355 do {
356 int highSearch = highPointIdx + i;
357 int lowSearch = lowPointIdx + i;
358
359 if (highSearch <= m_unitCount && zHigh > m_highPoints[highSearch - 1].z) {
360 highPointIdx = highSearch;
361 }
362
363 if (lowSearch <= m_unitCount && zLow > m_lowPoints[lowSearch - 1].z) {
364 lowPointIdx = lowSearch;
365 }
366
367 i = (i + 1) / 2;
368 } while (i > 1);
369
370 unit.m_highPointIdx = highPointIdx;
371 unit.m_lowPointIdx = lowPointIdx;
372
373 // Update high points
374 for (int i = m_unitCount; i > highPointIdx; --i) {
375 BoxColHighPoint &high = m_highPoints[i];
376 m_highPoints[i] = m_highPoints[i - 1];
377 BoxColLowPoint &low = m_lowPoints[high.lowPoint];
378
379 ++low.highPoint;
380 ++m_unitPool[low.unitID].m_highPointIdx;
381
382 if (high.minLowPoint >= lowPointIdx) {
383 ++high.minLowPoint;
384 }
385 }
386
387 m_highPoints[highPointIdx].lowPoint = lowPointIdx;
388 m_highPoints[highPointIdx].z = zHigh;
389
390 // Update min low point
391 if (highPointIdx == m_unitCount || m_highPoints[highPointIdx + 1].minLowPoint > lowPointIdx) {
392 m_highPoints[highPointIdx].minLowPoint = lowPointIdx;
393
394 for (int i = highPointIdx - 1; i >= 0 && m_highPoints[i].minLowPoint > lowPointIdx; --i) {
395 m_highPoints[i].minLowPoint = lowPointIdx;
396 }
397 } else {
398 m_highPoints[highPointIdx].minLowPoint = m_highPoints[highPointIdx + 1].minLowPoint;
399 }
400
401 // Update low points
402 for (int i = m_unitCount; i > lowPointIdx; --i) {
403 BoxColLowPoint &low = m_lowPoints[i];
404 m_lowPoints[i] = m_lowPoints[i - 1];
405 BoxColHighPoint &high = m_highPoints[low.highPoint];
406 ++high.lowPoint;
407 ++m_unitPool[low.unitID].m_lowPointIdx;
408 }
409
410 m_lowPoints[lowPointIdx].highPoint = highPointIdx;
411 m_lowPoints[lowPointIdx].unitID = unitID;
412 m_lowPoints[lowPointIdx].z = zLow;
413 ++m_unitCount;
414
415 return &unit;
416}
417
419void BoxColManager::remove(BoxColUnit *&pUnit) {
420 if (!pUnit || pUnit->m_flag.offBit(eBoxColFlag::Active)) {
421 return;
422 }
423
424 int highPointIdx = pUnit->m_highPointIdx;
425 int lowPointIdx = pUnit->m_lowPointIdx;
426
427 // Update high points
428 for (int i = highPointIdx; i < m_unitCount - 1; ++i) {
429 BoxColHighPoint &high = m_highPoints[i];
430 m_highPoints[i] = m_highPoints[i + 1];
431 BoxColLowPoint &low = m_lowPoints[high.lowPoint];
432 --low.highPoint;
433 --m_unitPool[low.unitID].m_highPointIdx;
434 if (high.minLowPoint > lowPointIdx) {
435 --high.minLowPoint;
436 }
437 }
438
439 // Update low points
440 for (int i = lowPointIdx; i < m_unitCount - 1; ++i) {
441 BoxColLowPoint &low = m_lowPoints[i];
442 m_lowPoints[i] = m_lowPoints[i + 1];
443 BoxColHighPoint &high = m_highPoints[low.highPoint];
444 --high.lowPoint;
445 --m_unitPool[low.unitID].m_lowPointIdx;
446
447 if (low.highPoint >= highPointIdx) {
448 continue;
449 }
450
451 int minLowPoint = m_highPoints[low.highPoint].minLowPoint;
452
453 if (minLowPoint != lowPointIdx) {
454 continue;
455 }
456
457 for (BoxColLowPoint *pLowPoint = &m_lowPoints[minLowPoint];
458 pLowPoint->highPoint < low.highPoint; ++minLowPoint) {
459 ++pLowPoint;
460 }
461
462 m_highPoints[low.highPoint].minLowPoint = minLowPoint;
463 }
464
465 pUnit->makeInactive();
466 int nextID = pUnit - m_unitPool.data();
467 m_unitIDs[nextID] = m_nextUnitID;
468 m_nextUnitID = nextID;
469 --m_unitCount;
470 pUnit = nullptr;
471}
472
474void BoxColManager::searchImpl(BoxColUnit *unit, const BoxColFlag &flag) {
475 if (unit->m_flag.offBit(eBoxColFlag::Active)) {
476 return;
477 }
478
479 int highPointIdx = unit->m_highPointIdx;
480 int lowPointIdx = unit->m_lowPointIdx;
481 int origLowPointIdx = unit->m_lowPointIdx;
482
483 f32 highZPos = m_highPoints[highPointIdx].z;
484 f32 lowZPos = m_lowPoints[origLowPointIdx].z;
485
486 f32 xMax = unit->m_xMax;
487 f32 xMin = unit->m_xMin;
488
489 const EGG::Vector3f *pos = unit->m_pos;
490 f32 radius = unit->m_radius;
491
492 f32 zHigh = pos->z + radius;
493 f32 zLow = pos->z - radius;
494 f32 xHigh = pos->x + radius;
495 f32 xLow = pos->x - radius;
496
497 int maxIdx = m_unitCount - 1;
498
499 m_maxID = 0;
500 m_cacheQueryUnit = unit;
501 m_cacheRadius = -1.0f;
502 m_cacheFlag = flag;
503
504 for (; highPointIdx > 7 && m_highPoints[highPointIdx - 8].z >= lowZPos;) {
505 highPointIdx -= 8;
506 }
507
508 for (; highPointIdx > 0 && m_highPoints[highPointIdx - 1].z >= lowZPos;) {
509 --highPointIdx;
510 }
511
512 for (; lowPointIdx < maxIdx - 7 && m_lowPoints[lowPointIdx + 8].z <= highZPos;) {
513 lowPointIdx += 8;
514 }
515
516 for (; lowPointIdx < maxIdx && m_lowPoints[lowPointIdx + 1].z <= highZPos;) {
517 ++lowPointIdx;
518 }
519
520 u8 minLowPoint = m_highPoints[highPointIdx].minLowPoint;
521
522 for (int i = lowPointIdx; i >= minLowPoint; --i, --lowPointIdx) {
523 BoxColLowPoint &low = m_lowPoints[i];
524
525 if (low.highPoint >= highPointIdx && lowPointIdx != origLowPointIdx) {
526 BoxColUnit &lowUnit = m_unitPool[low.unitID];
527
528 if (lowUnit.m_xMax < xMin || lowUnit.m_xMin > xMax) {
529 continue;
530 }
531
532 if (lowUnit.m_flag.off(flag) || lowUnit.m_flag.onBit(eBoxColFlag::Intangible)) {
533 continue;
534 }
535
536 f32 radius = lowUnit.m_radius;
537 if (lowUnit.m_pos->z + radius < zLow || lowUnit.m_pos->z - radius > zHigh) {
538 continue;
539 }
540
541 if (lowUnit.m_pos->x + radius < xLow || lowUnit.m_pos->x - radius > xHigh) {
542 continue;
543 }
544
545 m_units[m_maxID++] = &lowUnit;
546
547 if (m_maxID == MAX_UNIT_COUNT) {
548 break;
549 }
550 }
551
552 if (lowPointIdx == 0) {
553 break;
554 }
555 }
556}
557
559void BoxColManager::searchImpl(f32 radius, const EGG::Vector3f &pos, const BoxColFlag &flag) {
560 // Binary search
561 int highPointIdx = 0;
562 int lowPointIdx = 0;
563 f32 zHigh = pos.z + radius;
564 f32 zLow = pos.z - radius;
565 f32 xHigh = pos.x + radius;
566 f32 xLow = pos.x - radius;
567
568 m_maxID = 0;
569 m_cacheQueryUnit = nullptr;
570 m_cachePoint = pos;
571 m_cacheRadius = radius;
572 m_cacheFlag = flag;
573
574 int i = m_unitCount - 1;
575 do {
576 int highSearch = highPointIdx + i;
577 int lowSearch = lowPointIdx + i;
578 if (highSearch <= m_unitCount && zLow > m_highPoints[highSearch - 1].z) {
579 highPointIdx = highSearch;
580 }
581
582 if (lowSearch <= m_unitCount && zHigh >= m_lowPoints[lowSearch].z) {
583 lowPointIdx = lowSearch;
584 }
585
586 i = (i + 1) / 2;
587 } while (i > 1);
588
589 u8 minLowPoint = m_highPoints[highPointIdx].minLowPoint;
590
591 for (i = lowPointIdx; i >= minLowPoint; --i, --lowPointIdx) {
592 BoxColLowPoint &low = m_lowPoints[i];
593 if (low.highPoint >= highPointIdx) {
594 BoxColUnit &unit = m_unitPool[low.unitID];
595
596 if (unit.m_xMax < xLow || unit.m_xMin > xHigh) {
597 continue;
598 }
599
600 if (unit.m_flag.off(flag) || unit.m_flag.onBit(eBoxColFlag::Intangible)) {
601 continue;
602 }
603
604 m_units[m_maxID++] = &unit;
605
606 if (m_maxID == MAX_UNIT_COUNT) {
607 break;
608 }
609 }
610
611 if (lowPointIdx == 0) {
612 break;
613 }
614 }
615}
616
617BoxColManager *BoxColManager::s_instance = nullptr;
618
619} // namespace Field
std::array< BoxColUnit *, MAX_UNIT_COUNT > m_units
Units within our search bounds.
void * getNextImpl(s32 &id, const BoxColFlag &flag)
Helper function since the getters share all code except the flag.
std::array< BoxColHighPoint, MAX_UNIT_COUNT > m_highPoints
A unit's rightmost Z-axis point.
BoxColManager()
Creates two intangible units to represent the spatial bounds.
std::array< BoxColLowPoint, MAX_UNIT_COUNT > m_lowPoints
A unit's leftmost Z-axis point;.
std::array< u32, MAX_UNIT_COUNT > m_unitIDs
Specifies what unit to retrieve from the pool during allocation.
std::array< BoxColUnit, MAX_UNIT_COUNT > m_unitPool
Where all the units live.
void calc()
Recalculate the bounds of all active units having PermRecalcAABB or TempRecalcAABB flag,...
The highest level abstraction for a kart.
Definition KartObject.hh:11
Pertains to collision.
@ TempRecalcAABB
Only recalculate once.
@ PermRecalcAABB
Recalculate this unit's spatial indexing every frame.
@ Intangible
Ignore collision with the unit.
constexpr TBitFlag< T, E > & resetBit(Es... es)
Resets the corresponding bits for the provided enum values.
Definition BitFlag.hh:68
constexpr void makeAllZero()
Resets all the bits to zero.
Definition BitFlag.hh:185
constexpr bool onAll(T mask) const
Checks if all bits are on in the specified mask.
Definition BitFlag.hh:173
constexpr TBitFlag< T, E > & setBit(Es... es)
Sets the corresponding bits for the provided enum values.
Definition BitFlag.hh:57
A 3D float vector.
Definition Vector.hh:83
A representation of the boundaries of an entity that has dynamic collision.