3#include <egg/geom/Sphere.hh>
4#include <egg/math/Math.hh>
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);
20 u32 posOffset = stream.read_u32();
21 u32 nrmOffset = stream.read_u32();
22 u32 prismOffset = stream.read_u32();
23 u32 blockOffset = stream.read_u32();
25 m_posData = addOffset(file, posOffset);
26 m_nrmData = addOffset(file, nrmOffset);
27 m_prismData = addOffset(file, prismOffset);
28 m_blockData = addOffset(file, blockOffset);
30 m_prismThickness = stream.read_f32();
31 m_areaMinPos.
read(stream);
44 m_prismIter =
nullptr;
45 m_cachedPrismArray = m_prismCache.data() - 1;
56KColData::~KColData() {
58 delete[] m_nrms.data();
59 delete[] m_vertices.data();
63void KColData::narrowScopeLocal(
const EGG::Vector3f &pos, f32 radius, KCLTypeMask mask) {
64 m_prismCacheTop = m_prismCache.data();
69 m_cachedRadius = radius;
80 m_prismIter = prismArray;
82 while (checkSphereSingle(
nullptr,
nullptr,
nullptr)) {
85 *(m_prismCacheTop++) = *m_prismIter;
87 if (m_prismCacheTop == m_prismCache.end()) {
97 m_bbox.max.set(-999999.0f);
98 m_bbox.min.set(999999.0f);
100 for (
size_t i = 1; i <
m_prisms.size(); ++i) {
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);
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);
127bool KColData::checkSphereCollision(f32 *distOut,
EGG::Vector3f *fnrmOut,
u16 *flagsOut) {
145 while (*++m_prismIter != 0) {
147 if (
checkCollision(prism, distOut, fnrmOut, flagsOut, CollisionCheckType::Plane)) {
153 m_prismIter =
nullptr;
158bool KColData::checkSphereSingle(f32 *distOut,
EGG::Vector3f *fnrmOut,
u16 *flagsOut) {
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()) {
172 if (puVar10 >= m_prismCache.begin()) {
177 const KCollisionPrism &prism =
m_prisms[parse<u16>(*m_prismIter)];
178 if (
checkCollision(prism, distOut, fnrmOut, flagsOut, CollisionCheckType::Edge)) {
183 m_prismIter =
nullptr;
190 KCLTypeMask typeMask) {
194 m_movement = pos - prevPos;
195 m_typeMask = typeMask;
201 KCLTypeMask typeMask) {
205 m_movement = pos - prevPos;
207 m_typeMask = typeMask;
216 if (!sphere1.isInsideOtherSphere(sphere2)) {
221 m_prismIter = m_cachedPrismArray;
226 m_movement = p1 - p2;
227 m_typeMask = typeMask;
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;
249 const u8 *curBlock =
reinterpret_cast<const u8 *
>(m_blockData);
259 offset = parse<u32>(*
reinterpret_cast<const u32 *
>(curBlock + index));
262 if ((offset & 0x80000000) != 0) {
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);
275 index = 4 * (x_shift | y_shift | z_shift);
279 return reinterpret_cast<const u16 *
>(curBlock + (offset & ~0x80000000));
295 f32 dp = cross.
ps_dot(enrm3);
296 cross *= (height / dp);
298 return cross + vertex1;
306 (
reinterpret_cast<uintptr_t
>(m_blockData) -
reinterpret_cast<uintptr_t
>(m_prismData)) /
316 for (
size_t i = 1; i < prismCount; ++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();
332 size_t normalCount = (
reinterpret_cast<uintptr_t
>(m_prismData) +
sizeof(
KCollisionPrism) -
333 reinterpret_cast<uintptr_t
>(m_nrmData)) /
336 m_nrms = std::span<EGG::Vector3f>(
new EGG::Vector3f[normalCount], normalCount);
339 for (
auto &nrm : m_nrms) {
349 (
reinterpret_cast<uintptr_t
>(m_nrmData) -
reinterpret_cast<uintptr_t
>(m_posData)) /
352 m_vertices = std::span<EGG::Vector3f>(
new EGG::Vector3f[vertexCount], vertexCount);
355 for (
auto &vert : m_vertices) {
366 u16 *flagsOut, CollisionCheckType type) {
368 auto out = [&](f32 dist) {
373 *fnrmOut = m_nrms[prism.fnrm_i];
376 *flagsOut = prism.attribute;
384 if (!(attributeMask & m_typeMask)) {
388 const EGG::Vector3f relativePos = m_pos - m_vertices[prism.pos_i];
392 f32 dist_ca = relativePos.
ps_dot(enrm1);
393 if (m_radius <= dist_ca) {
398 f32 dist_ab = relativePos.
ps_dot(enrm2);
399 if (m_radius <= dist_ab) {
404 f32 dist_bc = relativePos.
ps_dot(enrm3) - prism.height;
405 if (m_radius <= dist_bc) {
410 f32 plane_dist = relativePos.
ps_dot(fnrm);
411 f32 dist_in_plane = m_radius - plane_dist;
412 if (dist_in_plane <= 0.0f) {
416 f32 typeDistance = m_prismThickness;
417 if (type == CollisionCheckType::Edge) {
418 typeDistance += m_radius;
421 if (dist_in_plane >= typeDistance) {
425 if (type == CollisionCheckType::Movement) {
433 if (dist_ab <= 0.0f && dist_bc <= 0.0f && dist_ca <= 0.0f) {
434 if (type == CollisionCheckType::Movement) {
437 if (plane_dist < 0.0f && lastPos.
ps_dot(fnrm) < 0.0f) {
441 return out(dist_in_plane);
445 f32 edge_dist, other_edge_dist;
447 bool swapNorms =
false;
449 if (dist_ab >= dist_ca && dist_ab > dist_bc) {
453 if (dist_ca >= dist_bc) {
455 other_edge_nor = enrm1;
456 other_edge_dist = dist_ca;
460 other_edge_nor = enrm3;
461 other_edge_dist = dist_bc;
464 }
else if (dist_bc >= dist_ca) {
468 if (dist_ab >= dist_ca) {
470 other_edge_nor = enrm2;
471 other_edge_dist = dist_ab;
475 other_edge_nor = enrm1;
476 other_edge_dist = dist_ca;
483 if (dist_bc >= dist_ab) {
485 other_edge_nor = enrm3;
486 other_edge_dist = dist_bc;
490 other_edge_nor = enrm2;
491 other_edge_dist = dist_ab;
496 f32 cos = edge_nor.
ps_dot(other_edge_nor);
498 if (cos * edge_dist > other_edge_dist) {
499 if (type == CollisionCheckType::Plane) {
500 if (edge_dist > plane_dist) {
504 sq_dist = m_radius * m_radius - edge_dist * edge_dist;
506 f32 sq_sin = cos * cos - 1.0f;
509 std::swap(edge_dist, other_edge_dist);
513 std::swap(edge_nor, other_edge_nor);
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;
521 if (type == CollisionCheckType::Plane) {
522 if (cornerDot > plane_dist * plane_dist) {
527 sq_dist = m_radius * m_radius - cornerDot;
530 if (sq_dist < plane_dist * plane_dist || sq_dist <= 0.0f) {
534 f32 dist = EGG::Mathf::sqrt(sq_dist) - plane_dist;
539 if (type == CollisionCheckType::Movement) {
542 if (lastPos.
ps_dot(fnrm) < 0.0f) {
555 if (!(attrMask & m_typeMask)) {
559 const EGG::Vector3f relativePos = m_pos - m_vertices[prism.pos_i];
562 f32 dist_ca = relativePos.
ps_dot(enrm1);
563 if (dist_ca >= 0.01f) {
568 f32 dist_ab = relativePos.
ps_dot(enrm2);
569 if (dist_ab >= 0.01f) {
574 f32 dist_bc = relativePos.
ps_dot(enrm3) - prism.height;
575 if (dist_bc >= 0.01f) {
580 f32 plane_dist = relativePos.
ps_dot(fnrm);
581 f32 dist_in_plane = 0.01f - plane_dist;
582 if (dist_in_plane <= 0.0f) {
586 if (m_prismThickness <= dist_in_plane && 0.02f + m_prismThickness <= dist_in_plane) {
595 *distOut = dist_in_plane;
603 *flagsOut = prism.attribute;
622 while (*++m_prismIter != 0) {
624 if (
checkCollision(prism, distOut, fnrmOut, attributeOut, CollisionCheckType::Movement)) {
630 m_prismIter =
nullptr;
635bool KColData::checkPoint(f32 *distOut,
EGG::Vector3f *fnrmOut,
u16 *attributeOut) {
642 while (*++m_prismIter != 0) {
643 const KCollisionPrism &prism =
m_prisms[parse<u16>(*m_prismIter)];
644 if (checkPointCollision(prism, distOut, fnrmOut, attributeOut,
false)) {
650 m_prismIter =
nullptr;
655bool KColData::checkPointMovement(f32 *distOut,
EGG::Vector3f *fnrmOut,
u16 *attributeOut) {
662 while (*++m_prismIter != 0) {
663 const KCollisionPrism &prism =
m_prisms[parse<u16>(*m_prismIter)];
664 if (checkPointCollision(prism, distOut, fnrmOut, attributeOut,
true)) {
670 m_prismIter =
nullptr;
674KColData::KCollisionPrism::KCollisionPrism() =
default;
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) {}
682 u32 kclAttributeTypeBit) {
683 bbox.min = bbox.min.
minimize(offset);
684 bbox.max = bbox.max.
maximize(offset);
687 updateFloor(now_dist, fnrm);
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);
696 updateWall(now_dist, fnrm);
701void CollisionInfo::transformInfo(CollisionInfo &rhs,
const EGG::Matrix34f &mtx) {
707 rhs.bbox.min = min.
minimize(rhs.bbox.max);
708 rhs.bbox.max = min.
maximize(rhs.bbox.max);
710 bbox.min = bbox.min.
minimize(rhs.bbox.min);
711 bbox.max = bbox.max.
maximize(rhs.bbox.max);
713 if (floorDist < rhs.floorDist) {
714 floorDist = rhs.floorDist;
718 if (wallDist < rhs.wallDist) {
719 wallDist = rhs.wallDist;
723 perpendicularity = std::max(perpendicularity, rhs.perpendicularity);
#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
Vector3f ps_multVector33(const Vector3f &vec) const
Paired-singles impl. of multVector33.
A stream of data stored in memory.
void preloadNormals()
Creates a copy of the normals in memory.
void preloadVertices()
Creates a copy of the vertices in memory.
bool checkSphereMovement(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut)
Iterates the local data block to check for directional collision.
u32 m_areaYWidthMask
The y dimension of the octree's bounding box.
u32 m_areaZWidthMask
The z dimension of the octree's bounding box.
f32 m_sphereRadius
Clamps the sphere we check collision against.
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.
void preloadPrisms()
Creates a copy of the prisms in memory.
bool checkSphere(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut)
Iterates the list of looked-up triangles to see if we are colliding.
void computeBBox()
Calculates a EGG::BoundBox3f that describes the boundary of the track's KCL.
void lookupPoint(const EGG::Vector3f &pos, const EGG::Vector3f &prevPos, KCLTypeMask typeMask)
Sets members in preparation of a subsequent point collision check call.
u32 m_areaXWidthMask
The x dimension of the octree's bounding box.
const u16 * searchBlock(const EGG::Vector3f &pos)
Finds the data block corresponding to the provided position.
u32 m_blockWidthShift
Used to initialize octree navigation.
u32 m_areaXYBlocksShift
Used to initialize octree navigation.
void narrowPolygon_EachBlock(const u16 *prismArray)
u32 m_areaXBlocksShift
Used to initialize octree navigation.
std::span< KCollisionPrism > m_prisms
Optimizes for time by avoiding unnecessary byteswapping. The Wii doesn't have this problem because bi...
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.
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.
Represents a sphere in 3D space.
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
f32 ps_squareMag() const
Differs from ps_dot due to variation in which operands are fused.
void read(Stream &stream)
Initializes a Vector3f by reading 12 bytes from the stream.
Vector3f maximize(const Vector3f &rhs) const
Returns a vector whose elements are the max of the elements of both vectors.
f32 ps_dot() const
Paired-singles dot product implementation.
Vector3f minimize(const Vector3f &rhs) const
Returns a vector whose elements are the min of the elements of both vectors.