1#include "ObjectCollisionBase.hh"
3#include <egg/math/Math.hh>
9ObjectCollisionBase::ObjectCollisionBase() =
default;
11ObjectCollisionBase::~ObjectCollisionBase() =
default;
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);
18 f32 rad = getBoundingRadius() + rhs.getBoundingRadius();
19 f32 sqDist = rad * rad;
20 f32 max = INITIAL_MAX_VALUE;
21 f32 lastRadius = 0.0f;
29 for (state.m_idx = 0, state.m_mask = 1; state.m_flags & state.m_mask; state.m_mask *= 2) {
33 state.m_support1[state.m_idx] = getSupport(-D);
34 state.m_support2[state.m_idx] = rhs.getSupport(D);
36 EGG::Vector3f A = state.m_support1[state.m_idx] - state.m_support2[state.m_idx];
40 if (dot > 0.0f && dot * dot > sqDist * max2) {
44 lastRadius = std::max(lastRadius, dot / max);
46 if (inSimplex(state, A) || (max2 - dot) < max2 * 0.000001f) {
47 getNearestPoint(state, state.m_flags, v0, v1);
49 v0 -= D * (getBoundingRadius() / max);
50 v1 += D * (rhs.getBoundingRadius() / max);
60 state.m_s[state.m_idx] = A;
61 state.m_00c = state.m_flags | state.m_mask;
63 if (!getNearestSimplex(state, D)) {
64 getNearestPoint(state, state.m_flags, v0, v1);
66 v0 -= D * (getBoundingRadius() / max);
67 v1 += D * (rhs.getBoundingRadius() / max);
79 if (max2 - max * max <= std::numeric_limits<f32>::epsilon() * max2) {
80 FUN_808350e4(state, D);
81 getNearestPoint(state, state.m_flags, v0, v1);
83 v0 -= D * (getBoundingRadius() / len);
84 v1 += D * (rhs.getBoundingRadius() / len);
94 }
while (state.m_flags < 0xf && (max > std::numeric_limits<f32>::epsilon()));
100bool ObjectCollisionBase::enclosesOrigin(
const GJKState &state, u32 idx)
const {
102 for (u8 i = 0; i < 4; ++i, mask *= 2) {
103 if ((idx & mask) && state.m_scales[idx][i] <= 0.0f) {
112void ObjectCollisionBase::FUN_808350e4(GJKState &state,
EGG::Vector3f &v)
const {
113 f32 min = std::numeric_limits<f32>::max();
115 for (u32 mask = state.m_00c; mask != 0; --mask) {
116 if (mask != (mask & state.m_00c) || !enclosesOrigin(state, mask)) {
122 getNearestPoint(state, mask, tmp);
126 state.m_flags = mask;
134bool ObjectCollisionBase::getNearestSimplex(GJKState &state,
EGG::Vector3f &v)
const {
137 for (u32 i = state.m_flags; i != 0; --i) {
138 if (i != (i & state.m_flags) || !FUN_808357e4(state, i | state.m_mask)) {
142 state.m_flags = i | state.m_mask;
143 getNearestPoint(state, state.m_flags, v);
147 if (FUN_808357e4(state, state.m_mask)) {
148 state.m_flags = state.m_mask;
149 v = state.m_s[state.m_idx];
158void ObjectCollisionBase::getNearestPoint(GJKState &state, u32 idx,
EGG::Vector3f &v0,
164 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
165 if ((idx & mask) == 0) {
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];
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) {
186 if (state.m_scales[idx][i] <= 0.0f) {
189 }
else if (state.m_scales[idx | mask][i] > 0.0f) {
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) {
209void ObjectCollisionBase::getNearestPoint(
const GJKState &state, u32 idx,
EGG::Vector3f &v)
const {
214 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
215 if ((idx & mask) == 0) {
219 sum += state.m_scales[idx][i];
220 v += state.m_s[i] * state.m_scales[idx][i];
227void ObjectCollisionBase::calcSimplex(GJKState &state)
const {
228 const u32 idx = state.m_idx;
230 for (u32 i = 0, mask = 1; i < 4; ++i, mask *= 2) {
231 if ((state.m_flags & mask) == 0) {
235 f32 result = state.m_s[i].dot(state.m_s[idx]);
236 s_dotProductCache[idx][i] = result;
237 s_dotProductCache[i][idx] = result;
240 state.m_scales[state.m_mask][idx] = 1.0f;
241 s_dotProductCache[idx][idx] = state.m_s[idx].squaredLength();
243 for (u32 i = 0, iMask = 1; i < 4; ++i, iMask *= 2) {
244 if ((state.m_flags & iMask) == 0) {
248 u32 iStateMask = iMask | state.m_mask;
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];
253 for (u32 j = 0, jMask = 1; j < i; ++j, jMask *= 2) {
254 if ((state.m_flags & jMask) == 0) {
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]);
273 if (state.m_00c != 0xf) {
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];
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]);
300std::array<std::array<f32, 4>, 4> ObjectCollisionBase::s_dotProductCache = {{}};
f32 dot(const Vector3f &rhs) const
The dot product between two vectors.
f32 length() const
The square root of the vector's dot product.
f32 squaredLength() const
The dot product between the vector and itself.