#include "Runtime/Collision/CAreaOctTree.hpp" #include "Runtime/Collision/CMaterialFilter.hpp" #include <array> #include <cfloat> #include <cmath> #include <utility> #include <hecl/hecl.hpp> #include <zeus/CVector2i.hpp> namespace metaforce { static bool _close_enough(float f1, float f2, float epsilon) { return std::fabs(f1 - f2) <= epsilon; } static bool BoxLineTest(const zeus::CAABox& aabb, const zeus::CLine& line, float& lT, float& hT) { zeus::simd_floats aabbMinF(aabb.min.mSimd); zeus::simd_floats aabbMaxF(aabb.max.mSimd); zeus::simd_floats lineOrigin(line.origin.mSimd); zeus::simd_floats lineDir(line.dir.mSimd); const float* aabbMin = aabbMinF.data(); const float* aabbMax = aabbMaxF.data(); const float* lorigin = lineOrigin.data(); const float* ldir = lineDir.data(); lT = -FLT_MAX; hT = FLT_MAX; for (int i = 0; i < 3; ++i) { if (_close_enough(*ldir, 0.f, 0.000099999997f)) if (*lorigin < *aabbMin || *lorigin > *aabbMax) return false; if (*ldir < 0.f) { if (*aabbMax - *lorigin < lT * *ldir) lT = (*aabbMax - *lorigin) * 1.f / *ldir; if (*aabbMin - *lorigin > hT * *ldir) hT = (*aabbMin - *lorigin) * 1.f / *ldir; } else { if (*aabbMin - *lorigin > lT * *ldir) lT = (*aabbMin - *lorigin) * 1.f / *ldir; if (*aabbMax - *lorigin < hT * *ldir) hT = (*aabbMax - *lorigin) * 1.f / *ldir; } ++aabbMin; ++aabbMax; ++lorigin; ++ldir; } return lT <= hT; } constexpr std::array SomeIndexA{1, 2, 4}; constexpr std::array SomeIndexB{1, 2, 0}; constexpr std::array<std::array<int, 8>, 8> SomeIndexC{{ {0, 1, 2, 4, 5, 6, 8, 0xA}, {0, 1, 2, 3, 5, 6, 8, 0xA}, {0, 1, 2, 4, 5, 6, 9, 0xB}, {0, 1, 2, 3, 5, 6, 9, 0xC}, {0, 1, 2, 4, 5, 7, 8, 0xD}, {0, 1, 2, 3, 5, 7, 8, 0xE}, {0, 1, 2, 4, 5, 7, 9, 0xF}, {0, 1, 2, 3, 5, 7, 9, 0xF}, }}; constexpr std::array<std::pair<int, std::array<int, 3>>, 16> SubdivIndex{{ {0, {0, 0, 0}}, {1, {0, 0, 0}}, {1, {1, 0, 0}}, {2, {0, 1, 0}}, {2, {1, 0, 0}}, {1, {2, 0, 0}}, {2, {0, 2, 0}}, {2, {2, 0, 0}}, {2, {2, 1, 0}}, {2, {1, 2, 0}}, {3, {0, 2, 1}}, {3, {1, 0, 2}}, {3, {0, 1, 2}}, {3, {2, 1, 0}}, {3, {2, 0, 1}}, {3, {1, 2, 0}}, }}; bool CAreaOctTree::Node::LineTestInternal(const zeus::CLine& line, const CMaterialFilter& filter, float lT, float hT, float maxT, const zeus::CVector3f& vec) const { float lowT = (1.f - FLT_EPSILON * 100.f) * lT; float highT = (1.f + FLT_EPSILON * 100.f) * hT; if (maxT != 0.f) { if (lowT < 0.f) lowT = 0.f; if (highT > maxT) highT = maxT; if (lowT > highT) return true; } if (x20_nodeType == ETreeType::Leaf) { TriListReference triList = GetTriangleArray(); for (u16 i = 0; i < triList.GetSize(); ++i) { CCollisionSurface triangle = x1c_owner.GetMasterListTriangle(triList.GetAt(i)); // https://en.wikipedia.org/wiki/Möller–Trumbore_intersection_algorithm // Find vectors for two edges sharing V0 zeus::CVector3f e0 = triangle.GetVert(1) - triangle.GetVert(0); zeus::CVector3f e1 = triangle.GetVert(2) - triangle.GetVert(0); // Begin calculating determinant - also used to calculate u parameter zeus::CVector3f P = line.dir.cross(e1); float det = P.dot(e0); // If determinant is near zero, ray lies in plane of triangle // or ray is parallel to plane of triangle if (std::fabs(det) < (FLT_EPSILON * 10.f)) continue; float invDet = 1.f / det; // Calculate distance from V1 to ray origin zeus::CVector3f T = line.origin - triangle.GetVert(0); // Calculate u parameter and test bound float u = invDet * T.dot(P); // The intersection lies outside of the triangle if (u < 0.f || u > 1.f) continue; // Prepare to test v parameter zeus::CVector3f Q = T.cross(e0); // Calculate T parameter and test bound float t = invDet * Q.dot(e1); if (t >= highT || t < lowT) continue; // Calculate V parameter and test bound float v = invDet * Q.dot(line.dir); if (v < 0.f || u + v > 1.f) continue; // Do material filter CMaterialList matList(triangle.GetSurfaceFlags()); if (filter.Passes(matList)) return false; } } else if (x20_nodeType == ETreeType::Branch) { if (GetChildFlags() == 0xA) // 2 leaves { for (int i = 0; i < 2; ++i) { Node child = GetChild(i); float tf1 = lT; float tf2 = hT; if (BoxLineTest(child.GetBoundingBox(), line, tf1, tf2)) if (!child.LineTestInternal(line, filter, tf1, tf2, maxT, vec)) return false; } return true; } zeus::CVector3f center = x0_aabb.center(); zeus::CVector3f r6 = line.origin + lT * line.dir; zeus::CVector3f r7 = line.origin + hT * line.dir; zeus::CVector3f r9 = vec * (center - line.origin); int r28 = 0; int r25 = 0; int r26 = 0; for (size_t i = 0; i < 3; ++i) { if (r6[i] >= center[i]) r28 |= SomeIndexA[i]; if (r7[i] >= center[i]) r25 |= SomeIndexA[i]; if (r9[i] < r9[SomeIndexB[i]]) r26 |= SomeIndexA[i]; } float f21 = lT; int r26b = r28; const std::pair<int, std::array<int, 3>>& idx = SubdivIndex[SomeIndexC[r26][r28 ^ r25]]; for (int i = 0; i <= idx.first; ++i) { float f22 = (i < idx.first) ? r9[idx.second[i]] : hT; if (f22 > lowT && f21 <= f22) { Node child = GetChild(r26b); if (child.x20_nodeType != ETreeType::Invalid) if (!child.LineTestInternal(line, filter, f21, f22, maxT, vec)) return false; } if (i < idx.first) r26b ^= 1 << idx.second[i]; f21 = f22; } } return true; } void CAreaOctTree::Node::LineTestExInternal(const zeus::CLine& line, const CMaterialFilter& filter, SRayResult& res, float lT, float hT, float maxT, const zeus::CVector3f& dirRecip) const { float lowT = (1.f - FLT_EPSILON * 100.f) * lT; float highT = (1.f + FLT_EPSILON * 100.f) * hT; if (maxT != 0.f) { if (lowT < 0.f) lowT = 0.f; if (highT > maxT) highT = maxT; if (lowT > highT) return; } if (x20_nodeType == ETreeType::Leaf) { TriListReference triList = GetTriangleArray(); float bestT = highT; bool foundTriangle = false; SRayResult tmpRes; for (u16 i = 0; i < triList.GetSize(); ++i) { CCollisionSurface triangle = x1c_owner.GetMasterListTriangle(triList.GetAt(i)); // https://en.wikipedia.org/wiki/Möller–Trumbore_intersection_algorithm // Find vectors for two edges sharing V0 zeus::CVector3f e0 = triangle.GetVert(1) - triangle.GetVert(0); zeus::CVector3f e1 = triangle.GetVert(2) - triangle.GetVert(0); // Begin calculating determinant - also used to calculate u parameter zeus::CVector3f P = line.dir.cross(e1); float det = P.dot(e0); // If determinant is near zero, ray lies in plane of triangle // or ray is parallel to plane of triangle if (std::fabs(det) < (FLT_EPSILON * 10.f)) continue; float invDet = 1.f / det; // Calculate distance from V1 to ray origin zeus::CVector3f T = line.origin - triangle.GetVert(0); // Calculate u parameter and test bound float u = invDet * T.dot(P); // The intersection lies outside of the triangle if (u < 0.f || u > 1.f) continue; // Prepare to test v parameter zeus::CVector3f Q = T.cross(e0); // Calculate T parameter and test bound float t = invDet * Q.dot(e1); if (t >= bestT || t < lowT) continue; // Calculate V parameter and test bound float v = invDet * Q.dot(line.dir); if (v < 0.f || u + v > 1.f) continue; // Do material filter CMaterialList matList(triangle.GetSurfaceFlags()); if (filter.Passes(matList) && t <= bestT) { bestT = t; foundTriangle = true; tmpRes.x10_surface.emplace(triangle); tmpRes.x3c_t = t; } } if (foundTriangle) { res = tmpRes; res.x0_plane = res.x10_surface->GetPlane(); } } else if (x20_nodeType == ETreeType::Branch) { if (GetChildFlags() == 0xA) // 2 leaves { std::array<SRayResult, 2> tmpRes; for (int i = 0; i < 2; ++i) { Node child = GetChild(i); float tf1 = lT; float tf2 = hT; if (BoxLineTest(child.GetBoundingBox(), line, tf1, tf2)) child.LineTestExInternal(line, filter, tmpRes[i], tf1, tf2, maxT, dirRecip); } if (!tmpRes[0].x10_surface && !tmpRes[1].x10_surface) { res = SRayResult(); } else if (tmpRes[0].x10_surface && tmpRes[1].x10_surface) { if (tmpRes[0].x3c_t < tmpRes[1].x3c_t) res = tmpRes[0]; else res = tmpRes[1]; } else if (tmpRes[0].x10_surface) { res = tmpRes[0]; } else { res = tmpRes[1]; } if (res.x3c_t > highT) res = SRayResult(); return; } zeus::CVector3f center = x0_aabb.center(); // r26 zeus::CVector3f lowPoint = line.origin + lT * line.dir; zeus::CVector3f highPoint = line.origin + hT * line.dir; std::array<int, 4> comps{-1, -1, -1, 0}; std::array<float, 3> compT; int numComps = 0; for (size_t i = 0; i < compT.size(); ++i) { if (lowPoint[i] >= center[i] || highPoint[i] <= center[i]) { if (highPoint[i] >= center[i] || lowPoint[i] <= center[i]) { continue; } } if (_close_enough(line.dir[i], 0.f, 0.000099999997f)) { continue; } comps[numComps++] = static_cast<int>(i); compT[i] = dirRecip[i] * (center[i] - line.origin[i]); } // Sort componentT least to greatest switch (numComps) { default: return; case 0: case 1: break; case 2: if (compT[comps[1]] < compT[comps[0]]) std::swap(comps[1], comps[0]); break; case 3: if (compT[0] < compT[1]) { if (compT[0] >= compT[2]) { comps[0] = 2; comps[1] = 0; comps[2] = 1; } else if (compT[1] < compT[2]) { comps[0] = 0; comps[1] = 1; comps[2] = 2; } else { comps[0] = 0; comps[1] = 2; comps[2] = 1; } } else { if (compT[1] >= compT[2]) { comps[0] = 2; comps[1] = 1; comps[2] = 0; } else if (compT[0] < compT[2]) { comps[0] = 1; comps[1] = 0; comps[2] = 2; } else { comps[0] = 1; comps[1] = 2; comps[2] = 0; } } break; } zeus::CVector3f lineStart = line.origin + (lT * line.dir); int selector = 0; if (lineStart.x() >= center.x()) selector = 1; if (lineStart.y() >= center.y()) selector |= 1 << 1; if (lineStart.z() >= center.z()) selector |= 1 << 2; float tmpLoT = lT; for (int i = -1; i < numComps; ++i) { if (i >= 0) selector ^= 1 << comps[i]; float tmpHiT = (i < numComps - 1) ? compT[comps[i + 1]] : hT; if (tmpHiT > lowT && tmpLoT <= tmpHiT) { Node child = GetChild(selector); if (child.x20_nodeType != ETreeType::Invalid) child.LineTestExInternal(line, filter, res, tmpLoT, tmpHiT, maxT, dirRecip); if (res.x10_surface) { if (res.x3c_t > highT) res = SRayResult(); break; } } tmpLoT = tmpHiT; } } } bool CAreaOctTree::Node::LineTest(const zeus::CLine& line, const CMaterialFilter& filter, float length) const { if (x20_nodeType == ETreeType::Invalid) return true; float f1 = 0.f; float f2 = 0.f; if (!BoxLineTest(x0_aabb, line, f1, f2)) return true; zeus::CVector3f recip = 1.f / line.dir; return LineTestInternal(line, filter, f1 - 0.000099999997f, f2 + 0.000099999997f, length, recip); } void CAreaOctTree::Node::LineTestEx(const zeus::CLine& line, const CMaterialFilter& filter, SRayResult& res, float length) const { if (x20_nodeType == ETreeType::Invalid) return; float lT = 0.f; float hT = 0.f; if (!BoxLineTest(x0_aabb, line, lT, hT)) return; zeus::CVector3f recip = 1.f / line.dir; LineTestExInternal(line, filter, res, lT - 0.000099999997f, hT + 0.000099999997f, length, recip); } CAreaOctTree::Node CAreaOctTree::Node::GetChild(int idx) const { u16 flags = *reinterpret_cast<const u16*>(x18_ptr); const u32* offsets = reinterpret_cast<const u32*>(x18_ptr + 4); ETreeType type = ETreeType((flags >> (2 * idx)) & 0x3); if (type == ETreeType::Branch) { zeus::CAABox pos, neg, res; x0_aabb.splitZ(neg, pos); if (idx & 4) { zeus::CAABox(pos).splitY(neg, pos); if (idx & 2) { zeus::CAABox(pos).splitX(neg, pos); if (idx & 1) res = pos; else res = neg; } else { zeus::CAABox(neg).splitX(neg, pos); if (idx & 1) res = pos; else res = neg; } } else { zeus::CAABox(neg).splitY(neg, pos); if (idx & 2) { zeus::CAABox(pos).splitX(neg, pos); if (idx & 1) res = pos; else res = neg; } else { zeus::CAABox(neg).splitX(neg, pos); if (idx & 1) res = pos; else res = neg; } } return Node(x18_ptr + offsets[idx] + 36, res, x1c_owner, ETreeType::Branch); } else if (type == ETreeType::Leaf) { const float* aabb = reinterpret_cast<const float*>(x18_ptr + offsets[idx] + 36); zeus::CAABox aabbObj(aabb[0], aabb[1], aabb[2], aabb[3], aabb[4], aabb[5]); return Node(aabb, aabbObj, x1c_owner, ETreeType::Leaf); } else { return Node(nullptr, zeus::skNullBox, x1c_owner, ETreeType::Invalid); } } void CAreaOctTree::SwapTreeNode(u8* ptr, Node::ETreeType type) { if (type == Node::ETreeType::Branch) { u16* typeBits = reinterpret_cast<u16*>(ptr); *typeBits = hecl::SBig(*typeBits); u32* offsets = reinterpret_cast<u32*>(ptr + 4); for (int i = 0; i < 8; ++i) { Node::ETreeType ctype = Node::ETreeType((*typeBits >> (2 * i)) & 0x3); offsets[i] = hecl::SBig(offsets[i]); SwapTreeNode(ptr + offsets[i] + 36, ctype); } } else if (type == Node::ETreeType::Leaf) { float* aabb = reinterpret_cast<float*>(ptr); aabb[0] = hecl::SBig(aabb[0]); aabb[1] = hecl::SBig(aabb[1]); aabb[2] = hecl::SBig(aabb[2]); aabb[3] = hecl::SBig(aabb[3]); aabb[4] = hecl::SBig(aabb[4]); aabb[5] = hecl::SBig(aabb[5]); u16* countIdxs = reinterpret_cast<u16*>(ptr + 24); *countIdxs = hecl::SBig(*countIdxs); for (u16 i = 0; i < *countIdxs; ++i) countIdxs[i + 1] = hecl::SBig(countIdxs[i + 1]); } } CAreaOctTree::CAreaOctTree(const zeus::CAABox& aabb, Node::ETreeType treeType, const u8* buf, const u8* treeBuf, u32 matCount, const u32* materials, const u8* vertMats, const u8* edgeMats, const u8* polyMats, u32 edgeCount, const CCollisionEdge* edges, u32 polyCount, const u16* polyEdges, u32 vertCount, const float* verts) : x0_aabb(aabb) , x18_treeType(treeType) , x1c_buf(buf) , x20_treeBuf(treeBuf) , x24_matCount(matCount) , x28_materials(materials) , x2c_vertMats(vertMats) , x30_edgeMats(edgeMats) , x34_polyMats(polyMats) , x38_edgeCount(edgeCount) , x3c_edges(edges) , x40_polyCount(polyCount) , x44_polyEdges(polyEdges) , x48_vertCount(vertCount) , x4c_verts(verts) { SwapTreeNode(const_cast<u8*>(x20_treeBuf), treeType); for (u32 i = 0; i < matCount; ++i) const_cast<u32*>(x28_materials)[i] = hecl::SBig(x28_materials[i]); for (u32 i = 0; i < edgeCount; ++i) const_cast<CCollisionEdge*>(x3c_edges)[i].swapBig(); for (u32 i = 0; i < polyCount; ++i) const_cast<u16*>(x44_polyEdges)[i] = hecl::SBig(x44_polyEdges[i]); for (u32 i = 0; i < vertCount * 3; ++i) const_cast<float*>(x4c_verts)[i] = hecl::SBig(x4c_verts[i]); } std::unique_ptr<CAreaOctTree> CAreaOctTree::MakeFromMemory(const u8* buf, unsigned int size) { athena::io::MemoryReader r(buf + 8, size - 8); r.readUint32Big(); r.readUint32Big(); zeus::CAABox aabb; aabb.readBoundingBoxBig(r); Node::ETreeType nodeType = Node::ETreeType(r.readUint32Big()); u32 treeSize = r.readUint32Big(); const u8* cur = reinterpret_cast<const u8*>(buf) + 8 + r.position(); const u8* treeBuf = cur; cur += treeSize; u32 matCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const u32* matBuf = reinterpret_cast<const u32*>(cur); cur += 4 * matCount; u32 vertMatsCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const u8* vertMatsBuf = cur; cur += vertMatsCount; u32 edgeMatsCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const u8* edgeMatsBuf = cur; cur += edgeMatsCount; u32 polyMatsCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const u8* polyMatsBuf = cur; cur += polyMatsCount; u32 edgeCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const CCollisionEdge* edgeBuf = reinterpret_cast<const CCollisionEdge*>(cur); cur += edgeCount * sizeof(edgeCount); u32 polyCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const u16* polyBuf = reinterpret_cast<const u16*>(cur); cur += polyCount * 2; u32 vertCount = hecl::SBig(*reinterpret_cast<const u32*>(cur)); cur += 4; const float* vertBuf = reinterpret_cast<const float*>(cur); return std::make_unique<CAreaOctTree>(aabb, nodeType, reinterpret_cast<const u8*>(buf + 8), treeBuf, matCount, matBuf, vertMatsBuf, edgeMatsBuf, polyMatsBuf, edgeCount, edgeBuf, polyCount, polyBuf, vertCount, vertBuf); } CCollisionSurface CAreaOctTree::GetMasterListTriangle(u16 idx) const { const CCollisionEdge& e0 = x3c_edges[x44_polyEdges[idx * 3]]; const CCollisionEdge& e1 = x3c_edges[x44_polyEdges[idx * 3 + 1]]; u16 vert2 = e1.GetVertIndex2(); if (e1.GetVertIndex1() != e0.GetVertIndex1()) if (e1.GetVertIndex1() != e0.GetVertIndex2()) vert2 = e1.GetVertIndex1(); u32 material = x28_materials[x34_polyMats[idx]]; if (material & 0x2000000) return CCollisionSurface(GetVert(e0.GetVertIndex2()), GetVert(e0.GetVertIndex1()), GetVert(vert2), material); else return CCollisionSurface(GetVert(e0.GetVertIndex1()), GetVert(e0.GetVertIndex2()), GetVert(vert2), material); } void CAreaOctTree::GetTriangleVertexIndices(u16 idx, u16 indicesOut[3]) const { const CCollisionEdge& e0 = x3c_edges[x44_polyEdges[idx * 3]]; const CCollisionEdge& e1 = x3c_edges[x44_polyEdges[idx * 3 + 1]]; indicesOut[2] = (e1.GetVertIndex1() != e0.GetVertIndex1() && e1.GetVertIndex1() != e0.GetVertIndex2()) ? e1.GetVertIndex1() : e1.GetVertIndex2(); u32 material = x28_materials[x34_polyMats[idx]]; if (material & 0x2000000) { indicesOut[0] = e0.GetVertIndex2(); indicesOut[1] = e0.GetVertIndex1(); } else { indicesOut[0] = e0.GetVertIndex1(); indicesOut[1] = e0.GetVertIndex2(); } } } // namespace metaforce