mirror of https://github.com/PrimeDecomp/prime.git
parent
2a8516d4cc
commit
1c3d63ed98
|
@ -0,0 +1,194 @@
|
||||||
|
#include "WorldFormat/CMetroidAreaCollider.hpp"
|
||||||
|
|
||||||
|
#include "Collision/CollisionUtil.hpp"
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
static uint gCalledClip = 0;
|
||||||
|
static uint gRejectedByClip = 0;
|
||||||
|
static uint gTrianglesProcessed = 0;
|
||||||
|
static uint gDupTrianglesProcessed = 0;
|
||||||
|
ushort CMetroidAreaCollider::sDupPrimitiveCheckCount = 0;
|
||||||
|
ushort CMetroidAreaCollider::sDupVertexList[0x2800];
|
||||||
|
ushort CMetroidAreaCollider::sDupEdgeList[0x6000];
|
||||||
|
ushort CMetroidAreaCollider::sDupTriangleList[0x4000];
|
||||||
|
|
||||||
|
static inline CVector3f ClipRayToPlane(const CVector3f& a, const CVector3f& b,
|
||||||
|
const CPlane& plane) {
|
||||||
|
float f = -plane.GetHeight(a) / CVector3f::Dot(b - a, plane.GetNormal());
|
||||||
|
return (a - b) * (1.f - f);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CMetroidAreaCollider::ConvexPolyCollision(const CPlane* planes, const CVector3f* verts,
|
||||||
|
CAABox& aabb) {
|
||||||
|
typedef rstl::reserved_vector< CVector3f, 20 > ClipVec;
|
||||||
|
++gCalledClip;
|
||||||
|
++gRejectedByClip;
|
||||||
|
|
||||||
|
ClipVec vecs[2];
|
||||||
|
int vecIdx = 0;
|
||||||
|
int otherVecIdx = 1;
|
||||||
|
|
||||||
|
vecs[0].push_back(verts[0]);
|
||||||
|
vecs[0].push_back(verts[1]);
|
||||||
|
vecs[0].push_back(verts[2]);
|
||||||
|
|
||||||
|
for (int i = 0; i < 6; ++i) {
|
||||||
|
ClipVec& vec = vecs[vecIdx];
|
||||||
|
ClipVec& otherVec = vecs[otherVecIdx];
|
||||||
|
otherVec.clear();
|
||||||
|
|
||||||
|
bool inFrontOf = planes[i].GetHeight(vec.front()) >= 0.f;
|
||||||
|
for (int j = 0; j < vec.size(); ++j) {
|
||||||
|
const CVector3f& b = vec[j == vec.size() - 1 ? 0 : j + 1];
|
||||||
|
if (inFrontOf) {
|
||||||
|
otherVec.push_back(vec[j]);
|
||||||
|
}
|
||||||
|
bool nextInFrontOf = planes[i].GetHeight(b) >= 0.f;
|
||||||
|
if (nextInFrontOf ^ inFrontOf) {
|
||||||
|
otherVec.push_back(ClipRayToPlane(vec[j], b, planes[i]) + b);
|
||||||
|
}
|
||||||
|
inFrontOf = nextInFrontOf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (otherVec.empty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
otherVecIdx ^= 1;
|
||||||
|
vecIdx ^= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
ClipVec& accumVec = vecs[otherVecIdx ^ 1];
|
||||||
|
for (ClipVec::const_iterator it = accumVec.begin(); it != accumVec.end(); ++it) {
|
||||||
|
aabb.AccumulateBounds(*it);
|
||||||
|
}
|
||||||
|
|
||||||
|
--gRejectedByClip;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CMetroidAreaCollider::ResetInternalCounters() {
|
||||||
|
gCalledClip = 0;
|
||||||
|
gRejectedByClip = 0;
|
||||||
|
gTrianglesProcessed = 0;
|
||||||
|
gDupTrianglesProcessed = 0;
|
||||||
|
if (sDupPrimitiveCheckCount == 0xffff) {
|
||||||
|
memset(sDupVertexList, 0, sizeof(sDupVertexList));
|
||||||
|
memset(sDupEdgeList, 0, sizeof(sDupEdgeList));
|
||||||
|
memset(sDupTriangleList, 0, sizeof(sDupTriangleList));
|
||||||
|
++sDupPrimitiveCheckCount;
|
||||||
|
}
|
||||||
|
++sDupPrimitiveCheckCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CMetroidAreaCollider::AABoxCollisionCheck_Internal(const CAreaOctTree::Node& node,
|
||||||
|
CAABoxAreaCache& cache) {
|
||||||
|
bool ret = false;
|
||||||
|
|
||||||
|
switch (node.GetTreeType()) {
|
||||||
|
case CAreaOctTree::Node::kTT_Invalid:
|
||||||
|
return false;
|
||||||
|
case CAreaOctTree::Node::kTT_Branch: {
|
||||||
|
for (int i = 0; i < 8; ++i) {
|
||||||
|
CAreaOctTree::Node ch = node.GetChild(i);
|
||||||
|
CAABox box = ch.GetBoundingBox();
|
||||||
|
if (box.DoBoundsOverlap(cache.x0_aabb))
|
||||||
|
if (AABoxCollisionCheck_Internal(ch, cache))
|
||||||
|
ret = true;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case CAreaOctTree::Node::kTT_Leaf: {
|
||||||
|
CAreaOctTree::TriListReference list = node.GetTriangleArray();
|
||||||
|
const CAreaOctTree& owner = node.GetOwner();
|
||||||
|
const CMaterialFilter& filter = cache.x8_filter;
|
||||||
|
const CPlane* planes = cache.x4_planes;
|
||||||
|
for (int j = 0; j < list.GetSize(); ++j) {
|
||||||
|
++gTrianglesProcessed;
|
||||||
|
ushort triIdx = list.GetAt(j);
|
||||||
|
if (sDupPrimitiveCheckCount == sDupTriangleList[triIdx]) {
|
||||||
|
++gDupTrianglesProcessed;
|
||||||
|
} else {
|
||||||
|
sDupTriangleList[triIdx] = sDupPrimitiveCheckCount;
|
||||||
|
const CCollisionSurface& surf = owner.GetMasterListTriangle(triIdx);
|
||||||
|
CMaterialList material(surf.GetSurfaceFlags());
|
||||||
|
if (filter.Passes(material)) {
|
||||||
|
if (CollisionUtil::TriBoxOverlap(cache.x14_center, cache.x20_halfExtent, surf.GetVert(0),
|
||||||
|
surf.GetVert(1), surf.GetVert(2)) == true) {
|
||||||
|
CAABox aabb = CAABox::MakeMaxInvertedBox();
|
||||||
|
if (ConvexPolyCollision(planes, &surf.GetVert(0), aabb)) {
|
||||||
|
CPlane plane = surf.GetPlane();
|
||||||
|
CCollisionInfo collision(aabb, cache.xc_material, material, plane.GetNormal(),
|
||||||
|
-plane.GetNormal());
|
||||||
|
cache.x10_collisionList.Add(collision, false);
|
||||||
|
ret = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool CMetroidAreaCollider::AABoxCollisionCheck_Cached(const COctreeLeafCache& leafCache,
|
||||||
|
const CAABox& aabb,
|
||||||
|
const CMaterialFilter& filter,
|
||||||
|
const CMaterialList& matList,
|
||||||
|
CCollisionInfoList& list) {
|
||||||
|
bool ret = false;
|
||||||
|
const CUnitVector3f right(1.f, 0.f, 0.f, CUnitVector3f::kN_No);
|
||||||
|
const CUnitVector3f forward(0.f, 1.f, 0.f, CUnitVector3f::kN_No);
|
||||||
|
const CUnitVector3f up(0.f, 0.f, 1.f, CUnitVector3f::kN_No);
|
||||||
|
const CPlane planes[6] = {
|
||||||
|
CPlane(aabb.GetMinPoint(), right),
|
||||||
|
CPlane(aabb.GetMaxPoint(), -right),
|
||||||
|
CPlane(aabb.GetMinPoint(), forward),
|
||||||
|
CPlane(aabb.GetMaxPoint(), -forward),
|
||||||
|
CPlane(aabb.GetMinPoint(), up),
|
||||||
|
CPlane(aabb.GetMaxPoint(), -up),
|
||||||
|
};
|
||||||
|
|
||||||
|
ResetInternalCounters();
|
||||||
|
CVector3f center = aabb.GetCenterPoint();
|
||||||
|
CVector3f halfExtent = aabb.GetHalfExtent();
|
||||||
|
|
||||||
|
for (int i = 0; i < leafCache.GetNumLeaves(); ++i) {
|
||||||
|
const CAreaOctTree::Node& node = leafCache.GetLeaf(i);
|
||||||
|
if (aabb.DoBoundsOverlap(node.GetBoundingBox())) {
|
||||||
|
CAreaOctTree::TriListReference listRef = node.GetTriangleArray();
|
||||||
|
for (int j = 0; j < listRef.GetSize(); ++j) {
|
||||||
|
++gTrianglesProcessed;
|
||||||
|
ushort triIdx = listRef.GetAt(j);
|
||||||
|
if (sDupPrimitiveCheckCount == sDupTriangleList[triIdx]) {
|
||||||
|
++gDupTrianglesProcessed;
|
||||||
|
} else {
|
||||||
|
sDupTriangleList[triIdx] = sDupPrimitiveCheckCount;
|
||||||
|
const CCollisionSurface& surf = node.GetOwner().GetMasterListTriangle(triIdx);
|
||||||
|
CMaterialList material(surf.GetSurfaceFlags());
|
||||||
|
if (filter.Passes(material)) {
|
||||||
|
if (CollisionUtil::TriBoxOverlap(center, halfExtent, surf.GetVert(0), surf.GetVert(1),
|
||||||
|
surf.GetVert(2)) == true) {
|
||||||
|
CAABox aabb2 = CAABox::MakeMaxInvertedBox();
|
||||||
|
if (ConvexPolyCollision(planes, &surf.GetVert(0), aabb2)) {
|
||||||
|
CPlane plane = surf.GetPlane();
|
||||||
|
CCollisionInfo collision(aabb2, matList, material, plane.GetNormal(),
|
||||||
|
-plane.GetNormal());
|
||||||
|
list.Add(collision, false);
|
||||||
|
ret = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
Loading…
Reference in New Issue