A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
KColData.hh
1#pragma once
2
4
5#include <egg/math/BoundBox.hh>
6#include <egg/math/Matrix.hh>
7
8// Credit: em-eight/mkw
9// Credit: stblr/Hanachan
10
11namespace Field {
12
14 EGG::BoundBox3f bbox;
15 EGG::Vector3f tangentOff;
16};
17
19 EGG::BoundBox3f bbox;
20 EGG::Vector3f tangentOff;
21 EGG::Vector3f floorNrm;
22 EGG::Vector3f wallNrm;
23 EGG::Vector3f _3c;
24 f32 floorDist;
25 f32 wallDist;
26 f32 _50;
27 f32 perpendicularity;
28
29 void updateFloor(f32 dist, const EGG::Vector3f &fnrm) {
30 if (dist > floorDist) {
31 floorDist = dist;
32 floorNrm = fnrm;
33 }
34 }
35
36 void updateWall(f32 dist, const EGG::Vector3f &fnrm) {
37 if (dist > wallDist) {
38 wallDist = dist;
39 wallNrm = fnrm;
40 }
41 }
42
43 void reset() {
44 bbox.setZero();
45 wallDist = -std::numeric_limits<f32>::min();
46 floorDist = -std::numeric_limits<f32>::min();
47 perpendicularity = 0.0f;
48 }
49
50 void update(f32 now_dist, const EGG::Vector3f &offset, const EGG::Vector3f &fnrm,
51 u32 kclAttributeTypeBit);
52 void transformInfo(CollisionInfo &rhs, const EGG::Matrix34f &mtx);
53};
54
57class KColData {
58public:
59 enum class CollisionCheckType {
60 Edge,
61 Plane,
62 Movement,
63 };
64
67 KCollisionPrism(f32 height, u16 posIndex, u16 faceNormIndex, u16 edge1NormIndex,
68 u16 edge2NormIndex, u16 edge3NormIndex, u16 attribute);
69
70 f32 height;
71 u16 pos_i;
72 u16 fnrm_i;
73 u16 enrm1_i;
74 u16 enrm2_i;
75 u16 enrm3_i;
76 u16 attribute;
77 };
78 STATIC_ASSERT(sizeof(KCollisionPrism) == 0x10);
79
80 KColData(const void *file);
81 ~KColData();
82
83 void narrowScopeLocal(const EGG::Vector3f &pos, f32 radius, KCLTypeMask mask);
84 void narrowPolygon_EachBlock(const u16 *prismArray);
85
86 void computeBBox();
87 [[nodiscard]] bool checkPointCollision(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut);
88 [[nodiscard]] bool checkSphereCollision(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut);
89 [[nodiscard]] bool checkSphere(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut);
90 [[nodiscard]] bool checkSphereSingle(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *flagsOut);
91
92 void lookupPoint(const EGG::Vector3f &pos, const EGG::Vector3f &prevPos, KCLTypeMask typeMask);
93 void lookupSphere(f32 radius, const EGG::Vector3f &pos, const EGG::Vector3f &prevPos,
94 KCLTypeMask typeMask);
95 void lookupSphereCached(const EGG::Vector3f &p1, const EGG::Vector3f &p2, u32 typeMask,
96 f32 radius);
97
98 [[nodiscard]] const u16 *searchBlock(const EGG::Vector3f &pos);
99
101 [[nodiscard]] const EGG::BoundBox3f &bbox() const {
102 return m_bbox;
103 }
104
105 [[nodiscard]] u16 prismCache(u32 idx) const {
106 return m_prismCache[idx];
107 }
109
110 [[nodiscard]] static EGG::Vector3f GetVertex(f32 height, const EGG::Vector3f &vertex1,
111 const EGG::Vector3f &fnrm, const EGG::Vector3f &enrm3, const EGG::Vector3f &enrm);
112
113private:
114 void preloadPrisms();
115 void preloadNormals();
116 void preloadVertices();
117
118 [[nodiscard]] bool checkCollision(const KCollisionPrism &prism, f32 *distOut,
119 EGG::Vector3f *fnrmOut, u16 *flagsOut, CollisionCheckType type);
120 [[nodiscard]] bool checkPointCollision(const KCollisionPrism &prism, f32 *distOut,
121 EGG::Vector3f *fnrmOut, u16 *flagsOut, bool movement);
122 [[nodiscard]] bool checkSphereMovement(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut);
123 [[nodiscard]] bool checkPointMovement(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut);
124 [[nodiscard]] bool checkPoint(f32 *distOut, EGG::Vector3f *fnrmOut, u16 *attributeOut);
125
126 const void *m_posData;
127 const void *m_nrmData;
128 const void *m_prismData;
129 const void *m_blockData;
130 f32 m_prismThickness;
131 EGG::Vector3f m_areaMinPos;
139 EGG::Vector3f m_pos;
140 EGG::Vector3f m_prevPos;
141 EGG::Vector3f m_movement;
142 f32 m_radius;
143 KCLTypeMask m_typeMask;
144 const u16 *m_prismIter;
145 EGG::BoundBox3f m_bbox;
146 std::array<u16, 256> m_prismCache;
147 u16 *m_prismCacheTop;
148 u16 *m_cachedPrismArray;
149 EGG::Vector3f m_cachedPos;
150 f32 m_cachedRadius;
151
154 std::span<KCollisionPrism> m_prisms;
155 std::span<EGG::Vector3f> m_nrms;
156 std::span<EGG::Vector3f> m_vertices;
157};
158
159} // namespace Field
A 3 x 4 matrix.
Definition Matrix.hh:8
Performs lookups for KCL triangles.
Definition KColData.hh:57
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.
A representation of a bounding cuboid.
Definition BoundBox.hh:30
A 3D float vector.
Definition Vector.hh:83