A reimplementation of Mario Kart Wii's physics engine in C++
Loading...
Searching...
No Matches
ObjectCollisionBase.cc
1#include "ObjectCollisionBase.hh"
2
3#include <egg/math/Math.hh>
4
5#include <cmath>
6
7namespace Field {
8
9ObjectCollisionBase::ObjectCollisionBase() = default;
10
11ObjectCollisionBase::~ObjectCollisionBase() = default;
12
14bool ObjectCollisionBase::check(ObjectCollisionBase &rhs, EGG::Vector3f &distance) {
15 constexpr f32 INITIAL_MAX_VALUE = std::sqrt(std::numeric_limits<f32>::max());
16 STATIC_ASSERT(std::bit_cast<u32>(INITIAL_MAX_VALUE) == 0x5f7fffff);
17
18 f32 rad = getBoundingRadius() + rhs.getBoundingRadius();
19 f32 sqDist = rad * rad;
20 f32 max = INITIAL_MAX_VALUE;
21 f32 lastRadius = 0.0f;
22
23 EGG::Vector3f D = EGG::Vector3f::zero;
26 GJKState state;
27
28 do {
29 for (state.m_idx = 0, state.m_mask = 1; state.m_flags & state.m_mask; state.m_mask *= 2) {
30 ++state.m_idx;
31 }
32
33 state.m_support1[state.m_idx] = getSupport(-D);
34 state.m_support2[state.m_idx] = rhs.getSupport(D);
35
36 EGG::Vector3f A = state.m_support1[state.m_idx] - state.m_support2[state.m_idx];
37 f32 max2 = max * max;
38
39 f32 dot = D.dot(A);
40 if (dot > 0.0f && dot * dot > sqDist * max2) {
41 return false;
42 }
43
44 lastRadius = std::max(lastRadius, dot / max);
45
46 if (inSimplex(state, A) || (max2 - dot) < max2 * 0.000001f) {
47 getNearestPoint(state, state.m_flags, v0, v1);
48
49 v0 -= D * (getBoundingRadius() / max);
50 v1 += D * (rhs.getBoundingRadius() / max);
51
52 distance = v1 - v0;
53
54 m_00 = v0;
55 rhs.m_00 = v1;
56
57 return true;
58 }
59
60 state.m_s[state.m_idx] = A;
61 state.m_00c = state.m_flags | state.m_mask;
62
63 if (!getNearestSimplex(state, D)) {
64 getNearestPoint(state, state.m_flags, v0, v1);
65
66 v0 -= D * (getBoundingRadius() / max);
67 v1 += D * (rhs.getBoundingRadius() / max);
68
69 distance = v1 - v0;
70
71 m_00 = v0;
72 rhs.m_00 = v1;
73
74 return true;
75 }
76
77 max = D.length();
78
79 if (max2 - max * max <= std::numeric_limits<f32>::epsilon() * max2) {
80 FUN_808350e4(state, D);
81 getNearestPoint(state, state.m_flags, v0, v1);
82 f32 len = D.length();
83 v0 -= D * (getBoundingRadius() / len);
84 v1 += D * (rhs.getBoundingRadius() / len);
85
86 distance = v1 - v0;
87
88 m_00 = v1;
89 rhs.m_00 = v1;
90
91 return true;
92 }
93
94 } while (state.m_flags < 0xf && (max > std::numeric_limits<f32>::epsilon()));
95
96 return false;
97}
98
100bool ObjectCollisionBase::enclosesOrigin(const GJKState &state, u32 idx) const {
101 u32 mask = 1;
102 for (u8 i = 0; i < 4; ++i, mask *= 2) {
103 if ((idx & mask) && state.m_scales[idx][i] <= 0.0f) {
104 return false;
105 }
106 }
107
108 return true;
109}
110
112void ObjectCollisionBase::FUN_808350e4(GJKState &state, EGG::Vector3f &v) const {
113 f32 min = std::numeric_limits<f32>::max();
114
115 for (u32 mask = state.m_00c; mask != 0; --mask) {
116 if (mask != (mask & state.m_00c) || !enclosesOrigin(state, mask)) {
117 continue;
118 }
119
120 f32 sqLen = 0.0f;
121 EGG::Vector3f tmp = EGG::Vector3f::zero;
122 getNearestPoint(state, mask, tmp);
123
124 sqLen = tmp.squaredLength();
125 if (sqLen < min) {
126 state.m_flags = mask;
127 v = tmp;
128 min = sqLen;
129 }
130 }
131}
132
134bool ObjectCollisionBase::getNearestSimplex(GJKState &state, EGG::Vector3f &v) const {
135 calcSimplex(state);
136
137 for (u32 i = state.m_flags; i != 0; --i) {
138 if (i != (i & state.m_flags) || !FUN_808357e4(state, i | state.m_mask)) {
139 continue;
140 }
141
142 state.m_flags = i | state.m_mask;
143 getNearestPoint(state, state.m_flags, v);
144 return true;
145 }
146
147 if (FUN_808357e4(state, state.m_mask)) {
148 state.m_flags = state.m_mask;
149 v = state.m_s[state.m_idx];
150
151 return true;
152 }
153
154 return false;
155}
156
158void ObjectCollisionBase::getNearestPoint(GJKState &state, u32 idx, EGG::Vector3f &v0,
159 EGG::Vector3f &v1) const {
160 v0.setZero();
161 v1.setZero();
162 f32 sum = 0.0f;
163
164 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
165 if ((idx & mask) == 0) {
166 continue;
167 }
168
169 sum += state.m_scales[idx][i];
170 v0 += state.m_support1[i] * state.m_scales[idx][i];
171 v1 += state.m_support2[i] * state.m_scales[idx][i];
172 }
173
174 v0 *= 1.0f / sum;
175 v1 *= 1.0f / sum;
176}
177
179bool ObjectCollisionBase::FUN_808357e4(const GJKState &state, u32 idx) const {
180 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
181 if ((state.m_00c & mask) == 0) {
182 continue;
183 }
184
185 if (idx & mask) {
186 if (state.m_scales[idx][i] <= 0.0f) {
187 return false;
188 }
189 } else if (state.m_scales[idx | mask][i] > 0.0f) {
190 return false;
191 }
192 }
193
194 return true;
195}
196
198bool ObjectCollisionBase::inSimplex(const GJKState &state, const EGG::Vector3f &v) const {
199 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
200 if ((state.m_00c & mask) && state.m_s[i] == v) {
201 return true;
202 }
203 }
204
205 return false;
206}
207
209void ObjectCollisionBase::getNearestPoint(const GJKState &state, u32 idx, EGG::Vector3f &v) const {
210 v.setZero();
211
212 f32 sum = 0.0f;
213
214 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
215 if ((idx & mask) == 0) {
216 continue;
217 }
218
219 sum += state.m_scales[idx][i];
220 v += state.m_s[i] * state.m_scales[idx][i];
221 }
222
223 v *= 1.0f / sum;
224}
225
227void ObjectCollisionBase::calcSimplex(GJKState &state) const {
228 const u32 idx = state.m_idx;
229
230 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
231 if ((state.m_flags & mask) == 0) {
232 continue;
233 }
234
235 f32 result = state.m_s[i].dot(state.m_s[idx]);
236 s_dotProductCache[idx][i] = result;
237 s_dotProductCache[i][idx] = result;
238 }
239
240 state.m_scales[state.m_mask][idx] = 1.0f;
241 s_dotProductCache[idx][idx] = state.m_s[idx].squaredLength();
242
243 for (u32 i = 0, iMask = 1; i < 4; ++i, iMask *= 2) {
244 if ((state.m_flags & iMask) == 0) {
245 continue;
246 }
247
248 u32 iStateMask = iMask | state.m_mask;
249
250 state.m_scales[iStateMask][i] = s_dotProductCache[idx][idx] - s_dotProductCache[idx][i];
251 state.m_scales[iStateMask][idx] = s_dotProductCache[i][i] - s_dotProductCache[i][idx];
252
253 for (u32 j = 0, jMask = 1; j < i; ++j, jMask *= 2) {
254 if ((state.m_flags & jMask) == 0) {
255 continue;
256 }
257
258 state.m_scales[jMask | iStateMask][j] = state.m_scales[iStateMask][i] *
259 (s_dotProductCache[i][i] - s_dotProductCache[i][j]) +
260 state.m_scales[iStateMask][idx] *
261 (s_dotProductCache[idx][i] - s_dotProductCache[idx][j]);
262 state.m_scales[jMask | iStateMask][i] = state.m_scales[jMask | state.m_mask][j] *
263 (s_dotProductCache[j][j] - s_dotProductCache[i][j]) +
264 state.m_scales[jMask | state.m_mask][idx] *
265 (s_dotProductCache[idx][j] - s_dotProductCache[idx][i]);
266 state.m_scales[jMask | iStateMask][idx] = state.m_scales[jMask | iMask][j] *
267 (s_dotProductCache[j][j] - s_dotProductCache[j][idx]) +
268 state.m_scales[jMask | iMask][i] *
269 (s_dotProductCache[i][j] - s_dotProductCache[i][idx]);
270 }
271 }
272
273 if (state.m_00c != 0xf) {
274 return;
275 }
276
277 f32 _1_1 = s_dotProductCache[0][0] - s_dotProductCache[0][1];
278 f32 _2_2 = s_dotProductCache[0][0] - s_dotProductCache[0][2];
279 f32 _3_2 = s_dotProductCache[0][0] - s_dotProductCache[0][3];
280 f32 _0_2 = s_dotProductCache[1][1] - s_dotProductCache[1][0];
281 f32 _0_3 = s_dotProductCache[2][1] - s_dotProductCache[2][0];
282 f32 _1_2 = s_dotProductCache[3][0] - s_dotProductCache[3][1];
283 f32 _1_3 = s_dotProductCache[2][0] - s_dotProductCache[2][1];
284 f32 _2_1 = s_dotProductCache[3][0] - s_dotProductCache[3][2];
285 f32 _0_1 = s_dotProductCache[3][1] - s_dotProductCache[3][0];
286 f32 _2_3 = s_dotProductCache[1][0] - s_dotProductCache[1][2];
287 f32 _3_1 = s_dotProductCache[2][0] - s_dotProductCache[2][3];
288 f32 _3_3 = s_dotProductCache[1][0] - s_dotProductCache[1][3];
289
290 state.m_scales[15][0] = _0_1 * state.m_scales[14][3] +
291 (_0_2 * state.m_scales[14][1] + _0_3 * state.m_scales[14][2]);
292 state.m_scales[15][1] = _1_2 * state.m_scales[13][3] +
293 (_1_1 * state.m_scales[13][0] + _1_3 * state.m_scales[13][2]);
294 state.m_scales[15][2] = _2_1 * state.m_scales[11][3] +
295 (_2_2 * state.m_scales[11][0] + _2_3 * state.m_scales[11][1]);
296 state.m_scales[15][3] = _3_1 * state.m_scales[7][2] +
297 (_3_2 * state.m_scales[7][0] + _3_3 * state.m_scales[7][1]);
298}
299
300std::array<std::array<f32, 4>, 4> ObjectCollisionBase::s_dotProductCache = {{}};
301
302} // namespace Field
Pertains to collision.
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 length() const
The square root of the vector's dot product.
Definition Vector.hh:187
f32 squaredLength() const
The dot product between the vector and itself.
Definition Vector.hh:177