/******************************************************************************** * ReactPhysics3D physics library, http://www.reactphysics3d.com * * Copyright (c) 2010-2018 Daniel Chappuis * ********************************************************************************* * * * This software is provided 'as-is', without any express or implied warranty. * * In no event will the authors be held liable for any damages arising from the * * use of this software. * * * * Permission is granted to anyone to use this software for any purpose, * * including commercial applications, and to alter it and redistribute it * * freely, subject to the following restrictions: * * * * 1. The origin of this software must not be misrepresented; you must not claim * * that you wrote the original software. If you use this software in a * * product, an acknowledgment in the product documentation would be * * appreciated but is not required. * * * * 2. Altered source versions must be plainly marked as such, and must not be * * misrepresented as being the original software. * * * * 3. This notice may not be removed or altered from any source distribution. * * * ********************************************************************************/ // Libraries #include "systems/CollisionDetectionSystem.h" #include "engine/CollisionWorld.h" #include "collision/OverlapCallback.h" #include "collision/shapes/BoxShape.h" #include "collision/shapes/ConcaveShape.h" #include "collision/ContactManifoldInfo.h" #include "constraint/ContactPoint.h" #include "body/RigidBody.h" #include "configuration.h" #include "collision/CollisionCallback.h" #include "collision/OverlapCallback.h" #include "collision/narrowphase/NarrowPhaseInfoBatch.h" #include "collision/ContactManifold.h" #include "utils/Profiler.h" #include "engine/EventListener.h" #include "collision/RaycastInfo.h" #include "engine/Islands.h" #include "containers/Pair.h" #include #include // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; using namespace std; // Constructor CollisionDetectionSystem::CollisionDetectionSystem(CollisionWorld* world, ColliderComponents& collidersComponents, TransformComponents& transformComponents, CollisionBodyComponents& collisionBodyComponents, RigidBodyComponents& rigidBodyComponents, MemoryManager& memoryManager) : mMemoryManager(memoryManager), mCollidersComponents(collidersComponents), mCollisionDispatch(mMemoryManager.getPoolAllocator()), mWorld(world), mNoCollisionPairs(mMemoryManager.getPoolAllocator()), mOverlappingPairs(mMemoryManager.getPoolAllocator(), mMemoryManager.getSingleFrameAllocator(), mCollidersComponents, collisionBodyComponents, rigidBodyComponents, mNoCollisionPairs, mCollisionDispatch), mBroadPhaseSystem(*this, mCollidersComponents, transformComponents, rigidBodyComponents), mMapBroadPhaseIdToColliderEntity(memoryManager.getPoolAllocator()), mNarrowPhaseInput(mMemoryManager.getSingleFrameAllocator(), mOverlappingPairs), mPotentialContactPoints(mMemoryManager.getSingleFrameAllocator()), mPotentialContactManifolds(mMemoryManager.getSingleFrameAllocator()), mContactPairs1(mMemoryManager.getPoolAllocator()), mContactPairs2(mMemoryManager.getPoolAllocator()), mPreviousContactPairs(&mContactPairs1), mCurrentContactPairs(&mContactPairs2), mMapPairIdToContactPairIndex1(mMemoryManager.getPoolAllocator()), mMapPairIdToContactPairIndex2(mMemoryManager.getPoolAllocator()), mPreviousMapPairIdToContactPairIndex(&mMapPairIdToContactPairIndex1), mCurrentMapPairIdToContactPairIndex(&mMapPairIdToContactPairIndex2), mContactManifolds1(mMemoryManager.getPoolAllocator()), mContactManifolds2(mMemoryManager.getPoolAllocator()), mPreviousContactManifolds(&mContactManifolds1), mCurrentContactManifolds(&mContactManifolds2), mContactPoints1(mMemoryManager.getPoolAllocator()), mContactPoints2(mMemoryManager.getPoolAllocator()), mPreviousContactPoints(&mContactPoints1), mCurrentContactPoints(&mContactPoints2), mMapBodyToContactPairs(mMemoryManager.getSingleFrameAllocator()) { #ifdef IS_PROFILING_ACTIVE mProfiler = nullptr; mCollisionDispatch.setProfiler(mProfiler); mOverlappingPairs.setProfiler(mProfiler); #endif } // Compute the collision detection void CollisionDetectionSystem::computeCollisionDetection() { RP3D_PROFILE("CollisionDetectionSystem::computeCollisionDetection()", mProfiler); // Compute the broad-phase collision detection computeBroadPhase(); // Compute the middle-phase collision detection computeMiddlePhase(mNarrowPhaseInput); // Compute the narrow-phase collision detection computeNarrowPhase(); } // Compute the broad-phase collision detection void CollisionDetectionSystem::computeBroadPhase() { RP3D_PROFILE("CollisionDetectionSystem::computeBroadPhase()", mProfiler); // Ask the broad-phase to compute all the shapes overlapping the shapes that // have moved or have been added in the last frame. This call can only add new // overlapping pairs in the collision detection. List> overlappingNodes(mMemoryManager.getPoolAllocator(), 32); mBroadPhaseSystem.computeOverlappingPairs(mMemoryManager, overlappingNodes); // Create new overlapping pairs if necessary updateOverlappingPairs(overlappingNodes); // Remove non overlapping pairs removeNonOverlappingPairs(); } // Remove pairs that are not overlapping anymore void CollisionDetectionSystem::removeNonOverlappingPairs() { RP3D_PROFILE("CollisionDetectionSystem::removeNonOverlappingPairs()", mProfiler); // For each possible convex vs convex pair of bodies for (uint64 i=0; i < mOverlappingPairs.getNbPairs(); i++) { // Check if we need to test overlap. If so, test if the two shapes are still overlapping. // Otherwise, we destroy the overlapping pair if (mOverlappingPairs.mNeedToTestOverlap[i]) { if(mBroadPhaseSystem.testOverlappingShapes(mOverlappingPairs.mPairBroadPhaseId1[i], mOverlappingPairs.mPairBroadPhaseId2[i])) { mOverlappingPairs.mNeedToTestOverlap[i] = false; } else { mOverlappingPairs.removePair(mOverlappingPairs.mPairIds[i]); i--; } } } } // Take a list of overlapping nodes in the broad-phase and create new overlapping pairs if necessary void CollisionDetectionSystem::updateOverlappingPairs(const List>& overlappingNodes) { RP3D_PROFILE("CollisionDetectionSystem::updateOverlappingPairs()", mProfiler); // For each overlapping pair of nodes for (uint i=0; i < overlappingNodes.size(); i++) { Pair nodePair = overlappingNodes[i]; assert(nodePair.first != -1); assert(nodePair.second != -1); // Skip pairs with same overlapping nodes if (nodePair.first != nodePair.second) { // Get the two colliders const Entity collider1Entity = mMapBroadPhaseIdToColliderEntity[nodePair.first]; const Entity collider2Entity = mMapBroadPhaseIdToColliderEntity[nodePair.second]; const uint collider1Index = mCollidersComponents.getEntityIndex(collider1Entity); const uint collider2Index = mCollidersComponents.getEntityIndex(collider2Entity); // Get the two bodies const Entity body1Entity = mCollidersComponents.mBodiesEntities[collider1Index]; const Entity body2Entity = mCollidersComponents.mBodiesEntities[collider2Index]; // If the two colliders are from the same body, skip it if (body1Entity != body2Entity) { // Compute the overlapping pair ID const uint64 pairId = pairNumbers(std::max(nodePair.first, nodePair.second), std::min(nodePair.first, nodePair.second)); // Check if the overlapping pair already exists auto it = mOverlappingPairs.mMapPairIdToPairIndex.find(pairId); if (it == mOverlappingPairs.mMapPairIdToPairIndex.end()) { const unsigned short shape1CollideWithMaskBits = mCollidersComponents.mCollideWithMaskBits[collider1Index]; const unsigned short shape2CollideWithMaskBits = mCollidersComponents.mCollideWithMaskBits[collider2Index]; const unsigned short shape1CollisionCategoryBits = mCollidersComponents.mCollisionCategoryBits[collider1Index]; const unsigned short shape2CollisionCategoryBits = mCollidersComponents.mCollisionCategoryBits[collider2Index]; // Check if the collision filtering allows collision between the two shapes if ((shape1CollideWithMaskBits & shape2CollisionCategoryBits) != 0 && (shape1CollisionCategoryBits & shape2CollideWithMaskBits) != 0) { Collider* shape1 = mCollidersComponents.mColliders[collider1Index]; Collider* shape2 = mCollidersComponents.mColliders[collider2Index]; // Check that at least one collision shape is convex if (shape1->getCollisionShape()->isConvex() || shape2->getCollisionShape()->isConvex()) { // Add the new overlapping pair mOverlappingPairs.addPair(shape1, shape2); } } } else { // We do not need to test the pair for overlap because it has just been reported that they still overlap mOverlappingPairs.mNeedToTestOverlap[it->second] = false; } } } } } // Compute the middle-phase collision detection void CollisionDetectionSystem::computeMiddlePhase(NarrowPhaseInput& narrowPhaseInput) { RP3D_PROFILE("CollisionDetectionSystem::computeMiddlePhase()", mProfiler); // Reserve memory for the narrow-phase input using cached capacity from previous frame narrowPhaseInput.reserveMemory(); // Remove the obsolete last frame collision infos and mark all the others as obsolete mOverlappingPairs.clearObsoleteLastFrameCollisionInfos(); // For each possible convex vs convex pair of bodies for (uint64 i=0; i < mOverlappingPairs.getNbConvexVsConvexPairs(); i++) { assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != -1); assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i]) != -1); assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i])); // Check that at least one body is enabled (active and awake) and not static if (mOverlappingPairs.mIsActive[i]) { const Entity collider1Entity = mOverlappingPairs.mColliders1[i]; const Entity collider2Entity = mOverlappingPairs.mColliders2[i]; const uint collider1Index = mCollidersComponents.getEntityIndex(collider1Entity); const uint collider2Index = mCollidersComponents.getEntityIndex(collider2Entity); CollisionShape* collisionShape1 = mCollidersComponents.mCollisionShapes[collider1Index]; CollisionShape* collisionShape2 = mCollidersComponents.mCollisionShapes[collider2Index]; NarrowPhaseAlgorithmType algorithmType = mOverlappingPairs.mNarrowPhaseAlgorithmType[i]; // No middle-phase is necessary, simply create a narrow phase info // for the narrow-phase collision detection narrowPhaseInput.addNarrowPhaseTest(mOverlappingPairs.mPairIds[i], i, collider1Entity, collider2Entity, collisionShape1, collisionShape2, mCollidersComponents.mLocalToWorldTransforms[collider1Index], mCollidersComponents.mLocalToWorldTransforms[collider2Index], algorithmType, mMemoryManager.getSingleFrameAllocator()); } } // For each possible convex vs concave pair of bodies const uint64 convexVsConcaveStartIndex = mOverlappingPairs.getConvexVsConcavePairsStartIndex(); for (uint64 i=convexVsConcaveStartIndex; i < convexVsConcaveStartIndex + mOverlappingPairs.getNbConvexVsConcavePairs(); i++) { assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != -1); assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i]) != -1); assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[i]) != mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[i])); // Check that at least one body is enabled (active and awake) and not static if (mOverlappingPairs.mIsActive[i]) { computeConvexVsConcaveMiddlePhase(i, mMemoryManager.getSingleFrameAllocator(), narrowPhaseInput); } } } // Compute the middle-phase collision detection void CollisionDetectionSystem::computeMiddlePhaseCollisionSnapshot(List& convexPairs, List& concavePairs, NarrowPhaseInput& narrowPhaseInput) { RP3D_PROFILE("CollisionDetectionSystem::computeMiddlePhase()", mProfiler); // Reserve memory for the narrow-phase input using cached capacity from previous frame narrowPhaseInput.reserveMemory(); // Remove the obsolete last frame collision infos and mark all the others as obsolete mOverlappingPairs.clearObsoleteLastFrameCollisionInfos(); // For each possible convex vs convex pair of bodies for (uint64 p=0; p < convexPairs.size(); p++) { const uint64 pairId = convexPairs[p]; const uint64 pairIndex = mOverlappingPairs.mMapPairIdToPairIndex[pairId]; assert(pairIndex < mOverlappingPairs.getNbPairs()); const Entity collider1Entity = mOverlappingPairs.mColliders1[pairIndex]; const Entity collider2Entity = mOverlappingPairs.mColliders2[pairIndex]; const uint collider1Index = mCollidersComponents.getEntityIndex(collider1Entity); const uint collider2Index = mCollidersComponents.getEntityIndex(collider2Entity); assert(mCollidersComponents.getBroadPhaseId(collider1Entity) != -1); assert(mCollidersComponents.getBroadPhaseId(collider2Entity) != -1); assert(mCollidersComponents.getBroadPhaseId(collider1Entity) != mCollidersComponents.getBroadPhaseId(collider2Entity)); CollisionShape* collisionShape1 = mCollidersComponents.mCollisionShapes[collider1Index]; CollisionShape* collisionShape2 = mCollidersComponents.mCollisionShapes[collider2Index]; NarrowPhaseAlgorithmType algorithmType = mOverlappingPairs.mNarrowPhaseAlgorithmType[pairIndex]; // No middle-phase is necessary, simply create a narrow phase info // for the narrow-phase collision detection narrowPhaseInput.addNarrowPhaseTest(pairId, pairIndex, collider1Entity, collider2Entity, collisionShape1, collisionShape2, mCollidersComponents.mLocalToWorldTransforms[collider1Index], mCollidersComponents.mLocalToWorldTransforms[collider2Index], algorithmType, mMemoryManager.getSingleFrameAllocator()); } // For each possible convex vs concave pair of bodies for (uint p=0; p < concavePairs.size(); p++) { const uint64 pairId = concavePairs[p]; const uint64 pairIndex = mOverlappingPairs.mMapPairIdToPairIndex[pairId]; assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[pairIndex]) != -1); assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[pairIndex]) != -1); assert(mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders1[pairIndex]) != mCollidersComponents.getBroadPhaseId(mOverlappingPairs.mColliders2[pairIndex])); computeConvexVsConcaveMiddlePhase(pairIndex, mMemoryManager.getSingleFrameAllocator(), narrowPhaseInput); } } // Compute the concave vs convex middle-phase algorithm for a given pair of bodies void CollisionDetectionSystem::computeConvexVsConcaveMiddlePhase(uint64 pairIndex, MemoryAllocator& allocator, NarrowPhaseInput& narrowPhaseInput) { RP3D_PROFILE("CollisionDetectionSystem::computeConvexVsConcaveMiddlePhase()", mProfiler); const Entity collider1 = mOverlappingPairs.mColliders1[pairIndex]; const Entity collider2 = mOverlappingPairs.mColliders2[pairIndex]; const uint collider1Index = mCollidersComponents.getEntityIndex(collider1); const uint collider2Index = mCollidersComponents.getEntityIndex(collider2); Transform& shape1LocalToWorldTransform = mCollidersComponents.mLocalToWorldTransforms[collider1Index]; Transform& shape2LocalToWorldTransform = mCollidersComponents.mLocalToWorldTransforms[collider2Index]; Transform convexToConcaveTransform; // Collision shape 1 is convex, collision shape 2 is concave ConvexShape* convexShape; ConcaveShape* concaveShape; const bool isShape1Convex = mOverlappingPairs.mIsShape1Convex[pairIndex]; if (isShape1Convex) { convexShape = static_cast(mCollidersComponents.mCollisionShapes[collider1Index]); concaveShape = static_cast(mCollidersComponents.mCollisionShapes[collider2Index]); convexToConcaveTransform = shape2LocalToWorldTransform.getInverse() * shape1LocalToWorldTransform; } else { // Collision shape 2 is convex, collision shape 1 is concave convexShape = static_cast(mCollidersComponents.mCollisionShapes[collider1Index]); concaveShape = static_cast(mCollidersComponents.mCollisionShapes[collider2Index]); convexToConcaveTransform = shape1LocalToWorldTransform.getInverse() * shape2LocalToWorldTransform; } assert(mOverlappingPairs.mNarrowPhaseAlgorithmType[pairIndex] != NarrowPhaseAlgorithmType::None); // Compute the convex shape AABB in the local-space of the convex shape AABB aabb; convexShape->computeAABB(aabb, convexToConcaveTransform); // Compute the concave shape triangles that are overlapping with the convex mesh AABB List triangleVertices(allocator); List triangleVerticesNormals(allocator); List shapeIds(allocator); concaveShape->computeOverlappingTriangles(aabb, triangleVertices, triangleVerticesNormals, shapeIds, allocator); assert(triangleVertices.size() == triangleVerticesNormals.size()); assert(shapeIds.size() == triangleVertices.size() / 3); assert(triangleVertices.size() % 3 == 0); assert(triangleVerticesNormals.size() % 3 == 0); // For each overlapping triangle for (uint i=0; i < shapeIds.size(); i++) { // Create a triangle collision shape (the allocated memory for the TriangleShape will be released in the // destructor of the corresponding NarrowPhaseInfo. TriangleShape* triangleShape = new (allocator.allocate(sizeof(TriangleShape))) TriangleShape(&(triangleVertices[i * 3]), &(triangleVerticesNormals[i * 3]), shapeIds[i], allocator); #ifdef IS_PROFILING_ACTIVE // Set the profiler to the triangle shape triangleShape->setProfiler(mProfiler); #endif // Create a narrow phase info for the narrow-phase collision detection narrowPhaseInput.addNarrowPhaseTest(mOverlappingPairs.mPairIds[pairIndex], pairIndex, collider1, collider2, isShape1Convex ? convexShape : triangleShape, isShape1Convex ? triangleShape : convexShape, shape1LocalToWorldTransform, shape2LocalToWorldTransform, mOverlappingPairs.mNarrowPhaseAlgorithmType[pairIndex], allocator); } } // Execute the narrow-phase collision detection algorithm on batches bool CollisionDetectionSystem::testNarrowPhaseCollision(NarrowPhaseInput& narrowPhaseInput, bool reportContacts, bool clipWithPreviousAxisIfStillColliding, MemoryAllocator& allocator) { bool contactFound = false; // Get the narrow-phase collision detection algorithms for each kind of collision shapes SphereVsSphereAlgorithm* sphereVsSphereAlgo = mCollisionDispatch.getSphereVsSphereAlgorithm(); SphereVsCapsuleAlgorithm* sphereVsCapsuleAlgo = mCollisionDispatch.getSphereVsCapsuleAlgorithm(); CapsuleVsCapsuleAlgorithm* capsuleVsCapsuleAlgo = mCollisionDispatch.getCapsuleVsCapsuleAlgorithm(); SphereVsConvexPolyhedronAlgorithm* sphereVsConvexPolyAlgo = mCollisionDispatch.getSphereVsConvexPolyhedronAlgorithm(); CapsuleVsConvexPolyhedronAlgorithm* capsuleVsConvexPolyAlgo = mCollisionDispatch.getCapsuleVsConvexPolyhedronAlgorithm(); ConvexPolyhedronVsConvexPolyhedronAlgorithm* convexPolyVsConvexPolyAlgo = mCollisionDispatch.getConvexPolyhedronVsConvexPolyhedronAlgorithm(); // get the narrow-phase batches to test for collision SphereVsSphereNarrowPhaseInfoBatch& sphereVsSphereBatch = narrowPhaseInput.getSphereVsSphereBatch(); SphereVsCapsuleNarrowPhaseInfoBatch& sphereVsCapsuleBatch = narrowPhaseInput.getSphereVsCapsuleBatch(); CapsuleVsCapsuleNarrowPhaseInfoBatch& capsuleVsCapsuleBatch = narrowPhaseInput.getCapsuleVsCapsuleBatch(); NarrowPhaseInfoBatch& sphereVsConvexPolyhedronBatch = narrowPhaseInput.getSphereVsConvexPolyhedronBatch(); NarrowPhaseInfoBatch& capsuleVsConvexPolyhedronBatch = narrowPhaseInput.getCapsuleVsConvexPolyhedronBatch(); NarrowPhaseInfoBatch& convexPolyhedronVsConvexPolyhedronBatch = narrowPhaseInput.getConvexPolyhedronVsConvexPolyhedronBatch(); // Compute the narrow-phase collision detection for each kind of collision shapes if (sphereVsSphereBatch.getNbObjects() > 0) { contactFound |= sphereVsSphereAlgo->testCollision(sphereVsSphereBatch, 0, sphereVsSphereBatch.getNbObjects(), reportContacts, allocator); } if (sphereVsCapsuleBatch.getNbObjects() > 0) { contactFound |= sphereVsCapsuleAlgo->testCollision(sphereVsCapsuleBatch, 0, sphereVsCapsuleBatch.getNbObjects(), reportContacts, allocator); } if (capsuleVsCapsuleBatch.getNbObjects() > 0) { contactFound |= capsuleVsCapsuleAlgo->testCollision(capsuleVsCapsuleBatch, 0, capsuleVsCapsuleBatch.getNbObjects(), reportContacts, allocator); } if (sphereVsConvexPolyhedronBatch.getNbObjects() > 0) { contactFound |= sphereVsConvexPolyAlgo->testCollision(sphereVsConvexPolyhedronBatch, 0, sphereVsConvexPolyhedronBatch.getNbObjects(), reportContacts, clipWithPreviousAxisIfStillColliding, allocator); } if (capsuleVsConvexPolyhedronBatch.getNbObjects() > 0) { contactFound |= capsuleVsConvexPolyAlgo->testCollision(capsuleVsConvexPolyhedronBatch, 0, capsuleVsConvexPolyhedronBatch.getNbObjects(), reportContacts, clipWithPreviousAxisIfStillColliding, allocator); } if (convexPolyhedronVsConvexPolyhedronBatch.getNbObjects() > 0) { contactFound |= convexPolyVsConvexPolyAlgo->testCollision(convexPolyhedronVsConvexPolyhedronBatch, 0, convexPolyhedronVsConvexPolyhedronBatch.getNbObjects(), reportContacts, clipWithPreviousAxisIfStillColliding, allocator); } return contactFound; } // Process the potential contacts after narrow-phase collision detection void CollisionDetectionSystem::processAllPotentialContacts(NarrowPhaseInput& narrowPhaseInput, bool updateLastFrameInfo, List& potentialContactPoints, Map* mapPairIdToContactPairIndex, List& potentialContactManifolds, List* contactPairs, Map>& mapBodyToContactPairs) { assert(contactPairs->size() == 0); assert(mapPairIdToContactPairIndex->size() == 0); // get the narrow-phase batches to test for collision NarrowPhaseInfoBatch& sphereVsSphereBatch = narrowPhaseInput.getSphereVsSphereBatch(); NarrowPhaseInfoBatch& sphereVsCapsuleBatch = narrowPhaseInput.getSphereVsCapsuleBatch(); NarrowPhaseInfoBatch& capsuleVsCapsuleBatch = narrowPhaseInput.getCapsuleVsCapsuleBatch(); NarrowPhaseInfoBatch& sphereVsConvexPolyhedronBatch = narrowPhaseInput.getSphereVsConvexPolyhedronBatch(); NarrowPhaseInfoBatch& capsuleVsConvexPolyhedronBatch = narrowPhaseInput.getCapsuleVsConvexPolyhedronBatch(); NarrowPhaseInfoBatch& convexPolyhedronVsConvexPolyhedronBatch = narrowPhaseInput.getConvexPolyhedronVsConvexPolyhedronBatch(); // Process the potential contacts processPotentialContacts(sphereVsSphereBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex, potentialContactManifolds, contactPairs, mapBodyToContactPairs); processPotentialContacts(sphereVsCapsuleBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex, potentialContactManifolds, contactPairs, mapBodyToContactPairs); processPotentialContacts(capsuleVsCapsuleBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex, potentialContactManifolds, contactPairs, mapBodyToContactPairs); processPotentialContacts(sphereVsConvexPolyhedronBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex, potentialContactManifolds, contactPairs, mapBodyToContactPairs); processPotentialContacts(capsuleVsConvexPolyhedronBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex, potentialContactManifolds, contactPairs, mapBodyToContactPairs); processPotentialContacts(convexPolyhedronVsConvexPolyhedronBatch, updateLastFrameInfo, potentialContactPoints, mapPairIdToContactPairIndex, potentialContactManifolds, contactPairs, mapBodyToContactPairs); } // Compute the narrow-phase collision detection void CollisionDetectionSystem::computeNarrowPhase() { RP3D_PROFILE("CollisionDetectionSystem::computeNarrowPhase()", mProfiler); MemoryAllocator& allocator = mMemoryManager.getSingleFrameAllocator(); // Swap the previous and current contacts lists swapPreviousAndCurrentContacts(); // Test the narrow-phase collision detection on the batches to be tested testNarrowPhaseCollision(mNarrowPhaseInput, true, true, allocator); // Process all the potential contacts after narrow-phase collision processAllPotentialContacts(mNarrowPhaseInput, true, mPotentialContactPoints, mCurrentMapPairIdToContactPairIndex, mPotentialContactManifolds, mCurrentContactPairs, mMapBodyToContactPairs); // Reduce the number of contact points in the manifolds reducePotentialContactManifolds(mCurrentContactPairs, mPotentialContactManifolds, mPotentialContactPoints); assert(mCurrentContactManifolds->size() == 0); assert(mCurrentContactPoints->size() == 0); // Create the actual narrow-phase contacts createContacts(); mNarrowPhaseInput.clear(); } // Compute the narrow-phase collision detection for the testOverlap() methods. /// This method returns true if contacts are found. bool CollisionDetectionSystem::computeNarrowPhaseOverlapSnapshot(NarrowPhaseInput& narrowPhaseInput, OverlapCallback* callback) { RP3D_PROFILE("CollisionDetectionSystem::computeNarrowPhaseOverlapSnapshot()", mProfiler); MemoryAllocator& allocator = mMemoryManager.getPoolAllocator(); // Test the narrow-phase collision detection on the batches to be tested bool collisionFound = testNarrowPhaseCollision(narrowPhaseInput, false, false, allocator); if (collisionFound && callback != nullptr) { // Compute the overlapping bodies List> overlapBodies(allocator); computeSnapshotContactPairs(narrowPhaseInput, overlapBodies); // Report overlapping bodies OverlapCallback::CallbackData callbackData(overlapBodies, *mWorld); (*callback).onOverlap(callbackData); } return collisionFound; } // Process the potential overlapping bodies for the testOverlap() methods void CollisionDetectionSystem::computeSnapshotContactPairs(NarrowPhaseInput& narrowPhaseInput, List>& overlapPairs) const { // get the narrow-phase batches to test for collision NarrowPhaseInfoBatch& sphereVsSphereBatch = narrowPhaseInput.getSphereVsSphereBatch(); NarrowPhaseInfoBatch& sphereVsCapsuleBatch = narrowPhaseInput.getSphereVsCapsuleBatch(); NarrowPhaseInfoBatch& capsuleVsCapsuleBatch = narrowPhaseInput.getCapsuleVsCapsuleBatch(); NarrowPhaseInfoBatch& sphereVsConvexPolyhedronBatch = narrowPhaseInput.getSphereVsConvexPolyhedronBatch(); NarrowPhaseInfoBatch& capsuleVsConvexPolyhedronBatch = narrowPhaseInput.getCapsuleVsConvexPolyhedronBatch(); NarrowPhaseInfoBatch& convexPolyhedronVsConvexPolyhedronBatch = narrowPhaseInput.getConvexPolyhedronVsConvexPolyhedronBatch(); // Process the potential contacts computeSnapshotContactPairs(sphereVsSphereBatch, overlapPairs); computeSnapshotContactPairs(sphereVsCapsuleBatch, overlapPairs); computeSnapshotContactPairs(capsuleVsCapsuleBatch, overlapPairs); computeSnapshotContactPairs(sphereVsConvexPolyhedronBatch, overlapPairs); computeSnapshotContactPairs(capsuleVsConvexPolyhedronBatch, overlapPairs); computeSnapshotContactPairs(convexPolyhedronVsConvexPolyhedronBatch, overlapPairs); } // Notify that the overlapping pairs where a given collider is involved need to be tested for overlap void CollisionDetectionSystem::notifyOverlappingPairsToTestOverlap(Collider* collider) { // Get the overlapping pairs involved with this collider List& overlappingPairs = mCollidersComponents.getOverlappingPairs(collider->getEntity()); for (uint i=0; i < overlappingPairs.size(); i++) { // Notify that the overlapping pair needs to be testbed for overlap mOverlappingPairs.setNeedToTestOverlap(overlappingPairs[i], true); } } // Convert the potential overlapping bodies for the testOverlap() methods void CollisionDetectionSystem::computeSnapshotContactPairs(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, List>& overlapPairs) const { RP3D_PROFILE("CollisionDetectionSystem::computeSnapshotContactPairs()", mProfiler); // For each narrow phase info object for(uint i=0; i < narrowPhaseInfoBatch.getNbObjects(); i++) { // If there is a contact if (narrowPhaseInfoBatch.isColliding[i]) { // Add the pair of bodies in contact into the list overlapPairs.add(Pair(mCollidersComponents.getBody(narrowPhaseInfoBatch.colliderEntities1[i]), mCollidersComponents.getBody(narrowPhaseInfoBatch.colliderEntities2[i]))); } narrowPhaseInfoBatch.resetContactPoints(i); } } // Compute the narrow-phase collision detection for the testCollision() methods. // This method returns true if contacts are found. bool CollisionDetectionSystem::computeNarrowPhaseCollisionSnapshot(NarrowPhaseInput& narrowPhaseInput, CollisionCallback& callback) { RP3D_PROFILE("CollisionDetectionSystem::computeNarrowPhaseCollisionSnapshot()", mProfiler); MemoryAllocator& allocator = mMemoryManager.getHeapAllocator(); // Test the narrow-phase collision detection on the batches to be tested bool collisionFound = testNarrowPhaseCollision(narrowPhaseInput, true, false, allocator); // If collision has been found, create contacts if (collisionFound) { List potentialContactPoints(allocator); List potentialContactManifolds(allocator); Map mapPairIdToContactPairIndex(allocator); List contactPairs(allocator); List contactManifolds(allocator); List contactPoints(allocator); Map> mapBodyToContactPairs(allocator); // Process all the potential contacts after narrow-phase collision processAllPotentialContacts(narrowPhaseInput, true, potentialContactPoints, &mapPairIdToContactPairIndex, potentialContactManifolds, &contactPairs, mapBodyToContactPairs); // Reduce the number of contact points in the manifolds reducePotentialContactManifolds(&contactPairs, potentialContactManifolds, potentialContactPoints); // Create the actual contact manifolds and contact points createSnapshotContacts(contactPairs, contactManifolds, contactPoints, potentialContactManifolds, potentialContactPoints); // Report the contacts to the user reportContacts(callback, &contactPairs, &contactManifolds, &contactPoints); } return collisionFound; } // Swap the previous and current contacts lists void CollisionDetectionSystem::swapPreviousAndCurrentContacts() { if (mPreviousContactPairs == &mContactPairs1) { mPreviousContactPairs = &mContactPairs2; mPreviousContactManifolds = &mContactManifolds2; mPreviousContactPoints = &mContactPoints2; mPreviousMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex2; mCurrentContactPairs = &mContactPairs1; mCurrentContactManifolds = &mContactManifolds1; mCurrentContactPoints = &mContactPoints1; mCurrentMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex1; } else { mPreviousContactPairs = &mContactPairs1; mPreviousContactManifolds = &mContactManifolds1; mPreviousContactPoints = &mContactPoints1; mPreviousMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex1; mCurrentContactPairs = &mContactPairs2; mCurrentContactManifolds = &mContactManifolds2; mCurrentContactPoints = &mContactPoints2; mCurrentMapPairIdToContactPairIndex = &mMapPairIdToContactPairIndex2; } } // Create the actual contact manifolds and contacts points void CollisionDetectionSystem::createContacts() { RP3D_PROFILE("CollisionDetectionSystem::createContacts()", mProfiler); mCurrentContactManifolds->reserve(mCurrentContactPairs->size()); mCurrentContactPoints->reserve(mCurrentContactManifolds->size()); // For each contact pair for (uint p=0; p < (*mCurrentContactPairs).size(); p++) { ContactPair& contactPair = (*mCurrentContactPairs)[p]; assert(contactPair.potentialContactManifoldsIndices.size() > 0); contactPair.contactManifoldsIndex = mCurrentContactManifolds->size(); contactPair.nbContactManifolds = contactPair.potentialContactManifoldsIndices.size(); contactPair.contactPointsIndex = mCurrentContactPoints->size(); // For each potential contact manifold of the pair for (uint m=0; m < contactPair.potentialContactManifoldsIndices.size(); m++) { ContactManifoldInfo& potentialManifold = mPotentialContactManifolds[contactPair.potentialContactManifoldsIndices[m]]; // Start index and number of contact points for this manifold const uint contactPointsIndex = mCurrentContactPoints->size(); const int8 nbContactPoints = static_cast(potentialManifold.potentialContactPointsIndices.size()); contactPair.nbToTalContactPoints += nbContactPoints; // We create a new contact manifold ContactManifold contactManifold(contactPair.body1Entity, contactPair.body2Entity, contactPair.collider1Entity, contactPair.collider2Entity, contactPointsIndex, nbContactPoints); // Add the contact manifold mCurrentContactManifolds->add(contactManifold); assert(potentialManifold.potentialContactPointsIndices.size() > 0); // For each contact point of the manifold for (uint c=0; c < potentialManifold.potentialContactPointsIndices.size(); c++) { ContactPointInfo& potentialContactPoint = mPotentialContactPoints[potentialManifold.potentialContactPointsIndices[c]]; // Create a new contact point ContactPoint contactPoint(potentialContactPoint, mWorld->mConfig); // Add the contact point mCurrentContactPoints->add(contactPoint); } } } // Initialize the current contacts with the contacts from the previous frame (for warmstarting) initContactsWithPreviousOnes(); // Report contacts to the user if (mWorld->mEventListener != nullptr) { reportContacts(*(mWorld->mEventListener), mCurrentContactPairs, mCurrentContactManifolds, mCurrentContactPoints); } // Reset the potential contacts mPotentialContactPoints.clear(true); mPotentialContactManifolds.clear(true); } // Create the actual contact manifolds and contacts points for testCollision() methods void CollisionDetectionSystem::createSnapshotContacts(List& contactPairs, List& contactManifolds, List& contactPoints, List& potentialContactManifolds, List& potentialContactPoints) { RP3D_PROFILE("CollisionDetectionSystem::createSnapshotContacts()", mProfiler); contactManifolds.reserve(contactPairs.size()); contactPoints.reserve(contactManifolds.size()); // For each contact pair for (uint p=0; p < contactPairs.size(); p++) { ContactPair& contactPair = contactPairs[p]; assert(contactPair.potentialContactManifoldsIndices.size() > 0); contactPair.contactManifoldsIndex = contactManifolds.size(); contactPair.nbContactManifolds = contactPair.potentialContactManifoldsIndices.size(); contactPair.contactPointsIndex = contactPoints.size(); // For each potential contact manifold of the pair for (uint m=0; m < contactPair.potentialContactManifoldsIndices.size(); m++) { ContactManifoldInfo& potentialManifold = potentialContactManifolds[contactPair.potentialContactManifoldsIndices[m]]; // Start index and number of contact points for this manifold const uint contactPointsIndex = contactPoints.size(); const int8 nbContactPoints = static_cast(potentialManifold.potentialContactPointsIndices.size()); contactPair.nbToTalContactPoints += nbContactPoints; // We create a new contact manifold ContactManifold contactManifold(contactPair.body1Entity, contactPair.body2Entity, contactPair.collider1Entity, contactPair.collider2Entity, contactPointsIndex, nbContactPoints); // Add the contact manifold contactManifolds.add(contactManifold); assert(potentialManifold.potentialContactPointsIndices.size() > 0); // For each contact point of the manifold for (uint c=0; c < potentialManifold.potentialContactPointsIndices.size(); c++) { ContactPointInfo& potentialContactPoint = potentialContactPoints[potentialManifold.potentialContactPointsIndices[c]]; // Create a new contact point ContactPoint contactPoint(potentialContactPoint, mWorld->mConfig); // Add the contact point contactPoints.add(contactPoint); } } } } // Initialize the current contacts with the contacts from the previous frame (for warmstarting) void CollisionDetectionSystem::initContactsWithPreviousOnes() { // For each contact pair of the current frame for (uint i=0; i < mCurrentContactPairs->size(); i++) { ContactPair& currentContactPair = (*mCurrentContactPairs)[i]; // Find the corresponding contact pair in the previous frame (if any) auto itPrevContactPair = mPreviousMapPairIdToContactPairIndex->find(currentContactPair.pairId); // If we have found a corresponding contact pair in the previous frame if (itPrevContactPair != mPreviousMapPairIdToContactPairIndex->end()) { const uint previousContactPairIndex = itPrevContactPair->second; ContactPair& previousContactPair = (*mPreviousContactPairs)[previousContactPairIndex]; // --------------------- Contact Manifolds --------------------- // const uint contactManifoldsIndex = currentContactPair.contactManifoldsIndex; const uint nbContactManifolds = currentContactPair.nbContactManifolds; // For each current contact manifold of the contact pair for (uint m=contactManifoldsIndex; m < contactManifoldsIndex + nbContactManifolds; m++) { assert(m < mCurrentContactManifolds->size()); ContactManifold& currentContactManifold = (*mCurrentContactManifolds)[m]; assert(currentContactManifold.nbContactPoints > 0); ContactPoint& currentContactPoint = (*mCurrentContactPoints)[currentContactManifold.contactPointsIndex]; const Vector3& currentContactPointNormal = currentContactPoint.getNormal(); // Find a similar contact manifold among the contact manifolds from the previous frame (for warmstarting) const uint previousContactManifoldIndex = previousContactPair.contactManifoldsIndex; const uint previousNbContactManifolds = previousContactPair.nbContactManifolds; for (uint p=previousContactManifoldIndex; p < previousContactManifoldIndex + previousNbContactManifolds; p++) { ContactManifold& previousContactManifold = (*mPreviousContactManifolds)[p]; assert(previousContactManifold.nbContactPoints > 0); ContactPoint& previousContactPoint = (*mPreviousContactPoints)[previousContactManifold.contactPointsIndex]; // If the previous contact manifold has a similar contact normal with the current manifold if (previousContactPoint.getNormal().dot(currentContactPointNormal) >= mWorld->mConfig.cosAngleSimilarContactManifold) { // Transfer data from the previous contact manifold to the current one currentContactManifold.frictionVector1 = previousContactManifold.frictionVector1; currentContactManifold.frictionVector2 = previousContactManifold.frictionVector2; currentContactManifold.frictionImpulse1 = previousContactManifold.frictionImpulse1; currentContactManifold.frictionImpulse2 = previousContactManifold.frictionImpulse2; currentContactManifold.frictionTwistImpulse = previousContactManifold.frictionTwistImpulse; currentContactManifold.rollingResistanceImpulse = previousContactManifold.rollingResistanceImpulse; break; } } } // --------------------- Contact Points --------------------- // const uint contactPointsIndex = currentContactPair.contactPointsIndex; const uint nbTotalContactPoints = currentContactPair.nbToTalContactPoints; // For each current contact point of the contact pair for (uint c=contactPointsIndex; c < contactPointsIndex + nbTotalContactPoints; c++) { assert(c < mCurrentContactPoints->size()); ContactPoint& currentContactPoint = (*mCurrentContactPoints)[c]; // Find a similar contact point among the contact points from the previous frame (for warmstarting) const uint previousContactPointsIndex = previousContactPair.contactPointsIndex; const uint previousNbContactPoints = previousContactPair.nbToTalContactPoints; for (uint p=previousContactPointsIndex; p < previousContactPointsIndex + previousNbContactPoints; p++) { ContactPoint& previousContactPoint = (*mPreviousContactPoints)[p]; // If the previous contact point is very close to th current one const decimal distSquare = (currentContactPoint.getLocalPointOnShape1() - previousContactPoint.getLocalPointOnShape1()).lengthSquare(); if (distSquare <= mWorld->mConfig.persistentContactDistanceThreshold * mWorld->mConfig.persistentContactDistanceThreshold) { // Transfer data from the previous contact point to the current one currentContactPoint.setPenetrationImpulse(previousContactPoint.getPenetrationImpulse()); currentContactPoint.setIsRestingContact(previousContactPoint.getIsRestingContact()); break; } } } } } mPreviousContactPoints->clear(); mPreviousContactManifolds->clear(); mPreviousContactPairs->clear(); mPreviousMapPairIdToContactPairIndex->clear(); } // Remove a body from the collision detection void CollisionDetectionSystem::removeCollider(Collider* collider) { const int colliderBroadPhaseId = collider->getBroadPhaseId(); assert(colliderBroadPhaseId != -1); assert(mMapBroadPhaseIdToColliderEntity.containsKey(colliderBroadPhaseId)); // Remove all the overlapping pairs involving this collider List& overlappingPairs = mCollidersComponents.getOverlappingPairs(collider->getEntity()); while(overlappingPairs.size() > 0) { // TODO : Remove all the contact manifold of the overlapping pair from the contact manifolds list of the two bodies involved // Remove the overlapping pair mOverlappingPairs.removePair(overlappingPairs[0]); } mMapBroadPhaseIdToColliderEntity.remove(colliderBroadPhaseId); // Remove the body from the broad-phase mBroadPhaseSystem.removeCollider(collider); } // Ray casting method void CollisionDetectionSystem::raycast(RaycastCallback* raycastCallback, const Ray& ray, unsigned short raycastWithCategoryMaskBits) const { RP3D_PROFILE("CollisionDetectionSystem::raycast()", mProfiler); RaycastTest rayCastTest(raycastCallback); // Ask the broad-phase algorithm to call the testRaycastAgainstShape() // callback method for each collider hit by the ray in the broad-phase mBroadPhaseSystem.raycast(ray, rayCastTest, raycastWithCategoryMaskBits); } // Convert the potential contact into actual contacts void CollisionDetectionSystem::processPotentialContacts(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, bool updateLastFrameInfo, List& potentialContactPoints, Map* mapPairIdToContactPairIndex, List& potentialContactManifolds, List* contactPairs, Map>& mapBodyToContactPairs) { RP3D_PROFILE("CollisionDetectionSystem::processPotentialContacts()", mProfiler); // For each narrow phase info object for(uint i=0; i < narrowPhaseInfoBatch.getNbObjects(); i++) { if (updateLastFrameInfo) { narrowPhaseInfoBatch.lastFrameCollisionInfos[i]->wasColliding = narrowPhaseInfoBatch.isColliding[i]; // The previous frame collision info is now valid narrowPhaseInfoBatch.lastFrameCollisionInfos[i]->isValid = true; } // Add the potential contacts for (uint j=0; j < narrowPhaseInfoBatch.contactPoints[i].size(); j++) { assert(narrowPhaseInfoBatch.isColliding[i]); const ContactPointInfo& contactPoint = *(narrowPhaseInfoBatch.contactPoints[i][j]); // Add the contact point to the list of potential contact points const uint contactPointIndex = static_cast(potentialContactPoints.size()); potentialContactPoints.add(contactPoint); bool similarManifoldFound = false; // If there is already a contact pair for this overlapping pair const uint64 pairId = narrowPhaseInfoBatch.overlappingPairIds[i]; auto it = mapPairIdToContactPairIndex->find(pairId); ContactPair* pairContact = nullptr; if (it != mapPairIdToContactPairIndex->end()) { assert(it->first == pairId); const uint pairContactIndex = it->second; pairContact = &((*contactPairs)[pairContactIndex]); assert(pairContact->potentialContactManifoldsIndices.size() > 0); // For each contact manifold of the overlapping pair for (uint m=0; m < pairContact->potentialContactManifoldsIndices.size(); m++) { uint contactManifoldIndex = pairContact->potentialContactManifoldsIndices[m]; // Get the first contact point of the current manifold assert(potentialContactManifolds[contactManifoldIndex].potentialContactPointsIndices.size() > 0); const uint manifoldContactPointIndex = potentialContactManifolds[contactManifoldIndex].potentialContactPointsIndices[0]; const ContactPointInfo& manifoldContactPoint = potentialContactPoints[manifoldContactPointIndex]; // If we have found a corresponding manifold for the new contact point // (a manifold with a similar contact normal direction) if (manifoldContactPoint.normal.dot(contactPoint.normal) >= mWorld->mConfig.cosAngleSimilarContactManifold) { // Add the contact point to the manifold potentialContactManifolds[contactManifoldIndex].potentialContactPointsIndices.add(contactPointIndex); similarManifoldFound = true; break; } } } // If we have not found a manifold with a similar contact normal for the contact point if (!similarManifoldFound) { // Create a new contact manifold for the overlapping pair ContactManifoldInfo contactManifoldInfo(pairId, mMemoryManager.getPoolAllocator()); // Add the contact point to the manifold contactManifoldInfo.potentialContactPointsIndices.add(contactPointIndex); // If the overlapping pair contact does not exists yet if (pairContact == nullptr) { const Entity collider1Entity = narrowPhaseInfoBatch.colliderEntities1[i]; const Entity collider2Entity = narrowPhaseInfoBatch.colliderEntities2[i]; const Entity body1Entity = mCollidersComponents.getBody(collider1Entity); const Entity body2Entity = mCollidersComponents.getBody(collider2Entity); assert(!mWorld->mCollisionBodyComponents.getIsEntityDisabled(body1Entity) || !mWorld->mCollisionBodyComponents.getIsEntityDisabled(body2Entity)); const uint newContactPairIndex = contactPairs->size(); ContactPair overlappingPairContact(pairId, body1Entity, body2Entity, collider1Entity, collider2Entity, newContactPairIndex, mMemoryManager.getHeapAllocator()); contactPairs->add(overlappingPairContact); pairContact = &((*contactPairs)[newContactPairIndex]); mapPairIdToContactPairIndex->add(Pair(pairId, newContactPairIndex)); auto itbodyContactPairs = mapBodyToContactPairs.find(body1Entity); if (itbodyContactPairs != mapBodyToContactPairs.end()) { itbodyContactPairs->second.add(newContactPairIndex); } else { List contactPairs(mMemoryManager.getPoolAllocator(), 1); contactPairs.add(newContactPairIndex); mapBodyToContactPairs.add(Pair>(body1Entity, contactPairs)); } itbodyContactPairs = mapBodyToContactPairs.find(body2Entity); if (itbodyContactPairs != mapBodyToContactPairs.end()) { itbodyContactPairs->second.add(newContactPairIndex); } else { List contactPairs(mMemoryManager.getPoolAllocator(), 1); contactPairs.add(newContactPairIndex); mapBodyToContactPairs.add(Pair>(body2Entity, contactPairs)); } } assert(pairContact != nullptr); // Add the potential contact manifold uint contactManifoldIndex = static_cast(potentialContactManifolds.size()); potentialContactManifolds.add(contactManifoldInfo); // Add the contact manifold to the overlapping pair contact assert(pairContact->pairId == contactManifoldInfo.pairId); pairContact->potentialContactManifoldsIndices.add(contactManifoldIndex); } assert(pairContact->potentialContactManifoldsIndices.size() > 0); } narrowPhaseInfoBatch.resetContactPoints(i); } } // Clear the obsolete manifolds and contact points and reduce the number of contacts points of the remaining manifolds void CollisionDetectionSystem::reducePotentialContactManifolds(List* contactPairs, List& potentialContactManifolds, const List& potentialContactPoints) const { RP3D_PROFILE("CollisionDetectionSystem::reducePotentialContactManifolds()", mProfiler); // Reduce the number of potential contact manifolds in a contact pair for (uint i=0; i < contactPairs->size(); i++) { ContactPair& contactPair = (*contactPairs)[i]; assert(contactPair.potentialContactManifoldsIndices.size() > 0); // While there are too many manifolds in the contact pair while(contactPair.potentialContactManifoldsIndices.size() > mWorld->mConfig.nbMaxContactManifolds) { // Look for a manifold with the smallest contact penetration depth. decimal minDepth = DECIMAL_LARGEST; int minDepthManifoldIndex = -1; for (uint j=0; j < contactPair.potentialContactManifoldsIndices.size(); j++) { ContactManifoldInfo& manifold = potentialContactManifolds[contactPair.potentialContactManifoldsIndices[j]]; // Get the largest contact point penetration depth of the manifold const decimal depth = computePotentialManifoldLargestContactDepth(manifold, potentialContactPoints); if (depth < minDepth) { minDepth = depth; minDepthManifoldIndex = static_cast(j); } } // Remove the non optimal manifold assert(minDepthManifoldIndex >= 0); contactPair.potentialContactManifoldsIndices.removeAt(minDepthManifoldIndex); } } // Reduce the number of potential contact points in the manifolds for (uint i=0; i < contactPairs->size(); i++) { const ContactPair& pairContact = (*contactPairs)[i]; // For each potential contact manifold for (uint j=0; j < pairContact.potentialContactManifoldsIndices.size(); j++) { ContactManifoldInfo& manifold = potentialContactManifolds[pairContact.potentialContactManifoldsIndices[j]]; // If there are two many contact points in the manifold if (manifold.potentialContactPointsIndices.size() > MAX_CONTACT_POINTS_IN_MANIFOLD) { Entity collider1 = mOverlappingPairs.mColliders1[mOverlappingPairs.mMapPairIdToPairIndex[manifold.pairId]]; Transform shape1LocalToWorldTransoform = mCollidersComponents.getLocalToWorldTransform(collider1); // Reduce the number of contact points in the manifold reduceContactPoints(manifold, shape1LocalToWorldTransoform, potentialContactPoints); } assert(manifold.potentialContactPointsIndices.size() <= MAX_CONTACT_POINTS_IN_MANIFOLD); } } } // Return the largest depth of all the contact points of a potential manifold decimal CollisionDetectionSystem::computePotentialManifoldLargestContactDepth(const ContactManifoldInfo& manifold, const List& potentialContactPoints) const { decimal largestDepth = 0.0f; assert(manifold.potentialContactPointsIndices.size() > 0); for (uint i=0; i < manifold.potentialContactPointsIndices.size(); i++) { decimal depth = potentialContactPoints[manifold.potentialContactPointsIndices[i]].penetrationDepth; if (depth > largestDepth) { largestDepth = depth; } } return largestDepth; } // Reduce the number of contact points of a potential contact manifold // This is based on the technique described by Dirk Gregorius in his // "Contacts Creation" GDC presentation. This method will reduce the number of // contact points to a maximum of 4 points (but it can be less). void CollisionDetectionSystem::reduceContactPoints(ContactManifoldInfo& manifold, const Transform& shape1ToWorldTransform, const List& potentialContactPoints) const { assert(manifold.potentialContactPointsIndices.size() > MAX_CONTACT_POINTS_IN_MANIFOLD); // The following algorithm only works to reduce to a maximum of 4 contact points assert(MAX_CONTACT_POINTS_IN_MANIFOLD == 4); // List of the candidate contact points indices in the manifold. Every time that we have found a // point we want to keep, we will remove it from this list List candidatePointsIndices(manifold.potentialContactPointsIndices); int8 nbReducedPoints = 0; uint pointsToKeepIndices[MAX_CONTACT_POINTS_IN_MANIFOLD]; for (int8 i=0; i maxDotProduct) { maxDotProduct = dotProduct; elementIndexToKeep = i; nbReducedPoints = 1; } } pointsToKeepIndices[0] = candidatePointsIndices[elementIndexToKeep]; candidatePointsIndices.removeAt(elementIndexToKeep); assert(nbReducedPoints == 1); // Compute the second contact point we need to keep. // The second point we keep is the one farthest away from the first point. decimal maxDistance = decimal(0.0); elementIndexToKeep = 0; for (uint i=0; i < candidatePointsIndices.size(); i++) { const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]]; const ContactPointInfo& pointToKeep0 = potentialContactPoints[pointsToKeepIndices[0]]; assert(candidatePointsIndices[i] != pointsToKeepIndices[0]); const decimal distance = (pointToKeep0.localPoint1 - element.localPoint1).lengthSquare(); if (distance >= maxDistance) { maxDistance = distance; elementIndexToKeep = i; nbReducedPoints = 2; } } pointsToKeepIndices[1] = candidatePointsIndices[elementIndexToKeep]; candidatePointsIndices.removeAt(elementIndexToKeep); assert(nbReducedPoints == 2); // Compute the third contact point we need to keep. // The third point is the one producing the triangle with the larger area // with first and second point. // We compute the most positive or most negative triangle area (depending on winding) uint thirdPointMaxAreaIndex = 0; uint thirdPointMinAreaIndex = 0; decimal minArea = decimal(0.0); decimal maxArea = decimal(0.0); bool isPreviousAreaPositive = true; for (uint i=0; i < candidatePointsIndices.size(); i++) { const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]]; const ContactPointInfo& pointToKeep0 = potentialContactPoints[pointsToKeepIndices[0]]; const ContactPointInfo& pointToKeep1 = potentialContactPoints[pointsToKeepIndices[1]]; assert(candidatePointsIndices[i] != pointsToKeepIndices[0]); assert(candidatePointsIndices[i] != pointsToKeepIndices[1]); const Vector3 newToFirst = pointToKeep0.localPoint1 - element.localPoint1; const Vector3 newToSecond = pointToKeep1.localPoint1 - element.localPoint1; // Compute the triangle area decimal area = newToFirst.cross(newToSecond).dot(contactNormalShape1Space); if (area >= maxArea) { maxArea = area; thirdPointMaxAreaIndex = i; } if (area <= minArea) { minArea = area; thirdPointMinAreaIndex = i; } } assert(minArea <= decimal(0.0)); assert(maxArea >= decimal(0.0)); if (maxArea > (-minArea)) { isPreviousAreaPositive = true; pointsToKeepIndices[2] = candidatePointsIndices[thirdPointMaxAreaIndex]; candidatePointsIndices.removeAt(thirdPointMaxAreaIndex); } else { isPreviousAreaPositive = false; pointsToKeepIndices[2] = candidatePointsIndices[thirdPointMinAreaIndex]; candidatePointsIndices.removeAt(thirdPointMinAreaIndex); } nbReducedPoints = 3; // Compute the 4th point by choosing the triangle that add the most // triangle area to the previous triangle and has opposite sign area (opposite winding) decimal largestArea = decimal(0.0); // Largest area (positive or negative) elementIndexToKeep = 0; nbReducedPoints = 4; decimal area; // For each remaining candidate points for (uint i=0; i < candidatePointsIndices.size(); i++) { const ContactPointInfo& element = potentialContactPoints[candidatePointsIndices[i]]; assert(candidatePointsIndices[i] != pointsToKeepIndices[0]); assert(candidatePointsIndices[i] != pointsToKeepIndices[1]); assert(candidatePointsIndices[i] != pointsToKeepIndices[2]); // For each edge of the triangle made by the first three points for (uint j=0; j<3; j++) { uint edgeVertex1Index = j; uint edgeVertex2Index = j < 2 ? j + 1 : 0; const ContactPointInfo& pointToKeepEdgeV1 = potentialContactPoints[pointsToKeepIndices[edgeVertex1Index]]; const ContactPointInfo& pointToKeepEdgeV2 = potentialContactPoints[pointsToKeepIndices[edgeVertex2Index]]; const Vector3 newToFirst = pointToKeepEdgeV1.localPoint1 - element.localPoint1; const Vector3 newToSecond = pointToKeepEdgeV2.localPoint1 - element.localPoint1; // Compute the triangle area area = newToFirst.cross(newToSecond).dot(contactNormalShape1Space); // We are looking at the triangle with maximal area (positive or negative). // If the previous area is positive, we are looking at negative area now. // If the previous area is negative, we are looking at the positive area now. if (isPreviousAreaPositive && area <= largestArea) { largestArea = area; elementIndexToKeep = i; } else if (!isPreviousAreaPositive && area >= largestArea) { largestArea = area; elementIndexToKeep = i; } } } pointsToKeepIndices[3] = candidatePointsIndices[elementIndexToKeep]; candidatePointsIndices.removeAt(elementIndexToKeep); // Only keep the four selected contact points in the manifold manifold.potentialContactPointsIndices.clear(); manifold.potentialContactPointsIndices.add(pointsToKeepIndices[0]); manifold.potentialContactPointsIndices.add(pointsToKeepIndices[1]); manifold.potentialContactPointsIndices.add(pointsToKeepIndices[2]); manifold.potentialContactPointsIndices.add(pointsToKeepIndices[3]); } // Report contacts void CollisionDetectionSystem::reportContacts() { if (mWorld->mEventListener != nullptr) { reportContacts(*(mWorld->mEventListener), mCurrentContactPairs, mCurrentContactManifolds, mCurrentContactPoints); } } // Report all contacts void CollisionDetectionSystem::reportContacts(CollisionCallback& callback, List* contactPairs, List* manifolds, List* contactPoints) { RP3D_PROFILE("CollisionDetectionSystem::reportContacts()", mProfiler); // If there are contacts if (contactPairs->size() > 0) { CollisionCallback::CallbackData callbackData(contactPairs, manifolds, contactPoints, *mWorld); // Call the callback method to report the contacts callback.onContact(callbackData); } } // Return true if two bodies overlap (collide) bool CollisionDetectionSystem::testOverlap(CollisionBody* body1, CollisionBody* body2) { NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs); // Compute the broad-phase collision detection computeBroadPhase(); // Filter the overlapping pairs to get only the ones with the selected body involved List convexPairs(mMemoryManager.getPoolAllocator()); List concavePairs(mMemoryManager.getPoolAllocator()); filterOverlappingPairs(body1->getEntity(), body2->getEntity(), convexPairs, concavePairs); if (convexPairs.size() > 0 || concavePairs.size() > 0) { // Compute the middle-phase collision detection computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput); // Compute the narrow-phase collision detection return computeNarrowPhaseOverlapSnapshot(narrowPhaseInput, nullptr); } return false; } // Report all the bodies that overlap (collide) in the world void CollisionDetectionSystem::testOverlap(OverlapCallback& callback) { NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs); // Compute the broad-phase collision detection computeBroadPhase(); // Compute the middle-phase collision detection computeMiddlePhase(narrowPhaseInput); // Compute the narrow-phase collision detection and report overlapping shapes computeNarrowPhaseOverlapSnapshot(narrowPhaseInput, &callback); } // Report all the bodies that overlap (collide) with the body in parameter void CollisionDetectionSystem::testOverlap(CollisionBody* body, OverlapCallback& callback) { NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs); // Compute the broad-phase collision detection computeBroadPhase(); // Filter the overlapping pairs to get only the ones with the selected body involved List convexPairs(mMemoryManager.getPoolAllocator()); List concavePairs(mMemoryManager.getPoolAllocator()); filterOverlappingPairs(body->getEntity(), convexPairs, concavePairs); if (convexPairs.size() > 0 || concavePairs.size() > 0) { // Compute the middle-phase collision detection computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput); // Compute the narrow-phase collision detection computeNarrowPhaseOverlapSnapshot(narrowPhaseInput, &callback); } } // Test collision and report contacts between two bodies. void CollisionDetectionSystem::testCollision(CollisionBody* body1, CollisionBody* body2, CollisionCallback& callback) { NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs); // Compute the broad-phase collision detection computeBroadPhase(); // Filter the overlapping pairs to get only the ones with the selected body involved List convexPairs(mMemoryManager.getPoolAllocator()); List concavePairs(mMemoryManager.getPoolAllocator()); filterOverlappingPairs(body1->getEntity(), body2->getEntity(), convexPairs, concavePairs); if (convexPairs.size() > 0 || concavePairs.size() > 0) { // Compute the middle-phase collision detection computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput); // Compute the narrow-phase collision detection and report contacts computeNarrowPhaseCollisionSnapshot(narrowPhaseInput, callback); } } // Test collision and report all the contacts involving the body in parameter void CollisionDetectionSystem::testCollision(CollisionBody* body, CollisionCallback& callback) { NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs); // Compute the broad-phase collision detection computeBroadPhase(); // Filter the overlapping pairs to get only the ones with the selected body involved List convexPairs(mMemoryManager.getPoolAllocator()); List concavePairs(mMemoryManager.getPoolAllocator()); filterOverlappingPairs(body->getEntity(), convexPairs, concavePairs); if (convexPairs.size() > 0 || concavePairs.size() > 0) { // Compute the middle-phase collision detection computeMiddlePhaseCollisionSnapshot(convexPairs, concavePairs, narrowPhaseInput); // Compute the narrow-phase collision detection and report contacts computeNarrowPhaseCollisionSnapshot(narrowPhaseInput, callback); } } // Test collision and report contacts between each colliding bodies in the world void CollisionDetectionSystem::testCollision(CollisionCallback& callback) { NarrowPhaseInput narrowPhaseInput(mMemoryManager.getPoolAllocator(), mOverlappingPairs); // Compute the broad-phase collision detection computeBroadPhase(); // Compute the middle-phase collision detection computeMiddlePhase(narrowPhaseInput); // Compute the narrow-phase collision detection and report contacts computeNarrowPhaseCollisionSnapshot(narrowPhaseInput, callback); } // Filter the overlapping pairs to keep only the pairs where a given body is involved void CollisionDetectionSystem::filterOverlappingPairs(Entity bodyEntity, List& convexPairs, List& concavePairs) const { // For each possible collision pair of bodies for (uint i=0; i < mOverlappingPairs.getNbPairs(); i++) { if (mCollidersComponents.getBody(mOverlappingPairs.mColliders1[i]) == bodyEntity || mCollidersComponents.getBody(mOverlappingPairs.mColliders2[i]) == bodyEntity) { if (i < mOverlappingPairs.getNbConvexVsConvexPairs()) { convexPairs.add(mOverlappingPairs.mPairIds[i]); } else { concavePairs.add(mOverlappingPairs.mPairIds[i]); } } } } // Filter the overlapping pairs to keep only the pairs where two given bodies are involved void CollisionDetectionSystem::filterOverlappingPairs(Entity body1Entity, Entity body2Entity, List& convexPairs, List& concavePairs) const { // TODO : Do not go through all the overlapping pairs here but get all the involded overlapping pairs directly from the bodies // For each possible collision pair of bodies for (uint i=0; i < mOverlappingPairs.getNbPairs(); i++) { const Entity collider1Body = mCollidersComponents.getBody(mOverlappingPairs.mColliders1[i]); const Entity collider2Body = mCollidersComponents.getBody(mOverlappingPairs.mColliders2[i]); if ((collider1Body == body1Entity && collider2Body == body2Entity) || (collider1Body == body2Entity && collider2Body == body1Entity)) { if (i < mOverlappingPairs.getNbConvexVsConvexPairs()) { convexPairs.add(mOverlappingPairs.mPairIds[i]); } else { concavePairs.add(mOverlappingPairs.mPairIds[i]); } } } } // Return the world event listener EventListener* CollisionDetectionSystem::getWorldEventListener() { return mWorld->mEventListener; } // Return the world-space AABB of a given collider const AABB CollisionDetectionSystem::getWorldAABB(const Collider* collider) const { assert(collider->getBroadPhaseId() > -1); return mBroadPhaseSystem.getFatAABB(collider->getBroadPhaseId()); }