A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KColData.cc
1#include "KColData.hh"
2
3#include <egg/geom/Sphere.hh>
4#include <egg/math/Math.hh>
5
6#include <cmath>
7
8// Credit: em-eight/mkw
9// Credit: stblr/Hanachan
10
11namespace Field {
12
14KColData::KColData(const void *file) {
15 auto addOffset = [](const void *file, u32 offset) -> const void * {
16 return reinterpret_cast<const void *>(reinterpret_cast<const u8 *>(file) + offset);
17 };
18 EGG::RamStream stream = EGG::RamStream(file, sizeof(KColHeader));
19
20 u32 posOffset = stream.read_u32();
21 u32 nrmOffset = stream.read_u32();
22 u32 prismOffset = stream.read_u32();
23 u32 blockOffset = stream.read_u32();
24
25 m_posData = addOffset(file, posOffset);
26 m_nrmData = addOffset(file, nrmOffset);
27 m_prismData = addOffset(file, prismOffset);
28 m_blockData = addOffset(file, blockOffset);
29
30 m_prismThickness = stream.read_f32();
31 m_areaMinPos.read(stream);
32 m_areaXWidthMask = stream.read_u32();
33 m_areaYWidthMask = stream.read_u32();
34 m_areaZWidthMask = stream.read_u32();
35 m_blockWidthShift = stream.read_u32();
36 m_areaXBlocksShift = stream.read_u32();
37 m_areaXYBlocksShift = stream.read_u32();
38 m_sphereRadius = stream.read_f32();
39
40 m_pos.setZero();
41 m_prevPos.setZero();
42 m_movement.setZero();
43 m_radius = 0.0f;
44 m_prismIter = nullptr;
45 m_cachedPrismArray = m_prismCache.data() - 1;
46
47 // NOTE: Collision is expensive on the CPU, so we preload all of the prism data to ensure we're
48 // not constantly handling endianness.
52
54}
55
56KColData::~KColData() {
57 delete[] m_prisms.data();
58 delete[] m_nrms.data();
59 delete[] m_vertices.data();
60}
61
63void KColData::narrowScopeLocal(const EGG::Vector3f &pos, f32 radius, KCLTypeMask mask) {
64 m_prismCacheTop = m_prismCache.data();
65 m_pos = pos;
66 m_radius = radius;
67 m_typeMask = mask;
68 m_cachedPos = pos;
69 m_cachedRadius = radius;
70
71 if (radius <= m_sphereRadius) {
73 }
74
75 *m_prismCacheTop = 0;
76}
77
79void KColData::narrowPolygon_EachBlock(const u16 *prismArray) {
80 m_prismIter = prismArray;
81
82 while (checkSphereSingle(nullptr, nullptr, nullptr)) {
85 *(m_prismCacheTop++) = *m_prismIter;
86
87 if (m_prismCacheTop == m_prismCache.end()) {
88 --m_prismCacheTop;
89 return;
90 }
91 }
92}
93
97 m_bbox.max.set(-999999.0f);
98 m_bbox.min.set(999999.0f);
99
100 for (size_t i = 1; i < m_prisms.size(); ++i) {
101 const auto &prism = m_prisms[i];
102 const EGG::Vector3f &fnrm = m_nrms[prism.fnrm_i];
103 const EGG::Vector3f &enrm1 = m_nrms[prism.enrm1_i];
104 const EGG::Vector3f &enrm2 = m_nrms[prism.enrm2_i];
105 const EGG::Vector3f &enrm3 = m_nrms[prism.enrm3_i];
106 const EGG::Vector3f &vtx1 = m_vertices[prism.pos_i];
107
108 const EGG::Vector3f vtx2 = GetVertex(prism.height, vtx1, fnrm, enrm3, enrm1);
109 const EGG::Vector3f vtx3 = GetVertex(prism.height, vtx1, fnrm, enrm3, enrm2);
110
111 m_bbox.min = m_bbox.min.minimize(vtx1);
112 m_bbox.min = m_bbox.min.minimize(vtx2);
113 m_bbox.min = m_bbox.min.minimize(vtx3);
114 m_bbox.max = m_bbox.max.maximize(vtx1);
115 m_bbox.max = m_bbox.max.maximize(vtx2);
116 m_bbox.max = m_bbox.max.maximize(vtx3);
117 }
118}
119
121bool KColData::checkPointCollision(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut) {
122 return std::isfinite(m_prevPos.y) ? checkPointMovement(distOut, fnrmOut, flagsOut) :
123 checkPoint(distOut, fnrmOut, flagsOut);
124}
125
127bool KColData::checkSphereCollision(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut) {
128 return std::isfinite(m_prevPos.y) ? checkSphereMovement(distOut, fnrmOut, flagsOut) :
129 checkSphere(distOut, fnrmOut, flagsOut);
130}
131
138bool KColData::checkSphere(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut) {
139 // If there's no list of triangles to check, there's no collision
140 if (!m_prismIter) {
141 return false;
142 }
143
144 // Check collision for all triangles, and continuously call the function until we're out
145 while (*++m_prismIter != 0) {
146 const KCollisionPrism &prism = m_prisms[parse<u16>(*m_prismIter)];
147 if (checkCollision(prism, distOut, fnrmOut, flagsOut, CollisionCheckType::Plane)) {
148 return true;
149 }
150 }
151
152 // We're out of triangles to check - another list must be prepared for subsequent calls
153 m_prismIter = nullptr;
154 return false;
155}
156
158bool KColData::checkSphereSingle(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut) {
159 if (!m_prismIter) {
160 return false;
161 }
162
163 while (*++m_prismIter != 0) {
164 if (m_prismCacheTop != m_prismCache.begin()) {
165 u16 *puVar10 = m_prismCacheTop - 1;
166 while (*m_prismIter != *puVar10) {
167 if (puVar10-- < m_prismCache.begin()) {
168 break;
169 }
170 }
171
172 if (puVar10 >= m_prismCache.begin()) {
173 continue;
174 }
175 }
176
177 const KCollisionPrism &prism = m_prisms[parse<u16>(*m_prismIter)];
178 if (checkCollision(prism, distOut, fnrmOut, flagsOut, CollisionCheckType::Edge)) {
179 return true;
180 }
181 }
182
183 m_prismIter = nullptr;
184 return false;
185}
186
189void KColData::lookupPoint(const EGG::Vector3f &pos, const EGG::Vector3f &prevPos,
190 KCLTypeMask typeMask) {
191 m_prismIter = searchBlock(pos);
192 m_pos = pos;
193 m_prevPos = prevPos;
194 m_movement = pos - prevPos;
195 m_typeMask = typeMask;
196}
197
200void KColData::lookupSphere(f32 radius, const EGG::Vector3f &pos, const EGG::Vector3f &prevPos,
201 KCLTypeMask typeMask) {
202 m_prismIter = searchBlock(pos);
203 m_pos = pos;
204 m_prevPos = prevPos;
205 m_movement = pos - prevPos;
206 m_radius = std::min(radius, m_sphereRadius);
207 m_typeMask = typeMask;
208}
209
211void KColData::lookupSphereCached(const EGG::Vector3f &p1, const EGG::Vector3f &p2, u32 typeMask,
212 f32 radius) {
213 EGG::Sphere3f sphere1(p1, radius);
214 EGG::Sphere3f sphere2(m_cachedPos, m_cachedRadius);
215
216 if (!sphere1.isInsideOtherSphere(sphere2)) {
217 m_prismIter = searchBlock(p1);
218 m_radius = std::min(m_sphereRadius, radius);
219 } else {
220 m_radius = radius;
221 m_prismIter = m_cachedPrismArray;
222 }
223
224 m_pos = p1;
225 m_prevPos = p2;
226 m_movement = p1 - p2;
227 m_typeMask = typeMask;
228}
229
235 // Calculate the x, y, and z offsets of the point from the minimum
236 // corner of the tree's bounding box.
237 const int x = point.x - m_areaMinPos.x;
238 const int y = point.y - m_areaMinPos.y;
239 const int z = point.z - m_areaMinPos.z;
240
241 // Check if the point is outside the tree's bounding box in the x, y,
242 // or z dimensions. If it is, return 0.
243 if (x & m_areaXWidthMask || y & m_areaYWidthMask || z & m_areaZWidthMask) {
244 return nullptr;
245 }
246
247 // Initialize the current tree node to the root node of the tree.
248 u32 shift = m_blockWidthShift;
249 const u8 *curBlock = reinterpret_cast<const u8 *>(m_blockData);
250 s32 offset;
251
252 // Traverse the tree to find the leaf node containing the input point.
253 u32 index = 4 *
254 (((u32)z >> shift) << m_areaXYBlocksShift | ((u32)y >> shift) << m_areaXBlocksShift |
255 (u32)x >> shift);
256
257 while (true) {
258 // Get the offset of the current node's child node.
259 offset = parse<u32>(*reinterpret_cast<const u32 *>(curBlock + index));
260
261 // If the offset is negative, the current node is a leaf node.
262 if ((offset & 0x80000000) != 0) {
263 break;
264 }
265
266 // If the offset is non-negative, update the current node to be
267 // the child node and continue traversing the tree.
268 shift--;
269 curBlock += offset;
270
271 u32 x_shift = ((1 * ((u32)x >> shift)) & 1);
272 u32 y_shift = ((2 * ((u32)y >> shift)) & 2);
273 u32 z_shift = ((4 * ((u32)z >> shift)) & 4);
274
275 index = 4 * (x_shift | y_shift | z_shift);
276 }
277
278 // We have to remove the MSB since it's solely used to identify leaves.
279 return reinterpret_cast<const u16 *>(curBlock + (offset & ~0x80000000));
280}
281
293 const EGG::Vector3f &fnrm, const EGG::Vector3f &enrm3, const EGG::Vector3f &enrm) {
294 EGG::Vector3f cross = fnrm.cross(enrm);
295 f32 dp = cross.ps_dot(enrm3);
296 cross *= (height / dp);
297
298 return cross + vertex1;
299}
300
305 size_t prismCount =
306 (reinterpret_cast<uintptr_t>(m_blockData) - reinterpret_cast<uintptr_t>(m_prismData)) /
307 sizeof(KCollisionPrism);
308
309 EGG::RamStream stream = EGG::RamStream(m_prismData, sizeof(KCollisionPrism) * prismCount);
310
311 m_prisms = std::span<KCollisionPrism>(new KCollisionPrism[prismCount], prismCount);
312
313 // Because the prisms are one-indexed, we insert an empty prism
314 stream.skip(sizeof(KCollisionPrism));
315
316 for (size_t i = 1; i < prismCount; ++i) {
317 KCollisionPrism &prism = m_prisms[i];
318 prism.height = stream.read_f32();
319 prism.pos_i = stream.read_u16();
320 prism.fnrm_i = stream.read_u16();
321 prism.enrm1_i = stream.read_u16();
322 prism.enrm2_i = stream.read_u16();
323 prism.enrm3_i = stream.read_u16();
324 prism.attribute = stream.read_u16();
325 }
326}
327
332 size_t normalCount = (reinterpret_cast<uintptr_t>(m_prismData) + sizeof(KCollisionPrism) -
333 reinterpret_cast<uintptr_t>(m_nrmData)) /
334 sizeof(EGG::Vector3f);
335
336 m_nrms = std::span<EGG::Vector3f>(new EGG::Vector3f[normalCount], normalCount);
337 EGG::RamStream stream = EGG::RamStream(m_nrmData, sizeof(EGG::Vector3f) * normalCount);
338
339 for (auto &nrm : m_nrms) {
340 nrm.read(stream);
341 }
342}
343
348 size_t vertexCount =
349 (reinterpret_cast<uintptr_t>(m_nrmData) - reinterpret_cast<uintptr_t>(m_posData)) /
350 sizeof(EGG::Vector3f);
351
352 m_vertices = std::span<EGG::Vector3f>(new EGG::Vector3f[vertexCount], vertexCount);
353 EGG::RamStream stream = EGG::RamStream(m_posData, sizeof(EGG::Vector3f) * vertexCount);
354
355 for (auto &vert : m_vertices) {
356 vert.read(stream);
357 }
358}
359
365bool KColData::checkCollision(const KCollisionPrism &prism, f32 *distOut, EGG::Vector3f *fnrmOut,
366 u16 *flagsOut, CollisionCheckType type) {
367 // Responsible for updating the output params
368 auto out = [&](f32 dist) {
369 if (distOut) {
370 *distOut = dist;
371 }
372 if (fnrmOut) {
373 *fnrmOut = m_nrms[prism.fnrm_i];
374 }
375 if (flagsOut) {
376 *flagsOut = prism.attribute;
377 }
378 return true;
379 };
380
381 // The flag check occurs earlier than in the base game here. We don't want to do math if the tri
382 // we're checking doesn't have matching flags.
383 u32 attributeMask = KCL_ATTRIBUTE_TYPE_BIT(prism.attribute);
384 if (!(attributeMask & m_typeMask)) {
385 return false;
386 }
387
388 const EGG::Vector3f relativePos = m_pos - m_vertices[prism.pos_i];
389
390 // Edge normals point outside the triangle
391 const EGG::Vector3f &enrm1 = m_nrms[prism.enrm1_i];
392 f32 dist_ca = relativePos.ps_dot(enrm1);
393 if (m_radius <= dist_ca) {
394 return false;
395 }
396
397 const EGG::Vector3f &enrm2 = m_nrms[prism.enrm2_i];
398 f32 dist_ab = relativePos.ps_dot(enrm2);
399 if (m_radius <= dist_ab) {
400 return false;
401 }
402
403 const EGG::Vector3f &enrm3 = m_nrms[prism.enrm3_i];
404 f32 dist_bc = relativePos.ps_dot(enrm3) - prism.height;
405 if (m_radius <= dist_bc) {
406 return false;
407 }
408
409 const EGG::Vector3f &fnrm = m_nrms[prism.fnrm_i];
410 f32 plane_dist = relativePos.ps_dot(fnrm);
411 f32 dist_in_plane = m_radius - plane_dist;
412 if (dist_in_plane <= 0.0f) {
413 return false;
414 }
415
416 f32 typeDistance = m_prismThickness;
417 if (type == CollisionCheckType::Edge) {
418 typeDistance += m_radius;
419 }
420
421 if (dist_in_plane >= typeDistance) {
422 return false;
423 }
424
425 if (type == CollisionCheckType::Movement) {
426 if (attributeMask & KCL_TYPE_DIRECTIONAL && m_movement.dot(fnrm) > 0.0f) {
427 return false;
428 }
429 }
430
431 // Originally part of the edge searching, but moved out for simplicity
432 // If these are all zero, then we're inside the triangle
433 if (dist_ab <= 0.0f && dist_bc <= 0.0f && dist_ca <= 0.0f) {
434 if (type == CollisionCheckType::Movement) {
435 EGG::Vector3f lastPos = relativePos - m_movement;
436 // We're only colliding if we are moving towards the face
437 if (plane_dist < 0.0f && lastPos.ps_dot(fnrm) < 0.0f) {
438 return false;
439 }
440 }
441 return out(dist_in_plane);
442 }
443
444 EGG::Vector3f edge_nor, other_edge_nor;
445 f32 edge_dist, other_edge_dist;
446 bool swap = false;
447 bool swapNorms = false;
448 // > means further, < means closer, = means same distance
449 if (dist_ab >= dist_ca && dist_ab > dist_bc) {
450 // AB is the furthest edge
451 edge_nor = enrm2;
452 edge_dist = dist_ab;
453 if (dist_ca >= dist_bc) {
454 // CA is the second furthest edge
455 other_edge_nor = enrm1;
456 other_edge_dist = dist_ca;
457 swapNorms = true;
458 } else {
459 // BC is the second furthest edge
460 other_edge_nor = enrm3;
461 other_edge_dist = dist_bc;
462 swap = true;
463 }
464 } else if (dist_bc >= dist_ca) {
465 // BC is the furthest edge
466 edge_nor = enrm3;
467 edge_dist = dist_bc;
468 if (dist_ab >= dist_ca) {
469 // AB is the second furthest edge
470 other_edge_nor = enrm2;
471 other_edge_dist = dist_ab;
472 swapNorms = true;
473 } else {
474 // CA is the second furthest edge
475 other_edge_nor = enrm1;
476 other_edge_dist = dist_ca;
477 swap = true;
478 }
479 } else {
480 // CA is the furthest edge
481 edge_nor = enrm1;
482 edge_dist = dist_ca;
483 if (dist_bc >= dist_ab) {
484 // BC is the second furthest edge
485 other_edge_nor = enrm3;
486 other_edge_dist = dist_bc;
487 swapNorms = true;
488 } else {
489 // AB is the second furthest edge
490 other_edge_nor = enrm2;
491 other_edge_dist = dist_ab;
492 swap = true;
493 }
494 }
495
496 f32 cos = edge_nor.ps_dot(other_edge_nor);
497 f32 sq_dist;
498 if (cos * edge_dist > other_edge_dist) {
499 if (type == CollisionCheckType::Plane) {
500 if (edge_dist > plane_dist) {
501 return false;
502 }
503 }
504 sq_dist = m_radius * m_radius - edge_dist * edge_dist;
505 } else {
506 f32 sq_sin = cos * cos - 1.0f;
507
508 if (swap) {
509 std::swap(edge_dist, other_edge_dist);
510 }
511
512 if (swapNorms) {
513 std::swap(edge_nor, other_edge_nor);
514 }
515
516 f32 t = (cos * edge_dist - other_edge_dist) / sq_sin;
517 f32 s = edge_dist - t * cos;
518 const EGG::Vector3f corner_pos = edge_nor * t + other_edge_nor * s;
519
520 f32 cornerDot = corner_pos.ps_squareMag();
521 if (type == CollisionCheckType::Plane) {
522 if (cornerDot > plane_dist * plane_dist) {
523 return false;
524 }
525 }
526
527 sq_dist = m_radius * m_radius - cornerDot;
528 }
529
530 if (sq_dist < plane_dist * plane_dist || sq_dist <= 0.0f) {
531 return false;
532 }
533
534 f32 dist = EGG::Mathf::sqrt(sq_dist) - plane_dist;
535 if (dist <= 0.0f) {
536 return false;
537 }
538
539 if (type == CollisionCheckType::Movement) {
540 EGG::Vector3f lastPos = relativePos - m_movement;
541 // We're only colliding if we are moving towards the face
542 if (lastPos.ps_dot(fnrm) < 0.0f) {
543 return false;
544 }
545 }
546
547 return out(dist);
548}
549
552bool KColData::checkPointCollision(const KCollisionPrism &prism, f32 *distOut,
553 EGG::Vector3f *fnrmOut, u16 *flagsOut, bool movement) {
554 KCLTypeMask attrMask = KCL_ATTRIBUTE_TYPE_BIT(prism.attribute);
555 if (!(attrMask & m_typeMask)) {
556 return false;
557 }
558
559 const EGG::Vector3f relativePos = m_pos - m_vertices[prism.pos_i];
560
561 const EGG::Vector3f &enrm1 = m_nrms[prism.enrm1_i];
562 f32 dist_ca = relativePos.ps_dot(enrm1);
563 if (dist_ca >= 0.01f) {
564 return false;
565 }
566
567 const EGG::Vector3f &enrm2 = m_nrms[prism.enrm2_i];
568 f32 dist_ab = relativePos.ps_dot(enrm2);
569 if (dist_ab >= 0.01f) {
570 return false;
571 }
572
573 const EGG::Vector3f &enrm3 = m_nrms[prism.enrm3_i];
574 f32 dist_bc = relativePos.ps_dot(enrm3) - prism.height;
575 if (dist_bc >= 0.01f) {
576 return false;
577 }
578
579 const EGG::Vector3f &fnrm = m_nrms[prism.fnrm_i];
580 f32 plane_dist = relativePos.ps_dot(fnrm);
581 f32 dist_in_plane = 0.01f - plane_dist;
582 if (dist_in_plane <= 0.0f) {
583 return false;
584 }
585
586 if (m_prismThickness <= dist_in_plane && 0.02f + m_prismThickness <= dist_in_plane) {
587 return false;
588 }
589
590 if (movement && (attrMask & KCL_TYPE_DIRECTIONAL) && m_movement.dot(fnrm) < 0.0f) {
591 return false;
592 }
593
594 if (distOut) {
595 *distOut = dist_in_plane;
596 }
597
598 if (fnrmOut) {
599 *fnrmOut = fnrm;
600 }
601
602 if (flagsOut) {
603 *flagsOut = prism.attribute;
604 }
605
606 return true;
607}
608
615bool KColData::checkSphereMovement(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut) {
616 // If there's no list of triangles to check, there's no collision
617 if (!m_prismIter) {
618 return false;
619 }
620
621 // Check collision for all triangles, and continuously call the function until we're out
622 while (*++m_prismIter != 0) {
623 const KCollisionPrism &prism = m_prisms[parse<u16>(*m_prismIter)];
624 if (checkCollision(prism, distOut, fnrmOut, attributeOut, CollisionCheckType::Movement)) {
625 return true;
626 }
627 }
628
629 // We're out of triangles to check - another list must be prepared for subsequent calls
630 m_prismIter = nullptr;
631 return false;
632}
633
635bool KColData::checkPoint(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut) {
636 // If there's no list of triangles to check, there's no collision
637 if (!m_prismIter) {
638 return false;
639 }
640
641 // Check collision for all triangles, and continuously call the function until we're out
642 while (*++m_prismIter != 0) {
643 const KCollisionPrism &prism = m_prisms[parse<u16>(*m_prismIter)];
644 if (checkPointCollision(prism, distOut, fnrmOut, attributeOut, false)) {
645 return true;
646 }
647 }
648
649 // We're out of triangles to check - another list must be prepared for subsequent calls
650 m_prismIter = nullptr;
651 return false;
652}
653
655bool KColData::checkPointMovement(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut) {
656 // If there's no list of triangles to check, there's no collision
657 if (!m_prismIter) {
658 return false;
659 }
660
661 // Check collision for all triangles, and continuously call the function until we're out
662 while (*++m_prismIter != 0) {
663 const KCollisionPrism &prism = m_prisms[parse<u16>(*m_prismIter)];
664 if (checkPointCollision(prism, distOut, fnrmOut, attributeOut, true)) {
665 return true;
666 }
667 }
668
669 // We're out of triangles to check - another list must be prepared for subsequent calls
670 m_prismIter = nullptr;
671 return false;
672}
673
674KColData::KCollisionPrism::KCollisionPrism() = default;
675
676KColData::KCollisionPrism::KCollisionPrism(f32 height, u16 posIndex, u16 faceNormIndex,
677 u16 edge1NormIndex, u16 edge2NormIndex, u16 edge3NormIndex, u16 attribute)
678 : height(height), pos_i(posIndex), fnrm_i(faceNormIndex), enrm1_i(edge1NormIndex),
679 enrm2_i(edge2NormIndex), enrm3_i(edge3NormIndex), attribute(attribute) {}
680
681void CollisionInfo::update(f32 now_dist, const EGG::Vector3f &offset, const EGG::Vector3f &fnrm,
682 u32 kclAttributeTypeBit) {
683 bbox.min = bbox.min.minimize(offset);
684 bbox.max = bbox.max.maximize(offset);
685
686 if (kclAttributeTypeBit & KCL_TYPE_FLOOR) {
687 updateFloor(now_dist, fnrm);
688 } else if (kclAttributeTypeBit & KCL_TYPE_WALL) {
689 if (wallDist > -std::numeric_limits<f32>::min()) {
690 f32 dot = 1.0f - wallNrm.ps_dot(fnrm);
691 if (dot > perpendicularity) {
692 perpendicularity = std::min(dot, 1.0f);
693 }
694 }
695
696 updateWall(now_dist, fnrm);
697 }
698}
699
701void CollisionInfo::transformInfo(CollisionInfo &rhs, const EGG::Matrix34f &mtx) {
702 rhs.bbox.min = mtx.ps_multVector33(rhs.bbox.min);
703 rhs.bbox.max = mtx.ps_multVector33(rhs.bbox.max);
704
705 EGG::Vector3f min = rhs.bbox.min;
706
707 rhs.bbox.min = min.minimize(rhs.bbox.max);
708 rhs.bbox.max = min.maximize(rhs.bbox.max);
709
710 bbox.min = bbox.min.minimize(rhs.bbox.min);
711 bbox.max = bbox.max.maximize(rhs.bbox.max);
712
713 if (floorDist < rhs.floorDist) {
714 floorDist = rhs.floorDist;
715 floorNrm = mtx.ps_multVector33(rhs.floorNrm);
716 }
717
718 if (wallDist < rhs.wallDist) {
719 wallDist = rhs.wallDist;
720 wallNrm = mtx.ps_multVector33(rhs.wallNrm);
721 }
722
723 perpendicularity = std::max(perpendicularity, rhs.perpendicularity);
724}
725
726} // namespace Field
#define KCL_ATTRIBUTE_TYPE_BIT(x)
Given the full 2 byte KCL flag for a triangle, extracts the "Base Type" portion of the flag.
#define KCL_TYPE_FLOOR
0x20E80FFF - Any KCL that the player or items can drive/land on.
#define KCL_TYPE_DIRECTIONAL
0x05070000
#define KCL_TYPE_WALL
0xD010F000
A 3 x 4 matrix.
Definition Matrix.hh:8
Vector3f ps_multVector33(const Vector3f &vec) const
Paired-singles impl. of multVector33.
Definition Matrix.cc:255
A stream of data stored in memory.
Definition Stream.hh:64
void preloadNormals()
Creates a copy of the normals in memory.
Definition KColData.cc:331
void preloadVertices()
Creates a copy of the vertices in memory.
Definition KColData.cc:347
bool checkSphereMovement(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut)
Iterates the local data block to check for directional collision.
Definition KColData.cc:615
u32 m_areaYWidthMask
The y dimension of the octree's bounding box.
Definition KColData.hh:133
u32 m_areaZWidthMask
The z dimension of the octree's bounding box.
Definition KColData.hh:134
f32 m_sphereRadius
Clamps the sphere we check collision against.
Definition KColData.hh:138
static EGG::Vector3f GetVertex(f32 height, const EGG::Vector3f &vertex1, const EGG::Vector3f &fnrm, const EGG::Vector3f &enrm3, const EGG::Vector3f &enrm)
Computes a prism vertex based off of the triangle's normal vectors.
Definition KColData.cc:292
void preloadPrisms()
Creates a copy of the prisms in memory.
Definition KColData.cc:304
bool checkSphere(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut)
Iterates the list of looked-up triangles to see if we are colliding.
Definition KColData.cc:138
void computeBBox()
Calculates a EGG::BoundBox3f that describes the boundary of the track's KCL.
Definition KColData.cc:96
void lookupPoint(const EGG::Vector3f &pos, const EGG::Vector3f &prevPos, KCLTypeMask typeMask)
Sets members in preparation of a subsequent point collision check call.
Definition KColData.cc:189
u32 m_areaXWidthMask
The x dimension of the octree's bounding box.
Definition KColData.hh:132
const u16 * searchBlock(const EGG::Vector3f &pos)
Finds the data block corresponding to the provided position.
Definition KColData.cc:234
u32 m_blockWidthShift
Used to initialize octree navigation.
Definition KColData.hh:135
u32 m_areaXYBlocksShift
Used to initialize octree navigation.
Definition KColData.hh:137
void narrowPolygon_EachBlock(const u16 *prismArray)
Definition KColData.cc:79
u32 m_areaXBlocksShift
Used to initialize octree navigation.
Definition KColData.hh:136
std::span< KCollisionPrism > m_prisms
Optimizes for time by avoiding unnecessary byteswapping. The Wii doesn't have this problem because bi...
Definition KColData.hh:154
void lookupSphere(f32 radius, const EGG::Vector3f &pos, const EGG::Vector3f &prevPos, KCLTypeMask typeMask)
Sets members in preparation of a subsequent sphere collision check call.
Definition KColData.cc:200
bool checkCollision(const KCollisionPrism &prism, f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut, CollisionCheckType type)
This is a combination of the three collision checks in the base game.
Definition KColData.cc:365
Pertains to collision.
Represents a sphere in 3D space.
Definition Sphere.hh:8
A 3D float vector.
Definition Vector.hh:83
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
Definition Vector.hh:182
f32 ps_squareMag() const
Differs from ps_dot due to variation in which operands are fused.
Definition Vector.cc:35
void read(Stream &stream)
Initializes a Vector3f by reading 12 bytes from the stream.
Definition Vector.cc:115
Vector3f maximize(const Vector3f &rhs) const
Returns a vector whose elements are the max of the elements of both vectors.
Definition Vector.cc:65
f32 ps_dot() const
Paired-singles dot product implementation.
Definition Vector.cc:22
Vector3f minimize(const Vector3f &rhs) const
Returns a vector whose elements are the min of the elements of both vectors.
Definition Vector.cc:77