From e8ed10314a035e3048502282ac8d9d5c25564b8d Mon Sep 17 00:00:00 2001 From: Daniel Chappuis Date: Wed, 3 Oct 2018 22:10:05 +0200 Subject: [PATCH] Refactor narrow-phase algorithms to iterate over batches of narrow-phase infos --- CMakeLists.txt | 4 +- src/collision/CollisionDetection.cpp | 227 ++--- src/collision/CollisionDetection.h | 10 +- src/collision/ContactManifoldSet.cpp | 10 +- src/collision/ContactManifoldSet.h | 7 +- src/collision/MiddlePhaseTriangleCallback.cpp | 14 +- src/collision/MiddlePhaseTriangleCallback.h | 10 +- ...PhaseInfo.cpp => NarrowPhaseInfoBatch.cpp} | 87 +- ...rrowPhaseInfo.h => NarrowPhaseInfoBatch.h} | 80 +- .../narrowphase/CapsuleVsCapsuleAlgorithm.cpp | 324 +++---- .../narrowphase/CapsuleVsCapsuleAlgorithm.h | 6 +- .../CapsuleVsConvexPolyhedronAlgorithm.cpp | 184 ++-- .../CapsuleVsConvexPolyhedronAlgorithm.h | 4 +- ...xPolyhedronVsConvexPolyhedronAlgorithm.cpp | 19 +- ...vexPolyhedronVsConvexPolyhedronAlgorithm.h | 4 +- .../narrowphase/GJK/GJKAlgorithm.cpp | 275 +++--- src/collision/narrowphase/GJK/GJKAlgorithm.h | 7 +- .../narrowphase/NarrowPhaseAlgorithm.h | 6 +- .../narrowphase/SAT/SATAlgorithm.cpp | 833 +++++++++--------- src/collision/narrowphase/SAT/SATAlgorithm.h | 14 +- .../narrowphase/SphereVsCapsuleAlgorithm.cpp | 151 ++-- .../narrowphase/SphereVsCapsuleAlgorithm.h | 4 +- .../SphereVsConvexPolyhedronAlgorithm.cpp | 71 +- .../SphereVsConvexPolyhedronAlgorithm.h | 4 +- .../narrowphase/SphereVsSphereAlgorithm.cpp | 96 +- .../narrowphase/SphereVsSphereAlgorithm.h | 4 +- src/constraint/ContactPoint.h | 2 - src/containers/List.h | 2 +- src/engine/OverlappingPair.cpp | 1 - src/engine/OverlappingPair.h | 8 +- 30 files changed, 1279 insertions(+), 1189 deletions(-) rename src/collision/{NarrowPhaseInfo.cpp => NarrowPhaseInfoBatch.cpp} (51%) rename src/collision/{NarrowPhaseInfo.h => NarrowPhaseInfoBatch.h} (52%) diff --git a/CMakeLists.txt b/CMakeLists.txt index decf32b2..dceef275 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -115,7 +115,7 @@ SET (REACTPHYSICS3D_HEADERS "src/collision/PolyhedronMesh.h" "src/collision/HalfEdgeStructure.h" "src/collision/CollisionDetection.h" - "src/collision/NarrowPhaseInfo.h" + "src/collision/NarrowPhaseInfoBatch.h" "src/collision/ContactManifold.h" "src/collision/ContactManifoldSet.h" "src/collision/MiddlePhaseTriangleCallback.h" @@ -198,7 +198,7 @@ SET (REACTPHYSICS3D_SOURCES "src/collision/PolyhedronMesh.cpp" "src/collision/HalfEdgeStructure.cpp" "src/collision/CollisionDetection.cpp" - "src/collision/NarrowPhaseInfo.cpp" + "src/collision/NarrowPhaseInfoBatch.cpp" "src/collision/ContactManifold.cpp" "src/collision/ContactManifoldSet.cpp" "src/collision/MiddlePhaseTriangleCallback.cpp" diff --git a/src/collision/CollisionDetection.cpp b/src/collision/CollisionDetection.cpp index 9ef9078e..00f38793 100644 --- a/src/collision/CollisionDetection.cpp +++ b/src/collision/CollisionDetection.cpp @@ -35,7 +35,7 @@ #include "collision/CollisionCallback.h" #include "collision/MiddlePhaseTriangleCallback.h" #include "collision/OverlapCallback.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" #include "collision/ContactManifold.h" #include "utils/Profiler.h" #include "engine/EventListener.h" @@ -49,7 +49,7 @@ using namespace std; // Constructor CollisionDetection::CollisionDetection(CollisionWorld* world, MemoryManager& memoryManager) - : mMemoryManager(memoryManager), mWorld(world), mNarrowPhaseInfos(mMemoryManager.getPoolAllocator()), + : mMemoryManager(memoryManager), mWorld(world), mNarrowPhaseInfoBatch(mMemoryManager.getPoolAllocator()), mOverlappingPairs(mMemoryManager.getPoolAllocator()), mBroadPhaseAlgorithm(*this), mNoCollisionPairs(mMemoryManager.getPoolAllocator()), mIsCollisionShapesAdded(false) { @@ -159,17 +159,15 @@ void CollisionDetection::computeMiddlePhase() { // No middle-phase is necessary, simply create a narrow phase info // for the narrow-phase collision detection - NarrowPhaseInfo* narrowPhaseInfo = new (mMemoryManager.allocate(MemoryManager::AllocationType::Frame, sizeof(NarrowPhaseInfo))) - NarrowPhaseInfo(pair, shape1->getCollisionShape(), - shape2->getCollisionShape(), shape1->getLocalToWorldTransform(), - shape2->getLocalToWorldTransform(), mMemoryManager.getSingleFrameAllocator()); - mNarrowPhaseInfos.add(narrowPhaseInfo); + mNarrowPhaseInfoBatch.addNarrowPhaseInfo(pair, shape1->getCollisionShape(), shape2->getCollisionShape(), + shape1->getLocalToWorldTransform(), shape2->getLocalToWorldTransform(), + mMemoryManager.getSingleFrameAllocator()); } // Concave vs Convex algorithm else if ((!isShape1Convex && isShape2Convex) || (!isShape2Convex && isShape1Convex)) { - computeConvexVsConcaveMiddlePhase(pair, mMemoryManager.getSingleFrameAllocator(), mNarrowPhaseInfos); + computeConvexVsConcaveMiddlePhase(pair, mMemoryManager.getSingleFrameAllocator(), mNarrowPhaseInfoBatch); } // Concave vs Concave shape else { @@ -185,7 +183,7 @@ void CollisionDetection::computeMiddlePhase() { // Compute the concave vs convex middle-phase algorithm for a given pair of bodies void CollisionDetection::computeConvexVsConcaveMiddlePhase(OverlappingPair* pair, MemoryAllocator& allocator, - List& narrowPhaseInfos) { + NarrowPhaseInfoBatch& narrowPhaseInfoBatch) { ProxyShape* shape1 = pair->getShape1(); ProxyShape* shape2 = pair->getShape2(); @@ -211,7 +209,7 @@ void CollisionDetection::computeConvexVsConcaveMiddlePhase(OverlappingPair* pair // Set the parameters of the callback object MiddlePhaseTriangleCallback middlePhaseCallback(pair, concaveProxyShape, convexProxyShape, - concaveShape, narrowPhaseInfos, allocator); + concaveShape, narrowPhaseInfoBatch, allocator); #ifdef IS_PROFILING_ACTIVE @@ -235,33 +233,32 @@ void CollisionDetection::computeNarrowPhase() { RP3D_PROFILE("CollisionDetection::computeNarrowPhase()", mProfiler); - List collidingNarrowPhaseInfos(mMemoryManager.getSingleFrameAllocator()); + List collidingBatchIndices(mMemoryManager.getSingleFrameAllocator()); // For each narrow phase info to process - for(uint i=0; i < mNarrowPhaseInfos.size(); i++) { + for(uint batchIndex=0; batchIndex < mNarrowPhaseInfoBatch.getNbObjects(); batchIndex++) { - NarrowPhaseInfo* narrowPhaseInfo = mNarrowPhaseInfos[i]; - - assert(narrowPhaseInfo->contactPoints.size() == 0); + assert(mNarrowPhaseInfoBatch.contactPoints[batchIndex].size() == 0); // Select the narrow phase algorithm to use according to the two collision shapes - const CollisionShapeType shape1Type = narrowPhaseInfo->collisionShape1->getType(); - const CollisionShapeType shape2Type = narrowPhaseInfo->collisionShape2->getType(); + const CollisionShapeType shape1Type = mNarrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType(); + const CollisionShapeType shape2Type = mNarrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType(); NarrowPhaseAlgorithm* narrowPhaseAlgorithm = selectNarrowPhaseAlgorithm(shape1Type, shape2Type); // If there is no collision algorithm between those two kinds of shapes, skip it if (narrowPhaseAlgorithm != nullptr) { - LastFrameCollisionInfo* lastCollisionFrameInfo = narrowPhaseInfo->getLastFrameCollisionInfo(); + LastFrameCollisionInfo* lastCollisionFrameInfo = mNarrowPhaseInfoBatch.getLastFrameCollisionInfo(batchIndex); // Use the narrow-phase collision detection algorithm to check // if there really is a collision. If a collision occurs, the // notifyContact() callback method will be called. - if (narrowPhaseAlgorithm->testCollision(narrowPhaseInfo, true, mMemoryManager.getSingleFrameAllocator())) { + narrowPhaseAlgorithm->testCollision(mNarrowPhaseInfoBatch, batchIndex, 1, true, mMemoryManager.getSingleFrameAllocator()); + if (mNarrowPhaseInfoBatch.isColliding[batchIndex]) { lastCollisionFrameInfo->wasColliding = true; - collidingNarrowPhaseInfos.add(narrowPhaseInfo); + collidingBatchIndices.add(batchIndex); } else { lastCollisionFrameInfo->wasColliding = false; @@ -273,7 +270,7 @@ void CollisionDetection::computeNarrowPhase() { } // Convert the potential contact into actual contacts - processAllPotentialContacts(collidingNarrowPhaseInfos, mOverlappingPairs); + processAllPotentialContacts(mNarrowPhaseInfoBatch, collidingBatchIndices, mOverlappingPairs); // Add all the contact manifolds (between colliding bodies) to the bodies addAllContactManifoldsToBodies(); @@ -281,20 +278,8 @@ void CollisionDetection::computeNarrowPhase() { // Report contacts to the user reportAllContacts(); - // Destroy the narrow phase infos - for(uint i=0; i < mNarrowPhaseInfos.size(); i++) { - - NarrowPhaseInfo* narrowPhaseInfo = mNarrowPhaseInfos[i]; - - // Call the destructor - narrowPhaseInfo->~NarrowPhaseInfo(); - - // Release the allocated memory for the narrow phase info - mMemoryManager.release(MemoryManager::AllocationType::Frame, narrowPhaseInfo, sizeof(NarrowPhaseInfo)); - } - // Clear the list of narrow-phase infos - mNarrowPhaseInfos.clear(); + mNarrowPhaseInfoBatch.clear(); } // Allow the broadphase to notify the collision detection about an overlapping pair. @@ -423,24 +408,23 @@ void CollisionDetection::addContactManifoldToBody(OverlappingPair* pair) { } /// Convert the potential contact into actual contacts -void CollisionDetection::processAllPotentialContacts(const List& collidingNarrowPhaseInfos, +void CollisionDetection::processAllPotentialContacts(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, const List& collidingBatchIndex, const OverlappingPairMap& overlappingPairs) { RP3D_PROFILE("CollisionDetection::processAllPotentialContacts()", mProfiler); // For each narrow phase info object - for(uint i=0; i < collidingNarrowPhaseInfos.size(); i++) { + for(uint i=0; i < collidingBatchIndex.size(); i++) { - NarrowPhaseInfo* narrowPhaseInfo = collidingNarrowPhaseInfos[i]; + uint batchIndex = collidingBatchIndex[i]; - assert(narrowPhaseInfo != nullptr); - assert(narrowPhaseInfo->contactPoints.size() > 0); + assert(narrowPhaseInfoBatch.contactPoints[batchIndex].size() > 0); // Transfer the contact points from the narrow phase info to the overlapping pair - narrowPhaseInfo->overlappingPair->addPotentialContactPoints(narrowPhaseInfo); + narrowPhaseInfoBatch.overlappingPairs[batchIndex]->addPotentialContactPoints(narrowPhaseInfoBatch, batchIndex); // Remove the contacts points from the narrow phase info object. - narrowPhaseInfo->resetContactPoints(); + narrowPhaseInfoBatch.resetContactPoints(batchIndex); } // For each overlapping pairs in contact during the narrow-phase @@ -478,7 +462,7 @@ void CollisionDetection::reportAllContacts() { } // Compute the middle-phase collision detection between two proxy shapes -void CollisionDetection::computeMiddlePhaseForProxyShapes(OverlappingPair* pair, List& outNarrowPhaseInfos) { +void CollisionDetection::computeMiddlePhaseForProxyShapes(OverlappingPair* pair, NarrowPhaseInfoBatch& outNarrowPhaseInfoBatch) { ProxyShape* shape1 = pair->getShape1(); ProxyShape* shape2 = pair->getShape2(); @@ -495,11 +479,9 @@ void CollisionDetection::computeMiddlePhaseForProxyShapes(OverlappingPair* pair, // No middle-phase is necessary, simply create a narrow phase info // for the narrow-phase collision detection - NarrowPhaseInfo* narrowPhaseInfo = new (mMemoryManager.allocate(MemoryManager::AllocationType::Pool, - sizeof(NarrowPhaseInfo))) NarrowPhaseInfo(pair, shape1->getCollisionShape(), - shape2->getCollisionShape(), shape1->getLocalToWorldTransform(), - shape2->getLocalToWorldTransform(), mMemoryManager.getPoolAllocator()); - outNarrowPhaseInfos.add(narrowPhaseInfo); + outNarrowPhaseInfoBatch.addNarrowPhaseInfo(pair, shape1->getCollisionShape(), shape2->getCollisionShape(), + shape1->getLocalToWorldTransform(), shape2->getLocalToWorldTransform(), + mMemoryManager.getPoolAllocator()); } // Concave vs Convex algorithm @@ -507,7 +489,7 @@ void CollisionDetection::computeMiddlePhaseForProxyShapes(OverlappingPair* pair, // Run the middle-phase collision detection algorithm to find the triangles of the concave // shape we need to use during the narrow-phase collision detection - computeConvexVsConcaveMiddlePhase(pair, mMemoryManager.getPoolAllocator(), outNarrowPhaseInfos); + computeConvexVsConcaveMiddlePhase(pair, mMemoryManager.getPoolAllocator(), outNarrowPhaseInfoBatch); } pair->clearObsoleteLastFrameCollisionInfos(); @@ -556,7 +538,7 @@ void CollisionDetection::testAABBOverlap(const AABB& aabb, OverlapCallback* over // Return true if two bodies overlap bool CollisionDetection::testOverlap(CollisionBody* body1, CollisionBody* body2) { - List narrowPhaseInfos(mMemoryManager.getPoolAllocator()); + NarrowPhaseInfoBatch narrowPhaseInfoBatch(mMemoryManager.getPoolAllocator()); // For each proxy shape proxy shape of the first body ProxyShape* body1ProxyShape = body1->getProxyShapesList(); @@ -577,23 +559,19 @@ bool CollisionDetection::testOverlap(CollisionBody* body1, CollisionBody* body2) OverlappingPair pair(body1ProxyShape, body2ProxyShape, mMemoryManager.getPoolAllocator(), mMemoryManager.getPoolAllocator(), mWorld->mConfig); - narrowPhaseInfos.clear(); - // Compute the middle-phase collision detection between the two shapes - computeMiddlePhaseForProxyShapes(&pair, narrowPhaseInfos); + computeMiddlePhaseForProxyShapes(&pair, narrowPhaseInfoBatch); bool isColliding = false; // For each narrow-phase info object - for(uint i=0; i < narrowPhaseInfos.size(); i++) { - - NarrowPhaseInfo* narrowPhaseInfo = narrowPhaseInfos[i]; + for(uint batchIndex=0; batchIndex < narrowPhaseInfoBatch.getNbObjects(); batchIndex++) { // If we have not found a collision yet if (!isColliding) { - const CollisionShapeType shape1Type = narrowPhaseInfo->collisionShape1->getType(); - const CollisionShapeType shape2Type = narrowPhaseInfo->collisionShape2->getType(); + const CollisionShapeType shape1Type = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType(); + const CollisionShapeType shape2Type = narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType(); // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm* narrowPhaseAlgorithm = selectNarrowPhaseAlgorithm(shape1Type, shape2Type); @@ -604,17 +582,14 @@ bool CollisionDetection::testOverlap(CollisionBody* body1, CollisionBody* body2) // Use the narrow-phase collision detection algorithm to check // if there really is a collision. If a collision occurs, the // notifyContact() callback method will be called. - isColliding |= narrowPhaseAlgorithm->testCollision(narrowPhaseInfo, false, mMemoryManager.getPoolAllocator()); + narrowPhaseAlgorithm->testCollision(narrowPhaseInfoBatch, batchIndex, 1, false, mMemoryManager.getPoolAllocator()); + isColliding |= narrowPhaseInfoBatch.isColliding[batchIndex]; } } - - // Call the destructor - narrowPhaseInfo->~NarrowPhaseInfo(); - - // Release the allocated memory - mMemoryManager.release(MemoryManager::AllocationType::Pool, narrowPhaseInfo, sizeof(NarrowPhaseInfo)); } + narrowPhaseInfoBatch.clear(); + // Return if we have found a narrow-phase collision if (isColliding) return true; } @@ -638,7 +613,7 @@ void CollisionDetection::testOverlap(CollisionBody* body, OverlapCallback* overl assert(overlapCallback != nullptr); Set reportedBodies(mMemoryManager.getPoolAllocator()); - List narrowPhaseInfos(mMemoryManager.getPoolAllocator()); + NarrowPhaseInfoBatch narrowPhaseInfoBatch(mMemoryManager.getPoolAllocator()); // For each proxy shape proxy shape of the body ProxyShape* bodyProxyShape = body->getProxyShapesList(); @@ -675,23 +650,19 @@ void CollisionDetection::testOverlap(CollisionBody* body, OverlapCallback* overl OverlappingPair pair(bodyProxyShape, proxyShape, mMemoryManager.getPoolAllocator(), mMemoryManager.getPoolAllocator(), mWorld->mConfig); - narrowPhaseInfos.clear(); - // Compute the middle-phase collision detection between the two shapes - computeMiddlePhaseForProxyShapes(&pair, narrowPhaseInfos); + computeMiddlePhaseForProxyShapes(&pair, narrowPhaseInfoBatch); bool isColliding = false; // For each narrow-phase info object - for (uint i=0; icollisionShape1->getType(); - const CollisionShapeType shape2Type = narrowPhaseInfo->collisionShape2->getType(); + const CollisionShapeType shape1Type = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType(); + const CollisionShapeType shape2Type = narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType(); // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm* narrowPhaseAlgorithm = selectNarrowPhaseAlgorithm(shape1Type, shape2Type); @@ -702,17 +673,14 @@ void CollisionDetection::testOverlap(CollisionBody* body, OverlapCallback* overl // Use the narrow-phase collision detection algorithm to check // if there really is a collision. If a collision occurs, the // notifyContact() callback method will be called. - isColliding |= narrowPhaseAlgorithm->testCollision(narrowPhaseInfo, false, mMemoryManager.getPoolAllocator()); + narrowPhaseAlgorithm->testCollision(narrowPhaseInfoBatch, batchIndex, 1, false, mMemoryManager.getPoolAllocator()); + isColliding |= narrowPhaseInfoBatch.isColliding[batchIndex]; } } - - // Call the destructor - narrowPhaseInfo->~NarrowPhaseInfo(); - - // Release the allocated memory - mMemoryManager.release(MemoryManager::AllocationType::Pool, narrowPhaseInfo, sizeof(NarrowPhaseInfo)); } + narrowPhaseInfoBatch.clear(); + // Return if we have found a narrow-phase collision if (isColliding) { @@ -742,8 +710,8 @@ void CollisionDetection::testCollision(CollisionBody* body1, CollisionBody* body assert(collisionCallback != nullptr); - List collidingNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); - List allNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); + List collidingNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); + NarrowPhaseInfoBatch narrowPhaseInfoBatch(mMemoryManager.getPoolAllocator()); OverlappingPairMap overlappingPairs(mMemoryManager.getPoolAllocator()); // For each proxy shape proxy shape of the first body @@ -784,7 +752,7 @@ void CollisionDetection::testCollision(CollisionBody* body1, CollisionBody* body } // Compute the middle-phase collision detection between the two shapes - computeMiddlePhaseForProxyShapes(pair, allNarrowPhaseInfos); + computeMiddlePhaseForProxyShapes(pair, narrowPhaseInfoBatch); } // Go to the next proxy shape @@ -796,12 +764,10 @@ void CollisionDetection::testCollision(CollisionBody* body1, CollisionBody* body } // For each narrow-phase info object - for (uint i=0; i < allNarrowPhaseInfos.size(); i++) { + for (uint batchIndex=0; batchIndex < narrowPhaseInfoBatch.getNbObjects(); batchIndex++) { - NarrowPhaseInfo* narrowPhaseInfo = allNarrowPhaseInfos[i]; - - const CollisionShapeType shape1Type = narrowPhaseInfo->collisionShape1->getType(); - const CollisionShapeType shape2Type = narrowPhaseInfo->collisionShape2->getType(); + const CollisionShapeType shape1Type = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType(); + const CollisionShapeType shape2Type = narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType(); // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm* narrowPhaseAlgorithm = selectNarrowPhaseAlgorithm(shape1Type, shape2Type); @@ -812,15 +778,16 @@ void CollisionDetection::testCollision(CollisionBody* body1, CollisionBody* body // Use the narrow-phase collision detection algorithm to check // if there really is a collision. If a collision occurs, the // notifyContact() callback method will be called. - if (narrowPhaseAlgorithm->testCollision(narrowPhaseInfo, true, mMemoryManager.getPoolAllocator())) { + narrowPhaseAlgorithm->testCollision(narrowPhaseInfoBatch, batchIndex, 1, true, mMemoryManager.getPoolAllocator()); + if (narrowPhaseInfoBatch.isColliding[batchIndex]) { - collidingNarrowPhaseInfos.add(narrowPhaseInfo); + collidingNarrowPhaseInfos.add(batchIndex); } } } // Process the potential contacts - processAllPotentialContacts(collidingNarrowPhaseInfos, overlappingPairs); + processAllPotentialContacts(narrowPhaseInfoBatch, collidingNarrowPhaseInfos, overlappingPairs); // For each overlapping pair for (auto it = overlappingPairs.begin(); it != overlappingPairs.end(); ++it) { @@ -838,18 +805,6 @@ void CollisionDetection::testCollision(CollisionBody* body1, CollisionBody* body pair->~OverlappingPair(); mMemoryManager.release(MemoryManager::AllocationType::Pool, pair, sizeof(OverlappingPair)); } - - // Destroy the narrow phase infos - for (uint i=0; i < allNarrowPhaseInfos.size(); i++) { - - NarrowPhaseInfo* narrowPhaseInfo = allNarrowPhaseInfos[i]; - - // Call the destructor - narrowPhaseInfo->~NarrowPhaseInfo(); - - // Release the allocated memory for the narrow phase info - mMemoryManager.release(MemoryManager::AllocationType::Pool, narrowPhaseInfo, sizeof(NarrowPhaseInfo)); - } } // Test and report collisions between a body and all the others bodies of the world @@ -857,8 +812,8 @@ void CollisionDetection::testCollision(CollisionBody* body, CollisionCallback* c assert(callback != nullptr); - List collidingNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); - List allNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); + List collidingBatchIndices(mMemoryManager.getPoolAllocator()); + NarrowPhaseInfoBatch narrowPhaseInfoBatch(mMemoryManager.getPoolAllocator()); OverlappingPairMap overlappingPairs(mMemoryManager.getPoolAllocator()); // For each proxy shape proxy shape of the body @@ -913,7 +868,7 @@ void CollisionDetection::testCollision(CollisionBody* body, CollisionCallback* c } // Compute the middle-phase collision detection between the two shapes - computeMiddlePhaseForProxyShapes(pair, allNarrowPhaseInfos); + computeMiddlePhaseForProxyShapes(pair, narrowPhaseInfoBatch); } } @@ -927,12 +882,10 @@ void CollisionDetection::testCollision(CollisionBody* body, CollisionCallback* c } // For each narrow-phase info object - for (auto it = allNarrowPhaseInfos.begin(); it != allNarrowPhaseInfos.end(); ++it) { + for (uint batchIndex = 0; batchIndex < narrowPhaseInfoBatch.getNbObjects(); batchIndex++) { - NarrowPhaseInfo* narrowPhaseInfo = *it; - - const CollisionShapeType shape1Type = narrowPhaseInfo->collisionShape1->getType(); - const CollisionShapeType shape2Type = narrowPhaseInfo->collisionShape2->getType(); + const CollisionShapeType shape1Type = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType(); + const CollisionShapeType shape2Type = narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType(); // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm* narrowPhaseAlgorithm = selectNarrowPhaseAlgorithm(shape1Type, shape2Type); @@ -943,15 +896,16 @@ void CollisionDetection::testCollision(CollisionBody* body, CollisionCallback* c // Use the narrow-phase collision detection algorithm to check // if there really is a collision. If a collision occurs, the // notifyContact() callback method will be called. - if (narrowPhaseAlgorithm->testCollision(narrowPhaseInfo, true, mMemoryManager.getPoolAllocator())) { + narrowPhaseAlgorithm->testCollision(narrowPhaseInfoBatch, batchIndex, 1, true, mMemoryManager.getPoolAllocator()); + if (narrowPhaseInfoBatch.isColliding[batchIndex]) { - collidingNarrowPhaseInfos.add(narrowPhaseInfo); + collidingBatchIndices.add(batchIndex); } } } // Process the potential contacts - processAllPotentialContacts(collidingNarrowPhaseInfos, overlappingPairs); + processAllPotentialContacts(narrowPhaseInfoBatch, collidingBatchIndices, overlappingPairs); // For each overlapping pair for (auto it = overlappingPairs.begin(); it != overlappingPairs.end(); ++it) { @@ -969,18 +923,6 @@ void CollisionDetection::testCollision(CollisionBody* body, CollisionCallback* c pair->~OverlappingPair(); mMemoryManager.release(MemoryManager::AllocationType::Pool, pair, sizeof(OverlappingPair)); } - - // Destroy the narrow phase infos - for (uint i=0; i < allNarrowPhaseInfos.size(); i++) { - - NarrowPhaseInfo* narrowPhaseInfo = allNarrowPhaseInfos[i]; - - // Call the destructor - narrowPhaseInfo->~NarrowPhaseInfo(); - - // Release the allocated memory for the narrow phase info - mMemoryManager.release(MemoryManager::AllocationType::Pool, narrowPhaseInfo, sizeof(NarrowPhaseInfo)); - } } // Test and report collisions between all shapes of the world @@ -991,8 +933,8 @@ void CollisionDetection::testCollision(CollisionCallback* callback) { // Compute the broad-phase collision detection computeBroadPhase(); - List collidingNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); - List allNarrowPhaseInfos(mMemoryManager.getPoolAllocator()); + List collidingBatchIndices(mMemoryManager.getPoolAllocator()); + NarrowPhaseInfoBatch narrowPhaseInfoBatch(mMemoryManager.getPoolAllocator()); OverlappingPairMap overlappingPairs(mMemoryManager.getPoolAllocator()); // For each possible collision pair of bodies @@ -1032,17 +974,15 @@ void CollisionDetection::testCollision(CollisionCallback* callback) { mBroadPhaseAlgorithm.testOverlappingShapes(shape1, shape2)) { // Compute the middle-phase collision detection between the two shapes - computeMiddlePhaseForProxyShapes(pair, allNarrowPhaseInfos); + computeMiddlePhaseForProxyShapes(pair, narrowPhaseInfoBatch); } } // For each narrow-phase info object - for (uint i=0; i < allNarrowPhaseInfos.size(); i++) { + for (uint batchIndex=0; batchIndex < narrowPhaseInfoBatch.getNbObjects(); batchIndex++) { - NarrowPhaseInfo* narrowPhaseInfo = allNarrowPhaseInfos[i]; - - const CollisionShapeType shape1Type = narrowPhaseInfo->collisionShape1->getType(); - const CollisionShapeType shape2Type = narrowPhaseInfo->collisionShape2->getType(); + const CollisionShapeType shape1Type = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType(); + const CollisionShapeType shape2Type = narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType(); // Select the narrow phase algorithm to use according to the two collision shapes NarrowPhaseAlgorithm* narrowPhaseAlgorithm = selectNarrowPhaseAlgorithm(shape1Type, shape2Type); @@ -1053,15 +993,16 @@ void CollisionDetection::testCollision(CollisionCallback* callback) { // Use the narrow-phase collision detection algorithm to check // if there really is a collision. If a collision occurs, the // notifyContact() callback method will be called. - if (narrowPhaseAlgorithm->testCollision(narrowPhaseInfo, true, mMemoryManager.getPoolAllocator())) { + narrowPhaseAlgorithm->testCollision(narrowPhaseInfoBatch, batchIndex, 1, true, mMemoryManager.getPoolAllocator()); + if (narrowPhaseInfoBatch.isColliding[batchIndex]) { - collidingNarrowPhaseInfos.add(narrowPhaseInfo); + collidingBatchIndices.add(batchIndex); } } } // Process the potential contacts - processAllPotentialContacts(collidingNarrowPhaseInfos, overlappingPairs); + processAllPotentialContacts(narrowPhaseInfoBatch, collidingBatchIndices, overlappingPairs); // For each overlapping pair for (auto it = overlappingPairs.begin(); it != overlappingPairs.end(); ++it) { @@ -1079,18 +1020,6 @@ void CollisionDetection::testCollision(CollisionCallback* callback) { pair->~OverlappingPair(); mMemoryManager.release(MemoryManager::AllocationType::Pool, pair, sizeof(OverlappingPair)); } - - // Destroy the narrow phase infos - for (uint i=0; i < allNarrowPhaseInfos.size(); i++) { - - NarrowPhaseInfo* narrowPhaseInfo = allNarrowPhaseInfos[i]; - - // Call the destructor - narrowPhaseInfo->~NarrowPhaseInfo(); - - // Release the allocated memory for the narrow phase info - mMemoryManager.release(MemoryManager::AllocationType::Pool, narrowPhaseInfo, sizeof(NarrowPhaseInfo)); - } } // Fill-in the collision detection matrix diff --git a/src/collision/CollisionDetection.h b/src/collision/CollisionDetection.h index bd49f10f..31f6de81 100644 --- a/src/collision/CollisionDetection.h +++ b/src/collision/CollisionDetection.h @@ -32,6 +32,7 @@ #include "collision/shapes/CollisionShape.h" #include "engine/OverlappingPair.h" #include "collision/narrowphase/DefaultCollisionDispatch.h" +#include "collision/NarrowPhaseInfoBatch.h" #include "containers/Map.h" #include "containers/Set.h" @@ -79,7 +80,7 @@ class CollisionDetection { CollisionWorld* mWorld; /// List of narrow phase infos - List mNarrowPhaseInfos; + NarrowPhaseInfoBatch mNarrowPhaseInfoBatch; /// Broad-phase overlapping pairs OverlappingPairMap mOverlappingPairs; @@ -127,13 +128,14 @@ class CollisionDetection { /// Compute the concave vs convex middle-phase algorithm for a given pair of bodies void computeConvexVsConcaveMiddlePhase(OverlappingPair* pair, MemoryAllocator& allocator, - List& narrowPhaseInfos); + NarrowPhaseInfoBatch& narrowPhaseInfoBatch); /// Compute the middle-phase collision detection between two proxy shapes - void computeMiddlePhaseForProxyShapes(OverlappingPair* pair, List& outNarrowPhaseInfos); + void computeMiddlePhaseForProxyShapes(OverlappingPair* pair, NarrowPhaseInfoBatch& outNarrowPhaseInfoBatch); /// Convert the potential contact into actual contacts - void processAllPotentialContacts(const List& collidingNarrowPhaseInfos, + void processAllPotentialContacts(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, + const List& collidingBatchIndex, const OverlappingPairMap& overlappingPairs); /// Report contacts for all the colliding overlapping pairs diff --git a/src/collision/ContactManifoldSet.cpp b/src/collision/ContactManifoldSet.cpp index 8c3b43fa..7ced9552 100644 --- a/src/collision/ContactManifoldSet.cpp +++ b/src/collision/ContactManifoldSet.cpp @@ -25,7 +25,7 @@ // Libraries #include "ContactManifoldSet.h" -#include "NarrowPhaseInfo.h" +#include "NarrowPhaseInfoBatch.h" #include "constraint/ContactPoint.h" #include "ProxyShape.h" #include "collision/ContactManifold.h" @@ -49,14 +49,14 @@ ContactManifoldSet::~ContactManifoldSet() { clear(); } -void ContactManifoldSet::addContactPoints(NarrowPhaseInfo* narrowPhaseInfo) { +void ContactManifoldSet::addContactPoints(const NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex) { - assert(narrowPhaseInfo->contactPoints.size() > 0); + assert(narrowPhaseInfoBatch.contactPoints[batchIndex].size() > 0); // For each potential contact point to add - for (uint i=0; i < narrowPhaseInfo->contactPoints.size(); i++) { + for (uint i=0; i < narrowPhaseInfoBatch.contactPoints[batchIndex].size(); i++) { - ContactPointInfo* contactPoint = narrowPhaseInfo->contactPoints[i]; + ContactPointInfo* contactPoint = narrowPhaseInfoBatch.contactPoints[batchIndex][i]; // Look if the contact point correspond to an existing potential manifold // (if the contact point normal is similar to the normal of an existing manifold) diff --git a/src/collision/ContactManifoldSet.h b/src/collision/ContactManifoldSet.h index ac22715c..8ba5017d 100644 --- a/src/collision/ContactManifoldSet.h +++ b/src/collision/ContactManifoldSet.h @@ -26,6 +26,9 @@ #ifndef REACTPHYSICS3D_CONTACT_MANIFOLD_SET_H #define REACTPHYSICS3D_CONTACT_MANIFOLD_SET_H +// Libraries +#include "configuration.h" + namespace reactphysics3d { // Declarations @@ -34,7 +37,7 @@ class ContactManifoldInfo; class ProxyShape; class MemoryAllocator; struct WorldSettings; -struct NarrowPhaseInfo; +struct NarrowPhaseInfoBatch; struct Vector3; class CollisionShape; class Transform; @@ -110,7 +113,7 @@ class ContactManifoldSet { ~ContactManifoldSet(); /// Add the contact points from the narrow phase - void addContactPoints(NarrowPhaseInfo* narrowPhaseInfo); + void addContactPoints(const NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex); /// Return the first proxy shape ProxyShape* getShape1() const; diff --git a/src/collision/MiddlePhaseTriangleCallback.cpp b/src/collision/MiddlePhaseTriangleCallback.cpp index 7520fbae..9930bc98 100644 --- a/src/collision/MiddlePhaseTriangleCallback.cpp +++ b/src/collision/MiddlePhaseTriangleCallback.cpp @@ -26,8 +26,8 @@ // Libraries #include "collision/MiddlePhaseTriangleCallback.h" #include "engine/OverlappingPair.h" -#include "collision/NarrowPhaseInfo.h" #include "collision/shapes/TriangleShape.h" +#include "collision/NarrowPhaseInfoBatch.h" using namespace reactphysics3d; @@ -51,12 +51,8 @@ void MiddlePhaseTriangleCallback::testTriangle(const Vector3* trianglePoints, co ProxyShape* shape2 = isShape1Convex ? mConcaveProxyShape : mConvexProxyShape; // Create a narrow phase info for the narrow-phase collision detection - NarrowPhaseInfo* narrowPhaseInfo = new (mAllocator.allocate(sizeof(NarrowPhaseInfo))) - NarrowPhaseInfo(mOverlappingPair, - isShape1Convex ? mConvexProxyShape->getCollisionShape() : triangleShape, - isShape1Convex ? triangleShape : mConvexProxyShape->getCollisionShape(), - shape1->getLocalToWorldTransform(), - shape2->getLocalToWorldTransform(), - mAllocator); - mOutNarrowPhaseInfos.add(narrowPhaseInfo); + mOutNarrowPhaseInfoBatch.addNarrowPhaseInfo(mOverlappingPair, + isShape1Convex ? mConvexProxyShape->getCollisionShape() : triangleShape, + isShape1Convex ? triangleShape : mConvexProxyShape->getCollisionShape(), + shape1->getLocalToWorldTransform(), shape2->getLocalToWorldTransform(), mAllocator); } diff --git a/src/collision/MiddlePhaseTriangleCallback.h b/src/collision/MiddlePhaseTriangleCallback.h index a952d7b3..abe624fd 100644 --- a/src/collision/MiddlePhaseTriangleCallback.h +++ b/src/collision/MiddlePhaseTriangleCallback.h @@ -41,7 +41,7 @@ class NarrowPhaseAlgorithm; class ProxyShape; class MemoryAllocator; class Profiler; -struct NarrowPhaseInfo; +struct NarrowPhaseInfoBatch; struct Vector3; // Class ConvexVsTriangleCallback @@ -66,8 +66,8 @@ class MiddlePhaseTriangleCallback : public TriangleCallback { /// Pointer to the concave collision shape const ConcaveShape* mConcaveShape; - /// Reference to the list of narrow-phase infos - List& mOutNarrowPhaseInfos; + /// Reference to the narrow phase info batch + NarrowPhaseInfoBatch& mOutNarrowPhaseInfoBatch; /// Reference to the single-frame memory allocator MemoryAllocator& mAllocator; @@ -85,11 +85,11 @@ class MiddlePhaseTriangleCallback : public TriangleCallback { MiddlePhaseTriangleCallback(OverlappingPair* overlappingPair, ProxyShape* concaveProxyShape, ProxyShape* convexProxyShape, const ConcaveShape* concaveShape, - List& outNarrowPhaseInfos, + NarrowPhaseInfoBatch& outNarrowPhaseInfoBatch, MemoryAllocator& allocator) :mOverlappingPair(overlappingPair), mConcaveProxyShape(concaveProxyShape), mConvexProxyShape(convexProxyShape), mConcaveShape(concaveShape), - mOutNarrowPhaseInfos(outNarrowPhaseInfos), mAllocator(allocator) { + mOutNarrowPhaseInfoBatch(outNarrowPhaseInfoBatch), mAllocator(allocator) { } diff --git a/src/collision/NarrowPhaseInfo.cpp b/src/collision/NarrowPhaseInfoBatch.cpp similarity index 51% rename from src/collision/NarrowPhaseInfo.cpp rename to src/collision/NarrowPhaseInfoBatch.cpp index b1394c98..932b99ed 100644 --- a/src/collision/NarrowPhaseInfo.cpp +++ b/src/collision/NarrowPhaseInfoBatch.cpp @@ -24,7 +24,7 @@ ********************************************************************************/ // Libraries -#include "NarrowPhaseInfo.h" +#include "NarrowPhaseInfoBatch.h" #include "ContactPointInfo.h" #include "collision/shapes/TriangleShape.h" #include "engine/OverlappingPair.h" @@ -32,61 +32,63 @@ using namespace reactphysics3d; // Constructor -NarrowPhaseInfo::NarrowPhaseInfo(OverlappingPair* pair, CollisionShape* shape1, - CollisionShape* shape2, const Transform& shape1Transform, - const Transform& shape2Transform, MemoryAllocator& shapeAllocator) - : overlappingPair(pair), collisionShape1(shape1), collisionShape2(shape2), - shape1ToWorldTransform(shape1Transform), shape2ToWorldTransform(shape2Transform), - contactPoints(overlappingPair->getTemporaryAllocator()), collisionShapeAllocator(shapeAllocator) { +NarrowPhaseInfoBatch::NarrowPhaseInfoBatch(MemoryAllocator& allocator) + : mMemoryAllocator(allocator), overlappingPairs(allocator), collisionShapes1(allocator), collisionShapes2(allocator), + shape1ToWorldTransforms(allocator), shape2ToWorldTransforms(allocator), + isColliding(allocator), contactPoints(allocator), collisionShapeAllocators(allocator) { - // Add a collision info for the two collision shapes into the overlapping pair (if not present yet) - overlappingPair->addLastFrameInfoIfNecessary(shape1->getId(), shape2->getId()); } // Destructor -NarrowPhaseInfo::~NarrowPhaseInfo() { +NarrowPhaseInfoBatch::~NarrowPhaseInfoBatch() { + clear(); +} - assert(contactPoints.size() == 0); +// Add shapes to be tested during narrow-phase collision detection into the batch +void NarrowPhaseInfoBatch::addNarrowPhaseInfo(OverlappingPair* pair, CollisionShape* shape1, CollisionShape* shape2, + const Transform& shape1Transform, const Transform& shape2Transform, + MemoryAllocator& shapeAllocator) { - // Release the memory of the TriangleShape (this memory was allocated in the - // MiddlePhaseTriangleCallback::testTriangle() method) - if (collisionShape1->getName() == CollisionShapeName::TRIANGLE) { - collisionShape1->~CollisionShape(); - collisionShapeAllocator.release(collisionShape1, sizeof(TriangleShape)); - } - if (collisionShape2->getName() == CollisionShapeName::TRIANGLE) { - collisionShape2->~CollisionShape(); - collisionShapeAllocator.release(collisionShape2, sizeof(TriangleShape)); - } + overlappingPairs.add(pair); + collisionShapes1.add(shape1); + collisionShapes2.add(shape2); + shape1ToWorldTransforms.add(shape1Transform); + shape2ToWorldTransforms.add(shape2Transform); + collisionShapeAllocators.add(&shapeAllocator); + contactPoints.add(List(mMemoryAllocator)); + isColliding.add(false); + + // Add a collision info for the two collision shapes into the overlapping pair (if not present yet) + pair->addLastFrameInfoIfNecessary(shape1->getId(), shape2->getId()); } // Add a new contact point -void NarrowPhaseInfo::addContactPoint(const Vector3& contactNormal, decimal penDepth, +void NarrowPhaseInfoBatch::addContactPoint(uint index, const Vector3& contactNormal, decimal penDepth, const Vector3& localPt1, const Vector3& localPt2) { assert(penDepth > decimal(0.0)); // Get the memory allocator - MemoryAllocator& allocator = overlappingPair->getTemporaryAllocator(); + MemoryAllocator& allocator = overlappingPairs[index]->getTemporaryAllocator(); // Create the contact point info ContactPointInfo* contactPointInfo = new (allocator.allocate(sizeof(ContactPointInfo))) ContactPointInfo(contactNormal, penDepth, localPt1, localPt2); // Add it into the list of contact points - contactPoints.add(contactPointInfo); + contactPoints[index].add(contactPointInfo); } // Reset the remaining contact points -void NarrowPhaseInfo::resetContactPoints() { +void NarrowPhaseInfoBatch::resetContactPoints(uint index) { // Get the memory allocator - MemoryAllocator& allocator = overlappingPair->getTemporaryAllocator(); + MemoryAllocator& allocator = overlappingPairs[index]->getTemporaryAllocator(); // For each remaining contact point info - for (uint i=0; i < contactPoints.size(); i++) { + for (uint i=0; i < contactPoints[index].size(); i++) { - ContactPointInfo* contactPoint = contactPoints[i]; + ContactPointInfo* contactPoint = contactPoints[index][i]; // Call the destructor contactPoint->~ContactPointInfo(); @@ -95,5 +97,34 @@ void NarrowPhaseInfo::resetContactPoints() { allocator.release(contactPoint, sizeof(ContactPointInfo)); } + contactPoints[index].clear(); +} + +// Clear all the objects in the batch +void NarrowPhaseInfoBatch::clear() { + + for (uint i=0; i < overlappingPairs.size(); i++) { + + assert(contactPoints[i].size() == 0); + + // Release the memory of the TriangleShape (this memory was allocated in the + // MiddlePhaseTriangleCallback::testTriangle() method) + if (collisionShapes1[i]->getName() == CollisionShapeName::TRIANGLE) { + collisionShapes1[i]->~CollisionShape(); + collisionShapeAllocators[i]->release(collisionShapes1[i], sizeof(TriangleShape)); + } + if (collisionShapes2[i]->getName() == CollisionShapeName::TRIANGLE) { + collisionShapes2[i]->~CollisionShape(); + collisionShapeAllocators[i]->release(collisionShapes2[i], sizeof(TriangleShape)); + } + } + + overlappingPairs.clear(); + collisionShapes1.clear(); + collisionShapes2.clear(); + shape1ToWorldTransforms.clear(); + shape2ToWorldTransforms.clear(); + collisionShapeAllocators.clear(); + isColliding.clear(); contactPoints.clear(); } diff --git a/src/collision/NarrowPhaseInfo.h b/src/collision/NarrowPhaseInfoBatch.h similarity index 52% rename from src/collision/NarrowPhaseInfo.h rename to src/collision/NarrowPhaseInfoBatch.h index ff2cf5a0..6ccff688 100644 --- a/src/collision/NarrowPhaseInfo.h +++ b/src/collision/NarrowPhaseInfoBatch.h @@ -23,8 +23,8 @@ * * ********************************************************************************/ -#ifndef REACTPHYSICS3D_NARROW_PHASE_INFO_H -#define REACTPHYSICS3D_NARROW_PHASE_INFO_H +#ifndef REACTPHYSICS3D_NARROW_PHASE_INFO_BATCH_H +#define REACTPHYSICS3D_NARROW_PHASE_INFO_BATCH_H // Libraries #include "engine/OverlappingPair.h" @@ -38,58 +38,82 @@ struct LastFrameCollisionInfo; class ContactManifoldInfo; struct ContactPointInfo; -// Class NarrowPhaseInfo +// Struct NarrowPhaseInfoBatch /** - * This structure regroups different things about a collision shape. This is - * used to pass information about a collision shape to a collision algorithm. + * This abstract structure collects all the potential collisions from the middle-phase algorithm + * that have to be tested during narrow-phase collision detection. There is an implementation of + * this class for each kind of collision detection test. For instance, one for sphere vs sphere, + * one for sphere vs capsule, ... */ -struct NarrowPhaseInfo { +struct NarrowPhaseInfoBatch { + + private: + + /// Memory allocator + MemoryAllocator& mMemoryAllocator; public: - /// Broadphase overlapping pair - OverlappingPair* overlappingPair; + /// List of Broadphase overlapping pairs + List overlappingPairs; - /// Pointer to the first collision shape to test collision with - CollisionShape* collisionShape1; + /// List of pointers to the first collision shapes to test collision with + List collisionShapes1; - /// Pointer to the second collision shape to test collision with - CollisionShape* collisionShape2; + /// List of pointers to the second collision shapes to test collision with + List collisionShapes2; - /// Transform that maps from collision shape 1 local-space to world-space - Transform shape1ToWorldTransform; + /// List of transforms that maps from collision shape 1 local-space to world-space + List shape1ToWorldTransforms; - /// Transform that maps from collision shape 2 local-space to world-space - Transform shape2ToWorldTransform; + /// List of transforms that maps from collision shape 2 local-space to world-space + List shape2ToWorldTransforms; + + /// Result of the narrow-phase collision detection test + List isColliding; /// List of contact points created during the narrow-phase - List contactPoints; + List> contactPoints; - /// Memory allocator for the collision shape (Used to release TriangleShape memory in destructor) - MemoryAllocator& collisionShapeAllocator; + /// Memory allocators for the collision shape (Used to release TriangleShape memory in destructor) + List collisionShapeAllocators; /// Constructor - NarrowPhaseInfo(OverlappingPair* pair, CollisionShape* shape1, + NarrowPhaseInfoBatch(MemoryAllocator& allocator); + + /// Destructor + ~NarrowPhaseInfoBatch(); + + /// Return the number of objects in the batch + uint getNbObjects() const; + + /// Add shapes to be tested during narrow-phase collision detection into the batch + void addNarrowPhaseInfo(OverlappingPair* pair, CollisionShape* shape1, CollisionShape* shape2, const Transform& shape1Transform, const Transform& shape2Transform, MemoryAllocator& shapeAllocator); - /// Destructor - ~NarrowPhaseInfo(); - /// Add a new contact point - void addContactPoint(const Vector3& contactNormal, decimal penDepth, + void addContactPoint(uint index, const Vector3& contactNormal, decimal penDepth, const Vector3& localPt1, const Vector3& localPt2); /// Reset the remaining contact points - void resetContactPoints(); + void resetContactPoints(uint index); + + /// Clear all the objects in the batch + void clear(); /// Get the last collision frame info for temporal coherence - LastFrameCollisionInfo* getLastFrameCollisionInfo() const; + LastFrameCollisionInfo* getLastFrameCollisionInfo(uint index) const; }; +/// Return the number of objects in the batch +inline uint NarrowPhaseInfoBatch::getNbObjects() const { + return overlappingPairs.size(); +} + // Get the last collision frame info for temporal coherence -inline LastFrameCollisionInfo* NarrowPhaseInfo::getLastFrameCollisionInfo() const { - return overlappingPair->getLastFrameCollisionInfo(collisionShape1->getId(), collisionShape2->getId()); +inline LastFrameCollisionInfo* NarrowPhaseInfoBatch::getLastFrameCollisionInfo(uint index) const { + return overlappingPairs[index]->getLastFrameCollisionInfo(collisionShapes1[index]->getId(), collisionShapes2[index]->getId()); } } diff --git a/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.cpp b/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.cpp index 51e3e88e..f7075fcd 100755 --- a/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.cpp +++ b/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.cpp @@ -26,7 +26,7 @@ // Libraries #include "CapsuleVsCapsuleAlgorithm.h" #include "collision/shapes/CapsuleShape.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; @@ -34,195 +34,199 @@ using namespace reactphysics3d; // Compute the narrow-phase collision detection between two capsules // This technique is based on the "Robust Contact Creation for Physics Simulations" presentation // by Dirk Gregorius. -bool CapsuleVsCapsuleAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, +void CapsuleVsCapsuleAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, uint batchNbItems, bool reportContacts, MemoryAllocator& memoryAllocator) { - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CAPSULE); - assert(narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CAPSULE); + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - // Get the capsule collision shapes - const CapsuleShape* capsuleShape1 = static_cast(narrowPhaseInfo->collisionShape1); - const CapsuleShape* capsuleShape2 = static_cast(narrowPhaseInfo->collisionShape2); + assert(!narrowPhaseInfoBatch.isColliding[batchIndex]); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CAPSULE); + assert(narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CAPSULE); - // Get the transform from capsule 1 local-space to capsule 2 local-space - const Transform capsule1ToCapsule2SpaceTransform = narrowPhaseInfo->shape2ToWorldTransform.getInverse() * narrowPhaseInfo->shape1ToWorldTransform; + // Get the capsule collision shapes + const CapsuleShape* capsuleShape1 = static_cast(narrowPhaseInfoBatch.collisionShapes1[batchIndex]); + const CapsuleShape* capsuleShape2 = static_cast(narrowPhaseInfoBatch.collisionShapes2[batchIndex]); - // Compute the end-points of the inner segment of the first capsule - Vector3 capsule1SegA(0, -capsuleShape1->getHeight() * decimal(0.5), 0); - Vector3 capsule1SegB(0, capsuleShape1->getHeight() * decimal(0.5), 0); - capsule1SegA = capsule1ToCapsule2SpaceTransform * capsule1SegA; - capsule1SegB = capsule1ToCapsule2SpaceTransform * capsule1SegB; + // Get the transform from capsule 1 local-space to capsule 2 local-space + const Transform capsule1ToCapsule2SpaceTransform = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getInverse() * + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; - // Compute the end-points of the inner segment of the second capsule - const Vector3 capsule2SegA(0, -capsuleShape2->getHeight() * decimal(0.5), 0); - const Vector3 capsule2SegB(0, capsuleShape2->getHeight() * decimal(0.5), 0); - - // The two inner capsule segments - const Vector3 seg1 = capsule1SegB - capsule1SegA; - const Vector3 seg2 = capsule2SegB - capsule2SegA; + // Compute the end-points of the inner segment of the first capsule + Vector3 capsule1SegA(0, -capsuleShape1->getHeight() * decimal(0.5), 0); + Vector3 capsule1SegB(0, capsuleShape1->getHeight() * decimal(0.5), 0); + capsule1SegA = capsule1ToCapsule2SpaceTransform * capsule1SegA; + capsule1SegB = capsule1ToCapsule2SpaceTransform * capsule1SegB; - // Compute the sum of the radius of the two capsules (virtual spheres) - decimal sumRadius = capsuleShape2->getRadius() + capsuleShape1->getRadius(); + // Compute the end-points of the inner segment of the second capsule + const Vector3 capsule2SegA(0, -capsuleShape2->getHeight() * decimal(0.5), 0); + const Vector3 capsule2SegB(0, capsuleShape2->getHeight() * decimal(0.5), 0); - // If the two capsules are parallel (we create two contact points) - bool areCapsuleInnerSegmentsParralel = areParallelVectors(seg1, seg2); - if (areCapsuleInnerSegmentsParralel) { + // The two inner capsule segments + const Vector3 seg1 = capsule1SegB - capsule1SegA; + const Vector3 seg2 = capsule2SegB - capsule2SegA; - // If the distance between the two segments is larger than the sum of the capsules radius (we do not have overlapping) - const decimal segmentsPerpendicularDistance = computePointToLineDistance(capsule1SegA, capsule1SegB, capsule2SegA); - if (segmentsPerpendicularDistance >= sumRadius) { + // Compute the sum of the radius of the two capsules (virtual spheres) + decimal sumRadius = capsuleShape2->getRadius() + capsuleShape1->getRadius(); - // The capsule are parallel but their inner segment distance is larger than the sum of the capsules radius. - // Therefore, we do not have overlap. If the inner segments overlap, we do not report any collision. - return false; - } + // If the two capsules are parallel (we create two contact points) + bool areCapsuleInnerSegmentsParralel = areParallelVectors(seg1, seg2); + if (areCapsuleInnerSegmentsParralel) { - // Compute the planes that goes through the extreme points of the inner segment of capsule 1 - decimal d1 = seg1.dot(capsule1SegA); - decimal d2 = -seg1.dot(capsule1SegB); + // If the distance between the two segments is larger than the sum of the capsules radius (we do not have overlapping) + const decimal segmentsPerpendicularDistance = computePointToLineDistance(capsule1SegA, capsule1SegB, capsule2SegA); + if (segmentsPerpendicularDistance >= sumRadius) { - // Clip the inner segment of capsule 2 with the two planes that go through extreme points of inner - // segment of capsule 1 - decimal t1 = computePlaneSegmentIntersection(capsule2SegB, capsule2SegA, d1, seg1); - decimal t2 = computePlaneSegmentIntersection(capsule2SegA, capsule2SegB, d2, -seg1); + // The capsule are parallel but their inner segment distance is larger than the sum of the capsules radius. + // Therefore, we do not have overlap. If the inner segments overlap, we do not report any collision. + continue; + } - // If the segments were overlapping (the clip segment is valid) - if (t1 > decimal(0.0) && t2 > decimal(0.0)) { + // Compute the planes that goes through the extreme points of the inner segment of capsule 1 + decimal d1 = seg1.dot(capsule1SegA); + decimal d2 = -seg1.dot(capsule1SegB); + + // Clip the inner segment of capsule 2 with the two planes that go through extreme points of inner + // segment of capsule 1 + decimal t1 = computePlaneSegmentIntersection(capsule2SegB, capsule2SegA, d1, seg1); + decimal t2 = computePlaneSegmentIntersection(capsule2SegA, capsule2SegB, d2, -seg1); + + // If the segments were overlapping (the clip segment is valid) + if (t1 > decimal(0.0) && t2 > decimal(0.0)) { + + if (reportContacts) { + + // Clip the inner segment of capsule 2 + if (t1 > decimal(1.0)) t1 = decimal(1.0); + const Vector3 clipPointA = capsule2SegB - t1 * seg2; + if (t2 > decimal(1.0)) t2 = decimal(1.0); + const Vector3 clipPointB = capsule2SegA + t2 * seg2; + + // Project point capsule2SegA onto line of innner segment of capsule 1 + const Vector3 seg1Normalized = seg1.getUnit(); + Vector3 pointOnInnerSegCapsule1 = capsule1SegA + seg1Normalized.dot(capsule2SegA - capsule1SegA) * seg1Normalized; + + Vector3 normalCapsule2SpaceNormalized; + Vector3 segment1ToSegment2; + + // If the inner capsule segments perpendicular distance is not zero (the inner segments are not overlapping) + if (segmentsPerpendicularDistance > MACHINE_EPSILON) { + + // Compute a perpendicular vector from segment 1 to segment 2 + segment1ToSegment2 = (capsule2SegA - pointOnInnerSegCapsule1); + normalCapsule2SpaceNormalized = segment1ToSegment2.getUnit(); + } + else { // If the capsule inner segments are overlapping (degenerate case) + + // We cannot use the vector between segments as a contact normal. To generate a contact normal, we take + // any vector that is orthogonal to the inner capsule segments. + + Vector3 vec1(1, 0, 0); + Vector3 vec2(0, 1, 0); + + Vector3 seg2Normalized = seg2.getUnit(); + + // Get the vectors (among vec1 and vec2) that is the most orthogonal to the capsule 2 inner segment (smallest absolute dot product) + decimal cosA1 = std::abs(seg2Normalized.x); // abs(vec1.dot(seg2)) + decimal cosA2 = std::abs(seg2Normalized.y); // abs(vec2.dot(seg2)) + + segment1ToSegment2.setToZero(); + + // We choose as a contact normal, any direction that is perpendicular to the inner capsules segments + normalCapsule2SpaceNormalized = cosA1 < cosA2 ? seg2Normalized.cross(vec1) : seg2Normalized.cross(vec2); + } + + Transform capsule2ToCapsule1SpaceTransform = capsule1ToCapsule2SpaceTransform.getInverse(); + const Vector3 contactPointACapsule1Local = capsule2ToCapsule1SpaceTransform * (clipPointA - segment1ToSegment2 + normalCapsule2SpaceNormalized * capsuleShape1->getRadius()); + const Vector3 contactPointBCapsule1Local = capsule2ToCapsule1SpaceTransform * (clipPointB - segment1ToSegment2 + normalCapsule2SpaceNormalized * capsuleShape1->getRadius()); + const Vector3 contactPointACapsule2Local = clipPointA - normalCapsule2SpaceNormalized * capsuleShape2->getRadius(); + const Vector3 contactPointBCapsule2Local = clipPointB - normalCapsule2SpaceNormalized * capsuleShape2->getRadius(); + + decimal penetrationDepth = sumRadius - segmentsPerpendicularDistance; + + const Vector3 normalWorld = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * normalCapsule2SpaceNormalized; + + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, penetrationDepth, contactPointACapsule1Local, contactPointACapsule2Local); + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, penetrationDepth, contactPointBCapsule1Local, contactPointBCapsule2Local); + } + + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; + } + } + + // Compute the closest points between the two inner capsule segments + Vector3 closestPointCapsule1Seg; + Vector3 closestPointCapsule2Seg; + computeClosestPointBetweenTwoSegments(capsule1SegA, capsule1SegB, capsule2SegA, capsule2SegB, + closestPointCapsule1Seg, closestPointCapsule2Seg); + + // Compute the distance between the sphere center and the closest point on the segment + Vector3 closestPointsSeg1ToSeg2 = (closestPointCapsule2Seg - closestPointCapsule1Seg); + const decimal closestPointsDistanceSquare = closestPointsSeg1ToSeg2.lengthSquare(); + + // If the collision shapes overlap + if (closestPointsDistanceSquare < sumRadius * sumRadius) { if (reportContacts) { - // Clip the inner segment of capsule 2 - if (t1 > decimal(1.0)) t1 = decimal(1.0); - const Vector3 clipPointA = capsule2SegB - t1 * seg2; - if (t2 > decimal(1.0)) t2 = decimal(1.0); - const Vector3 clipPointB = capsule2SegA + t2 * seg2; + // If the distance between the inner segments is not zero + if (closestPointsDistanceSquare > MACHINE_EPSILON) { - // Project point capsule2SegA onto line of innner segment of capsule 1 - const Vector3 seg1Normalized = seg1.getUnit(); - Vector3 pointOnInnerSegCapsule1 = capsule1SegA + seg1Normalized.dot(capsule2SegA - capsule1SegA) * seg1Normalized; + decimal closestPointsDistance = std::sqrt(closestPointsDistanceSquare); + closestPointsSeg1ToSeg2 /= closestPointsDistance; - Vector3 normalCapsule2SpaceNormalized; - Vector3 segment1ToSegment2; + const Vector3 contactPointCapsule1Local = capsule1ToCapsule2SpaceTransform.getInverse() * (closestPointCapsule1Seg + closestPointsSeg1ToSeg2 * capsuleShape1->getRadius()); + const Vector3 contactPointCapsule2Local = closestPointCapsule2Seg - closestPointsSeg1ToSeg2 * capsuleShape2->getRadius(); - // If the inner capsule segments perpendicular distance is not zero (the inner segments are not overlapping) - if (segmentsPerpendicularDistance > MACHINE_EPSILON) { + const Vector3 normalWorld = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * closestPointsSeg1ToSeg2; - // Compute a perpendicular vector from segment 1 to segment 2 - segment1ToSegment2 = (capsule2SegA - pointOnInnerSegCapsule1); - normalCapsule2SpaceNormalized = segment1ToSegment2.getUnit(); - } - else { // If the capsule inner segments are overlapping (degenerate case) + decimal penetrationDepth = sumRadius - closestPointsDistance; - // We cannot use the vector between segments as a contact normal. To generate a contact normal, we take - // any vector that is orthogonal to the inner capsule segments. + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, penetrationDepth, contactPointCapsule1Local, contactPointCapsule2Local); + } + else { // The segment are overlapping (degenerate case) - Vector3 vec1(1, 0, 0); - Vector3 vec2(0, 1, 0); + // If the capsule segments are parralel + if (areCapsuleInnerSegmentsParralel) { - Vector3 seg2Normalized = seg2.getUnit(); + // The segment are parallel, not overlapping and their distance is zero. + // Therefore, the capsules are just touching at the top of their inner segments + decimal squareDistCapsule2PointToCapsuleSegA = (capsule1SegA - closestPointCapsule2Seg).lengthSquare(); - // Get the vectors (among vec1 and vec2) that is the most orthogonal to the capsule 2 inner segment (smallest absolute dot product) - decimal cosA1 = std::abs(seg2Normalized.x); // abs(vec1.dot(seg2)) - decimal cosA2 = std::abs(seg2Normalized.y); // abs(vec2.dot(seg2)) + Vector3 capsule1SegmentMostExtremePoint = squareDistCapsule2PointToCapsuleSegA > MACHINE_EPSILON ? capsule1SegA : capsule1SegB; + Vector3 normalCapsuleSpace2 = (closestPointCapsule2Seg - capsule1SegmentMostExtremePoint); + normalCapsuleSpace2.normalize(); - segment1ToSegment2.setToZero(); + const Vector3 contactPointCapsule1Local = capsule1ToCapsule2SpaceTransform.getInverse() * (closestPointCapsule1Seg + normalCapsuleSpace2 * capsuleShape1->getRadius()); + const Vector3 contactPointCapsule2Local = closestPointCapsule2Seg - normalCapsuleSpace2 * capsuleShape2->getRadius(); - // We choose as a contact normal, any direction that is perpendicular to the inner capsules segments - normalCapsule2SpaceNormalized = cosA1 < cosA2 ? seg2Normalized.cross(vec1) : seg2Normalized.cross(vec2); - } + const Vector3 normalWorld = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * normalCapsuleSpace2; - Transform capsule2ToCapsule1SpaceTransform = capsule1ToCapsule2SpaceTransform.getInverse(); - const Vector3 contactPointACapsule1Local = capsule2ToCapsule1SpaceTransform * (clipPointA - segment1ToSegment2 + normalCapsule2SpaceNormalized * capsuleShape1->getRadius()); - const Vector3 contactPointBCapsule1Local = capsule2ToCapsule1SpaceTransform * (clipPointB - segment1ToSegment2 + normalCapsule2SpaceNormalized * capsuleShape1->getRadius()); - const Vector3 contactPointACapsule2Local = clipPointA - normalCapsule2SpaceNormalized * capsuleShape2->getRadius(); - const Vector3 contactPointBCapsule2Local = clipPointB - normalCapsule2SpaceNormalized * capsuleShape2->getRadius(); + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, sumRadius, contactPointCapsule1Local, contactPointCapsule2Local); + } + else { // If the capsules inner segments are not parallel - decimal penetrationDepth = sumRadius - segmentsPerpendicularDistance; + // We cannot use a vector between the segments as contact normal. We need to compute a new contact normal with the cross + // product between the two segments. + Vector3 normalCapsuleSpace2 = seg1.cross(seg2); + normalCapsuleSpace2.normalize(); - const Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * normalCapsule2SpaceNormalized; + // Compute the contact points on both shapes + const Vector3 contactPointCapsule1Local = capsule1ToCapsule2SpaceTransform.getInverse() * (closestPointCapsule1Seg + normalCapsuleSpace2 * capsuleShape1->getRadius()); + const Vector3 contactPointCapsule2Local = closestPointCapsule2Seg - normalCapsuleSpace2 * capsuleShape2->getRadius(); - // Create the contact info object - narrowPhaseInfo->addContactPoint(normalWorld, penetrationDepth, contactPointACapsule1Local, contactPointACapsule2Local); - narrowPhaseInfo->addContactPoint(normalWorld, penetrationDepth, contactPointBCapsule1Local, contactPointBCapsule2Local); + const Vector3 normalWorld = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * normalCapsuleSpace2; + + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, sumRadius, contactPointCapsule1Local, contactPointCapsule2Local); + } + } } - return true; - } - } - - // Compute the closest points between the two inner capsule segments - Vector3 closestPointCapsule1Seg; - Vector3 closestPointCapsule2Seg; - computeClosestPointBetweenTwoSegments(capsule1SegA, capsule1SegB, capsule2SegA, capsule2SegB, - closestPointCapsule1Seg, closestPointCapsule2Seg); - - // Compute the distance between the sphere center and the closest point on the segment - Vector3 closestPointsSeg1ToSeg2 = (closestPointCapsule2Seg - closestPointCapsule1Seg); - const decimal closestPointsDistanceSquare = closestPointsSeg1ToSeg2.lengthSquare(); - - // If the collision shapes overlap - if (closestPointsDistanceSquare < sumRadius * sumRadius) { - - if (reportContacts) { - - // If the distance between the inner segments is not zero - if (closestPointsDistanceSquare > MACHINE_EPSILON) { - - decimal closestPointsDistance = std::sqrt(closestPointsDistanceSquare); - closestPointsSeg1ToSeg2 /= closestPointsDistance; - - const Vector3 contactPointCapsule1Local = capsule1ToCapsule2SpaceTransform.getInverse() * (closestPointCapsule1Seg + closestPointsSeg1ToSeg2 * capsuleShape1->getRadius()); - const Vector3 contactPointCapsule2Local = closestPointCapsule2Seg - closestPointsSeg1ToSeg2 * capsuleShape2->getRadius(); - - const Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * closestPointsSeg1ToSeg2; - - decimal penetrationDepth = sumRadius - closestPointsDistance; - - // Create the contact info object - narrowPhaseInfo->addContactPoint(normalWorld, penetrationDepth, contactPointCapsule1Local, contactPointCapsule2Local); - } - else { // The segment are overlapping (degenerate case) - - // If the capsule segments are parralel - if (areCapsuleInnerSegmentsParralel) { - - // The segment are parallel, not overlapping and their distance is zero. - // Therefore, the capsules are just touching at the top of their inner segments - decimal squareDistCapsule2PointToCapsuleSegA = (capsule1SegA - closestPointCapsule2Seg).lengthSquare(); - - Vector3 capsule1SegmentMostExtremePoint = squareDistCapsule2PointToCapsuleSegA > MACHINE_EPSILON ? capsule1SegA : capsule1SegB; - Vector3 normalCapsuleSpace2 = (closestPointCapsule2Seg - capsule1SegmentMostExtremePoint); - normalCapsuleSpace2.normalize(); - - const Vector3 contactPointCapsule1Local = capsule1ToCapsule2SpaceTransform.getInverse() * (closestPointCapsule1Seg + normalCapsuleSpace2 * capsuleShape1->getRadius()); - const Vector3 contactPointCapsule2Local = closestPointCapsule2Seg - normalCapsuleSpace2 * capsuleShape2->getRadius(); - - const Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * normalCapsuleSpace2; - - // Create the contact info object - narrowPhaseInfo->addContactPoint(normalWorld, sumRadius, contactPointCapsule1Local, contactPointCapsule2Local); - } - else { // If the capsules inner segments are not parallel - - // We cannot use a vector between the segments as contact normal. We need to compute a new contact normal with the cross - // product between the two segments. - Vector3 normalCapsuleSpace2 = seg1.cross(seg2); - normalCapsuleSpace2.normalize(); - - // Compute the contact points on both shapes - const Vector3 contactPointCapsule1Local = capsule1ToCapsule2SpaceTransform.getInverse() * (closestPointCapsule1Seg + normalCapsuleSpace2 * capsuleShape1->getRadius()); - const Vector3 contactPointCapsule2Local = closestPointCapsule2Seg - normalCapsuleSpace2 * capsuleShape2->getRadius(); - - const Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * normalCapsuleSpace2; - - // Create the contact info object - narrowPhaseInfo->addContactPoint(normalWorld, sumRadius, contactPointCapsule1Local, contactPointCapsule2Local); - } - } - } - - return true; - } - - return false; + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + } + } } diff --git a/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.h b/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.h index a178a2a8..3e1a5871 100644 --- a/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.h +++ b/src/collision/narrowphase/CapsuleVsCapsuleAlgorithm.h @@ -28,11 +28,13 @@ // Libraries #include "NarrowPhaseAlgorithm.h" +#include "configuration.h" /// Namespace ReactPhysics3D namespace reactphysics3d { // Declarations +struct NarrowPhaseInfoBatch; class Body; class ContactPoint; @@ -65,7 +67,9 @@ class CapsuleVsCapsuleAlgorithm : public NarrowPhaseAlgorithm { CapsuleVsCapsuleAlgorithm& operator=(const CapsuleVsCapsuleAlgorithm& algorithm) = delete; /// Compute the narrow-phase collision detection between two capsules - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, MemoryAllocator& memoryAllocator) override; + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, + MemoryAllocator& memoryAllocator) override; }; } diff --git a/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.cpp b/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.cpp index 5751578e..c34aa031 100644 --- a/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.cpp +++ b/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.cpp @@ -29,7 +29,7 @@ #include "GJK/GJKAlgorithm.h" #include "collision/shapes/CapsuleShape.h" #include "collision/shapes/ConvexPolyhedronShape.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" #include "collision/ContactPointInfo.h" #include @@ -39,7 +39,9 @@ using namespace reactphysics3d; // Compute the narrow-phase collision detection between a capsule and a polyhedron // This technique is based on the "Robust Contact Creation for Physics Simulations" presentation // by Dirk Gregorius. -bool CapsuleVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, +void CapsuleVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, + uint batchStartIndex, uint batchNbItems, + bool reportContacts, MemoryAllocator& memoryAllocator) { // First, we run the GJK algorithm @@ -53,114 +55,128 @@ bool CapsuleVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfo* narrowPh #endif - // Get the last frame collision info - LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfo->getLastFrameCollisionInfo(); + // Run the GJK algorithm + List gjkResults(memoryAllocator); + gjkAlgorithm.testCollision(narrowPhaseInfoBatch, batchStartIndex, batchNbItems, gjkResults, reportContacts); + assert(gjkResults.size() == batchNbItems); - GJKAlgorithm::GJKResult result = gjkAlgorithm.testCollision(narrowPhaseInfo, reportContacts); + uint resultIndex = 0; + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - lastFrameCollisionInfo->wasUsingGJK = true; - lastFrameCollisionInfo->wasUsingSAT = false; + // Get the last frame collision info + LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfoBatch.getLastFrameCollisionInfo(batchIndex); - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CONVEX_POLYHEDRON || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CONVEX_POLYHEDRON); - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CAPSULE || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CAPSULE); + lastFrameCollisionInfo->wasUsingGJK = true; + lastFrameCollisionInfo->wasUsingSAT = false; - // If we have found a contact point inside the margins (shallow penetration) - if (result == GJKAlgorithm::GJKResult::COLLIDE_IN_MARGIN) { + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CAPSULE || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CAPSULE); - if (reportContacts) { + // If we have found a contact point inside the margins (shallow penetration) + if (gjkResults[resultIndex] == GJKAlgorithm::GJKResult::COLLIDE_IN_MARGIN) { - // GJK has found a shallow contact. If the face of the polyhedron mesh is orthogonal to the - // capsule inner segment and parallel to the contact point normal, we would like to create - // two contact points instead of a single one (as in the deep contact case with SAT algorithm) + if (reportContacts) { - // Get the contact point created by GJK - assert(narrowPhaseInfo->contactPoints.size() > 0); - ContactPointInfo*& contactPoint = narrowPhaseInfo->contactPoints[0]; + bool noContact = false; - bool isCapsuleShape1 = narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CAPSULE; + // GJK has found a shallow contact. If the face of the polyhedron mesh is orthogonal to the + // capsule inner segment and parallel to the contact point normal, we would like to create + // two contact points instead of a single one (as in the deep contact case with SAT algorithm) - // Get the collision shapes - const CapsuleShape* capsuleShape = static_cast(isCapsuleShape1 ? narrowPhaseInfo->collisionShape1 : narrowPhaseInfo->collisionShape2); - const ConvexPolyhedronShape* polyhedron = static_cast(isCapsuleShape1 ? narrowPhaseInfo->collisionShape2 : narrowPhaseInfo->collisionShape1); + // Get the contact point created by GJK + assert(narrowPhaseInfoBatch.contactPoints[batchIndex].size() > 0); + ContactPointInfo*& contactPoint = narrowPhaseInfoBatch.contactPoints[batchIndex][0]; - // For each face of the polyhedron - for (uint f = 0; f < polyhedron->getNbFaces(); f++) { + bool isCapsuleShape1 = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CAPSULE; - const Transform polyhedronToWorld = isCapsuleShape1 ? narrowPhaseInfo->shape2ToWorldTransform : narrowPhaseInfo->shape1ToWorldTransform; - const Transform capsuleToWorld = isCapsuleShape1 ? narrowPhaseInfo->shape1ToWorldTransform : narrowPhaseInfo->shape2ToWorldTransform; + // Get the collision shapes + const CapsuleShape* capsuleShape = static_cast(isCapsuleShape1 ? narrowPhaseInfoBatch.collisionShapes1[batchIndex] : narrowPhaseInfoBatch.collisionShapes2[batchIndex]); + const ConvexPolyhedronShape* polyhedron = static_cast(isCapsuleShape1 ? narrowPhaseInfoBatch.collisionShapes2[batchIndex] : narrowPhaseInfoBatch.collisionShapes1[batchIndex]); - // Get the face normal - const Vector3 faceNormal = polyhedron->getFaceNormal(f); - Vector3 faceNormalWorld = polyhedronToWorld.getOrientation() * faceNormal; + // For each face of the polyhedron + for (uint f = 0; f < polyhedron->getNbFaces(); f++) { - const Vector3 capsuleSegA(0, -capsuleShape->getHeight() * decimal(0.5), 0); - const Vector3 capsuleSegB(0, capsuleShape->getHeight() * decimal(0.5), 0); - Vector3 capsuleInnerSegmentDirection = capsuleToWorld.getOrientation() * (capsuleSegB - capsuleSegA); - capsuleInnerSegmentDirection.normalize(); + const Transform polyhedronToWorld = isCapsuleShape1 ? narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; + const Transform capsuleToWorld = isCapsuleShape1 ? narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; - bool isFaceNormalInDirectionOfContactNormal = faceNormalWorld.dot(contactPoint->normal) > decimal(0.0); - bool isFaceNormalInContactDirection = (isCapsuleShape1 && !isFaceNormalInDirectionOfContactNormal) || (!isCapsuleShape1 && isFaceNormalInDirectionOfContactNormal); + // Get the face normal + const Vector3 faceNormal = polyhedron->getFaceNormal(f); + Vector3 faceNormalWorld = polyhedronToWorld.getOrientation() * faceNormal; - // If the polyhedron face normal is orthogonal to the capsule inner segment and parallel to the contact point normal and the face normal - // is in direction of the contact normal (from the polyhedron point of view). - if (isFaceNormalInContactDirection && areOrthogonalVectors(faceNormalWorld, capsuleInnerSegmentDirection) - && areParallelVectors(faceNormalWorld, contactPoint->normal)) { - - // Remove the previous contact point computed by GJK - narrowPhaseInfo->resetContactPoints(); - - const Transform capsuleToWorld = isCapsuleShape1 ? narrowPhaseInfo->shape1ToWorldTransform : narrowPhaseInfo->shape2ToWorldTransform; - const Transform polyhedronToCapsuleTransform = capsuleToWorld.getInverse() * polyhedronToWorld; - - // Compute the end-points of the inner segment of the capsule const Vector3 capsuleSegA(0, -capsuleShape->getHeight() * decimal(0.5), 0); const Vector3 capsuleSegB(0, capsuleShape->getHeight() * decimal(0.5), 0); + Vector3 capsuleInnerSegmentDirection = capsuleToWorld.getOrientation() * (capsuleSegB - capsuleSegA); + capsuleInnerSegmentDirection.normalize(); - // Convert the inner capsule segment points into the polyhedron local-space - const Transform capsuleToPolyhedronTransform = polyhedronToCapsuleTransform.getInverse(); - const Vector3 capsuleSegAPolyhedronSpace = capsuleToPolyhedronTransform * capsuleSegA; - const Vector3 capsuleSegBPolyhedronSpace = capsuleToPolyhedronTransform * capsuleSegB; + bool isFaceNormalInDirectionOfContactNormal = faceNormalWorld.dot(contactPoint->normal) > decimal(0.0); + bool isFaceNormalInContactDirection = (isCapsuleShape1 && !isFaceNormalInDirectionOfContactNormal) || (!isCapsuleShape1 && isFaceNormalInDirectionOfContactNormal); - const Vector3 separatingAxisCapsuleSpace = polyhedronToCapsuleTransform.getOrientation() * faceNormal; + // If the polyhedron face normal is orthogonal to the capsule inner segment and parallel to the contact point normal and the face normal + // is in direction of the contact normal (from the polyhedron point of view). + if (isFaceNormalInContactDirection && areOrthogonalVectors(faceNormalWorld, capsuleInnerSegmentDirection) + && areParallelVectors(faceNormalWorld, contactPoint->normal)) { - if (isCapsuleShape1) { - faceNormalWorld = -faceNormalWorld; + // Remove the previous contact point computed by GJK + narrowPhaseInfoBatch.resetContactPoints(batchIndex); + + const Transform capsuleToWorld = isCapsuleShape1 ? narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; + const Transform polyhedronToCapsuleTransform = capsuleToWorld.getInverse() * polyhedronToWorld; + + // Compute the end-points of the inner segment of the capsule + const Vector3 capsuleSegA(0, -capsuleShape->getHeight() * decimal(0.5), 0); + const Vector3 capsuleSegB(0, capsuleShape->getHeight() * decimal(0.5), 0); + + // Convert the inner capsule segment points into the polyhedron local-space + const Transform capsuleToPolyhedronTransform = polyhedronToCapsuleTransform.getInverse(); + const Vector3 capsuleSegAPolyhedronSpace = capsuleToPolyhedronTransform * capsuleSegA; + const Vector3 capsuleSegBPolyhedronSpace = capsuleToPolyhedronTransform * capsuleSegB; + + const Vector3 separatingAxisCapsuleSpace = polyhedronToCapsuleTransform.getOrientation() * faceNormal; + + if (isCapsuleShape1) { + faceNormalWorld = -faceNormalWorld; + } + + // Compute and create two contact points + bool contactsFound = satAlgorithm.computeCapsulePolyhedronFaceContactPoints(f, capsuleShape->getRadius(), polyhedron, contactPoint->penetrationDepth, + polyhedronToCapsuleTransform, faceNormalWorld, separatingAxisCapsuleSpace, + capsuleSegAPolyhedronSpace, capsuleSegBPolyhedronSpace, + narrowPhaseInfoBatch, batchIndex, isCapsuleShape1); + if (!contactsFound) { + noContact = true; + narrowPhaseInfoBatch.isColliding[batchIndex] = false; + break; + } + + break; } + } - // Compute and create two contact points - bool contactsFound = satAlgorithm.computeCapsulePolyhedronFaceContactPoints(f, capsuleShape->getRadius(), polyhedron, contactPoint->penetrationDepth, - polyhedronToCapsuleTransform, faceNormalWorld, separatingAxisCapsuleSpace, - capsuleSegAPolyhedronSpace, capsuleSegBPolyhedronSpace, - narrowPhaseInfo, isCapsuleShape1); - if (!contactsFound) { - return false; - } - - break; + if (noContact) { + continue; } } + + lastFrameCollisionInfo->wasUsingSAT = false; + lastFrameCollisionInfo->wasUsingGJK = false; + + // Return true + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; } - lastFrameCollisionInfo->wasUsingSAT = false; - lastFrameCollisionInfo->wasUsingGJK = false; + // If we have overlap even without the margins (deep penetration) + if (gjkResults[resultIndex] == GJKAlgorithm::GJKResult::INTERPENETRATE) { - // Return true - return true; + // Run the SAT algorithm to find the separating axis and compute contact point + narrowPhaseInfoBatch.isColliding[batchIndex] = satAlgorithm.testCollisionCapsuleVsConvexPolyhedron(narrowPhaseInfoBatch, batchIndex, reportContacts); + + lastFrameCollisionInfo->wasUsingGJK = false; + lastFrameCollisionInfo->wasUsingSAT = true; + } + + resultIndex++; } - - // If we have overlap even without the margins (deep penetration) - if (result == GJKAlgorithm::GJKResult::INTERPENETRATE) { - - // Run the SAT algorithm to find the separating axis and compute contact point - bool isColliding = satAlgorithm.testCollisionCapsuleVsConvexPolyhedron(narrowPhaseInfo, reportContacts); - - lastFrameCollisionInfo->wasUsingGJK = false; - lastFrameCollisionInfo->wasUsingSAT = true; - - return isColliding; - } - - return false; } diff --git a/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.h b/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.h index 7f6b0453..3ccfc932 100644 --- a/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.h +++ b/src/collision/narrowphase/CapsuleVsConvexPolyhedronAlgorithm.h @@ -70,7 +70,9 @@ class CapsuleVsConvexPolyhedronAlgorithm : public NarrowPhaseAlgorithm { CapsuleVsConvexPolyhedronAlgorithm& operator=(const CapsuleVsConvexPolyhedronAlgorithm& algorithm) = delete; /// Compute the narrow-phase collision detection between a capsule and a polyhedron - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, MemoryAllocator& memoryAllocator) override; + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, + MemoryAllocator& memoryAllocator) override; }; } diff --git a/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.cpp b/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.cpp index 4f8676fa..eb8cd159 100644 --- a/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.cpp +++ b/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.cpp @@ -27,7 +27,7 @@ #include "ConvexPolyhedronVsConvexPolyhedronAlgorithm.h" #include "GJK/GJKAlgorithm.h" #include "SAT/SATAlgorithm.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; @@ -35,7 +35,9 @@ using namespace reactphysics3d; // Compute the narrow-phase collision detection between two convex polyhedra // This technique is based on the "Robust Contact Creation for Physics Simulations" presentation // by Dirk Gregorius. -bool ConvexPolyhedronVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, +void ConvexPolyhedronVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, + uint batchStartIndex, uint batchNbItems, + bool reportContacts, MemoryAllocator& memoryAllocator) { // Run the SAT algorithm to find the separating axis and compute contact point @@ -47,13 +49,14 @@ bool ConvexPolyhedronVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfo* #endif - // Get the last frame collision info - LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfo->getLastFrameCollisionInfo(); + satAlgorithm.testCollisionConvexPolyhedronVsConvexPolyhedron(narrowPhaseInfoBatch, batchStartIndex, batchNbItems, reportContacts); - bool isColliding = satAlgorithm.testCollisionConvexPolyhedronVsConvexPolyhedron(narrowPhaseInfo, reportContacts); + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - lastFrameCollisionInfo->wasUsingSAT = true; - lastFrameCollisionInfo->wasUsingGJK = false; + // Get the last frame collision info + LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfoBatch.getLastFrameCollisionInfo(batchIndex); - return isColliding; + lastFrameCollisionInfo->wasUsingSAT = true; + lastFrameCollisionInfo->wasUsingGJK = false; + } } diff --git a/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.h b/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.h index c4bc5e23..9cf47689 100644 --- a/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.h +++ b/src/collision/narrowphase/ConvexPolyhedronVsConvexPolyhedronAlgorithm.h @@ -65,7 +65,9 @@ class ConvexPolyhedronVsConvexPolyhedronAlgorithm : public NarrowPhaseAlgorithm ConvexPolyhedronVsConvexPolyhedronAlgorithm& operator=(const ConvexPolyhedronVsConvexPolyhedronAlgorithm& algorithm) = delete; /// Compute the narrow-phase collision detection between two convex polyhedra - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, MemoryAllocator& memoryAllocator) override; + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, + MemoryAllocator& memoryAllocator) override; }; } diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp index c40ba7ed..e046fcfd 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.cpp +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.cpp @@ -30,7 +30,8 @@ #include "collision/shapes/TriangleShape.h" #include "configuration.h" #include "utils/Profiler.h" -#include "collision/NarrowPhaseInfo.h" +#include "containers/List.h" +#include "collision/NarrowPhaseInfoBatch.h" #include "collision/narrowphase/GJK/VoronoiSimplex.h" #include @@ -46,168 +47,184 @@ using namespace reactphysics3d; /// algorithm on the enlarged object to obtain a simplex polytope that contains the /// origin, they we give that simplex polytope to the EPA algorithm which will compute /// the correct penetration depth and contact points between the enlarged objects. -GJKAlgorithm::GJKResult GJKAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) { +void GJKAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, List& gjkResults, bool reportContacts) { RP3D_PROFILE("GJKAlgorithm::testCollision()", mProfiler); - Vector3 suppA; // Support point of object A - Vector3 suppB; // Support point of object B - Vector3 w; // Support point of Minkowski difference A-B - Vector3 pA; // Closest point of object A - Vector3 pB; // Closest point of object B - decimal vDotw; - decimal prevDistSquare; - bool contactFound = false; + // For each item in the batch + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - assert(narrowPhaseInfo->collisionShape1->isConvex()); - assert(narrowPhaseInfo->collisionShape2->isConvex()); + Vector3 suppA; // Support point of object A + Vector3 suppB; // Support point of object B + Vector3 w; // Support point of Minkowski difference A-B + Vector3 pA; // Closest point of object A + Vector3 pB; // Closest point of object B + decimal vDotw; + decimal prevDistSquare; + bool contactFound = false; - const ConvexShape* shape1 = static_cast(narrowPhaseInfo->collisionShape1); - const ConvexShape* shape2 = static_cast(narrowPhaseInfo->collisionShape2); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->isConvex()); + assert(narrowPhaseInfoBatch.collisionShapes2[batchIndex]->isConvex()); - // Get the local-space to world-space transforms - const Transform& transform1 = narrowPhaseInfo->shape1ToWorldTransform; - const Transform& transform2 = narrowPhaseInfo->shape2ToWorldTransform; + const ConvexShape* shape1 = static_cast(narrowPhaseInfoBatch.collisionShapes1[batchIndex]); + const ConvexShape* shape2 = static_cast(narrowPhaseInfoBatch.collisionShapes2[batchIndex]); - // Transform a point from local space of body 2 to local - // space of body 1 (the GJK algorithm is done in local space of body 1) - Transform transform1Inverse = transform1.getInverse(); - Transform body2Tobody1 = transform1Inverse * transform2; + // Get the local-space to world-space transforms + const Transform& transform1 = narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; + const Transform& transform2 = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; - // Quaternion that transform a direction from local - // space of body 1 into local space of body 2 - Quaternion rotateToBody2 = transform2.getOrientation().getInverse() * transform1.getOrientation(); + // Transform a point from local space of body 2 to local + // space of body 1 (the GJK algorithm is done in local space of body 1) + Transform transform1Inverse = transform1.getInverse(); + Transform body2Tobody1 = transform1Inverse * transform2; - // Initialize the margin (sum of margins of both objects) - decimal margin = shape1->getMargin() + shape2->getMargin(); - decimal marginSquare = margin * margin; - assert(margin > decimal(0.0)); + // Quaternion that transform a direction from local + // space of body 1 into local space of body 2 + Quaternion rotateToBody2 = transform2.getOrientation().getInverse() * transform1.getOrientation(); - // Create a simplex set - VoronoiSimplex simplex; + // Initialize the margin (sum of margins of both objects) + decimal margin = shape1->getMargin() + shape2->getMargin(); + decimal marginSquare = margin * margin; + assert(margin > decimal(0.0)); - // Get the last collision frame info - LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfo->getLastFrameCollisionInfo(); + // Create a simplex set + VoronoiSimplex simplex; - // Get the previous point V (last cached separating axis) - Vector3 v; - if (lastFrameCollisionInfo->isValid && lastFrameCollisionInfo->wasUsingGJK) { - v = lastFrameCollisionInfo->gjkSeparatingAxis; - assert(v.lengthSquare() > decimal(0.000001)); - } - else { - v.setAllValues(0, 1, 0); - } + // Get the last collision frame info + LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfoBatch.getLastFrameCollisionInfo(batchIndex); - // Initialize the upper bound for the square distance - decimal distSquare = DECIMAL_LARGEST; - - do { - - // Compute the support points for original objects (without margins) A and B - suppA = shape1->getLocalSupportPointWithoutMargin(-v); - suppB = body2Tobody1 * shape2->getLocalSupportPointWithoutMargin(rotateToBody2 * v); - - // Compute the support point for the Minkowski difference A-B - w = suppA - suppB; - - vDotw = v.dot(w); - - // If the enlarge objects (with margins) do not intersect - if (vDotw > decimal(0.0) && vDotw * vDotw > distSquare * marginSquare) { - - // Cache the current separating axis for frame coherence - lastFrameCollisionInfo->gjkSeparatingAxis = v; - - // No intersection, we return - return GJKResult::SEPARATED; + // Get the previous point V (last cached separating axis) + Vector3 v; + if (lastFrameCollisionInfo->isValid && lastFrameCollisionInfo->wasUsingGJK) { + v = lastFrameCollisionInfo->gjkSeparatingAxis; + assert(v.lengthSquare() > decimal(0.000001)); + } + else { + v.setAllValues(0, 1, 0); } - // If the objects intersect only in the margins - if (simplex.isPointInSimplex(w) || distSquare - vDotw <= distSquare * REL_ERROR_SQUARE) { + // Initialize the upper bound for the square distance + decimal distSquare = DECIMAL_LARGEST; - // Contact point has been found - contactFound = true; - break; - } + bool noIntersection = false; - // Add the new support point to the simplex - simplex.addPoint(w, suppA, suppB); + do { - // If the simplex is affinely dependent - if (simplex.isAffinelyDependent()) { + // Compute the support points for original objects (without margins) A and B + suppA = shape1->getLocalSupportPointWithoutMargin(-v); + suppB = body2Tobody1 * shape2->getLocalSupportPointWithoutMargin(rotateToBody2 * v); - // Contact point has been found - contactFound = true; - break; - } + // Compute the support point for the Minkowski difference A-B + w = suppA - suppB; - // Compute the point of the simplex closest to the origin - // If the computation of the closest point fails - if (!simplex.computeClosestPoint(v)) { + vDotw = v.dot(w); - // Contact point has been found - contactFound = true; - break; - } + // If the enlarge objects (with margins) do not intersect + if (vDotw > decimal(0.0) && vDotw * vDotw > distSquare * marginSquare) { - // Store and update the squared distance of the closest point - prevDistSquare = distSquare; - distSquare = v.lengthSquare(); + // Cache the current separating axis for frame coherence + lastFrameCollisionInfo->gjkSeparatingAxis = v; - // If the distance to the closest point doesn't improve a lot - if (prevDistSquare - distSquare <= MACHINE_EPSILON * prevDistSquare) { + // No intersection, we return + gjkResults.add(GJKResult::SEPARATED); + noIntersection = true; + break; + } - simplex.backupClosestPointInSimplex(v); - - // Get the new squared distance + // If the objects intersect only in the margins + if (simplex.isPointInSimplex(w) || distSquare - vDotw <= distSquare * REL_ERROR_SQUARE) { + + // Contact point has been found + contactFound = true; + break; + } + + // Add the new support point to the simplex + simplex.addPoint(w, suppA, suppB); + + // If the simplex is affinely dependent + if (simplex.isAffinelyDependent()) { + + // Contact point has been found + contactFound = true; + break; + } + + // Compute the point of the simplex closest to the origin + // If the computation of the closest point fails + if (!simplex.computeClosestPoint(v)) { + + // Contact point has been found + contactFound = true; + break; + } + + // Store and update the squared distance of the closest point + prevDistSquare = distSquare; distSquare = v.lengthSquare(); - // Contact point has been found - contactFound = true; - break; + // If the distance to the closest point doesn't improve a lot + if (prevDistSquare - distSquare <= MACHINE_EPSILON * prevDistSquare) { + + simplex.backupClosestPointInSimplex(v); + + // Get the new squared distance + distSquare = v.lengthSquare(); + + // Contact point has been found + contactFound = true; + break; + } + + } while(!simplex.isFull() && distSquare > MACHINE_EPSILON * simplex.getMaxLengthSquareOfAPoint()); + + if (noIntersection) { + continue; } - } while(!simplex.isFull() && distSquare > MACHINE_EPSILON * simplex.getMaxLengthSquareOfAPoint()); + if (contactFound && distSquare > MACHINE_EPSILON) { - if (contactFound && distSquare > MACHINE_EPSILON) { + // Compute the closet points of both objects (without the margins) + simplex.computeClosestPointsOfAandB(pA, pB); - // Compute the closet points of both objects (without the margins) - simplex.computeClosestPointsOfAandB(pA, pB); + // Project those two points on the margins to have the closest points of both + // object with the margins + decimal dist = std::sqrt(distSquare); + assert(dist > decimal(0.0)); + pA = (pA - (shape1->getMargin() / dist) * v); + pB = body2Tobody1.getInverse() * (pB + (shape2->getMargin() / dist) * v); - // Project those two points on the margins to have the closest points of both - // object with the margins - decimal dist = std::sqrt(distSquare); - assert(dist > decimal(0.0)); - pA = (pA - (shape1->getMargin() / dist) * v); - pB = body2Tobody1.getInverse() * (pB + (shape2->getMargin() / dist) * v); + // Compute the contact info + Vector3 normal = transform1.getOrientation() * (-v.getUnit()); + decimal penetrationDepth = margin - dist; - // Compute the contact info - Vector3 normal = transform1.getOrientation() * (-v.getUnit()); - decimal penetrationDepth = margin - dist; + // If the penetration depth is negative (due too numerical errors), there is no contact + if (penetrationDepth <= decimal(0.0)) { + gjkResults.add(GJKResult::SEPARATED); + continue; + } - // If the penetration depth is negative (due too numerical errors), there is no contact - if (penetrationDepth <= decimal(0.0)) { - return GJKResult::SEPARATED; + // Do not generate a contact point with zero normal length + if (normal.lengthSquare() < MACHINE_EPSILON) { + gjkResults.add(GJKResult::SEPARATED); + continue; + } + + if (reportContacts) { + + // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle + TriangleShape::computeSmoothTriangleMeshContact(shape1, shape2, pA, pB, transform1, transform2, + penetrationDepth, normal); + + // Add a new contact point + narrowPhaseInfoBatch.addContactPoint(batchIndex, normal, penetrationDepth, pA, pB); + } + + gjkResults.add(GJKResult::COLLIDE_IN_MARGIN); + continue; } - // Do not generate a contact point with zero normal length - if (normal.lengthSquare() < MACHINE_EPSILON) { - return GJKResult::SEPARATED; - } - - if (reportContacts) { - - // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(shape1, shape2, pA, pB, transform1, transform2, - penetrationDepth, normal); - - // Add a new contact point - narrowPhaseInfo->addContactPoint(normal, penetrationDepth, pA, pB); - } - - return GJKResult::COLLIDE_IN_MARGIN; + gjkResults.add(GJKResult::INTERPENETRATE); } - - return GJKResult::INTERPENETRATE; } diff --git a/src/collision/narrowphase/GJK/GJKAlgorithm.h b/src/collision/narrowphase/GJK/GJKAlgorithm.h index 48b75c5c..5404f1c1 100644 --- a/src/collision/narrowphase/GJK/GJKAlgorithm.h +++ b/src/collision/narrowphase/GJK/GJKAlgorithm.h @@ -28,16 +28,18 @@ // Libraries #include "decimal.h" +#include "configuration.h" /// ReactPhysics3D namespace namespace reactphysics3d { // Declarations class ContactManifoldInfo; -struct NarrowPhaseInfo; +struct NarrowPhaseInfoBatch; class ConvexShape; class Profiler; class VoronoiSimplex; +template class List; // Constants constexpr decimal REL_ERROR = decimal(1.0e-3); @@ -95,7 +97,8 @@ class GJKAlgorithm { GJKAlgorithm& operator=(const GJKAlgorithm& algorithm) = delete; /// Compute a contact info if the two bounding volumes collide. - GJKResult testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts); + void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, List& gjkResults, bool reportContacts); #ifdef IS_PROFILING_ACTIVE diff --git a/src/collision/narrowphase/NarrowPhaseAlgorithm.h b/src/collision/narrowphase/NarrowPhaseAlgorithm.h index e7cdff25..6d376172 100644 --- a/src/collision/narrowphase/NarrowPhaseAlgorithm.h +++ b/src/collision/narrowphase/NarrowPhaseAlgorithm.h @@ -27,6 +27,7 @@ #define REACTPHYSICS3D_NARROW_PHASE_ALGORITHM_H // Libraries +#include "configuration.h" /// Namespace ReactPhysics3D namespace reactphysics3d { @@ -36,7 +37,7 @@ class Body; class ContactManifoldInfo; class PoolAllocator; class OverlappingPair; -struct NarrowPhaseInfo; +struct NarrowPhaseInfoBatch; struct ContactPointInfo; class Profiler; class MemoryAllocator; @@ -94,7 +95,8 @@ class NarrowPhaseAlgorithm { NarrowPhaseAlgorithm& operator=(const NarrowPhaseAlgorithm& algorithm) = delete; /// Compute a contact info if the two bounding volumes collide - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, MemoryAllocator& memoryAllocator)=0; #ifdef IS_PROFILING_ACTIVE diff --git a/src/collision/narrowphase/SAT/SATAlgorithm.cpp b/src/collision/narrowphase/SAT/SATAlgorithm.cpp index 374a812a..d88911e6 100644 --- a/src/collision/narrowphase/SAT/SATAlgorithm.cpp +++ b/src/collision/narrowphase/SAT/SATAlgorithm.cpp @@ -30,8 +30,8 @@ #include "collision/shapes/CapsuleShape.h" #include "collision/shapes/SphereShape.h" #include "engine/OverlappingPair.h" +#include "collision/NarrowPhaseInfoBatch.h" #include "collision/shapes/TriangleShape.h" -#include "collision/NarrowPhaseInfo.h" #include "configuration.h" #include "utils/Profiler.h" #include @@ -52,77 +52,88 @@ SATAlgorithm::SATAlgorithm(MemoryAllocator& memoryAllocator) : mMemoryAllocator( } // Test collision between a sphere and a convex mesh -bool SATAlgorithm::testCollisionSphereVsConvexPolyhedron(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) const { +void SATAlgorithm::testCollisionSphereVsConvexPolyhedron(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, + uint batchStartIndex, uint batchNbItems, + bool reportContacts) const { RP3D_PROFILE("SATAlgorithm::testCollisionSphereVsConvexPolyhedron()", mProfiler); - bool isSphereShape1 = narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::SPHERE; + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CONVEX_POLYHEDRON || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CONVEX_POLYHEDRON); - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::SPHERE || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::SPHERE); + bool isSphereShape1 = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::SPHERE; - // Get the capsule collision shapes - const SphereShape* sphere = static_cast(isSphereShape1 ? narrowPhaseInfo->collisionShape1 : narrowPhaseInfo->collisionShape2); - const ConvexPolyhedronShape* polyhedron = static_cast(isSphereShape1 ? narrowPhaseInfo->collisionShape2 : narrowPhaseInfo->collisionShape1); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::SPHERE || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::SPHERE); - const Transform& sphereToWorldTransform = isSphereShape1 ? narrowPhaseInfo->shape1ToWorldTransform : narrowPhaseInfo->shape2ToWorldTransform; - const Transform& polyhedronToWorldTransform = isSphereShape1 ? narrowPhaseInfo->shape2ToWorldTransform : narrowPhaseInfo->shape1ToWorldTransform; + // Get the capsule collision shapes + const SphereShape* sphere = static_cast(isSphereShape1 ? narrowPhaseInfoBatch.collisionShapes1[batchIndex] : narrowPhaseInfoBatch.collisionShapes2[batchIndex]); + const ConvexPolyhedronShape* polyhedron = static_cast(isSphereShape1 ? narrowPhaseInfoBatch.collisionShapes2[batchIndex] : narrowPhaseInfoBatch.collisionShapes1[batchIndex]); - // Get the transform from sphere local-space to polyhedron local-space - const Transform worldToPolyhedronTransform = polyhedronToWorldTransform.getInverse(); - const Transform sphereToPolyhedronSpaceTransform = worldToPolyhedronTransform * sphereToWorldTransform; + const Transform& sphereToWorldTransform = isSphereShape1 ? narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; + const Transform& polyhedronToWorldTransform = isSphereShape1 ? narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; - // Transform the center of the sphere into the local-space of the convex polyhedron - const Vector3 sphereCenter = sphereToPolyhedronSpaceTransform.getPosition(); + // Get the transform from sphere local-space to polyhedron local-space + const Transform worldToPolyhedronTransform = polyhedronToWorldTransform.getInverse(); + const Transform sphereToPolyhedronSpaceTransform = worldToPolyhedronTransform * sphereToWorldTransform; - // Minimum penetration depth - decimal minPenetrationDepth = DECIMAL_LARGEST; - uint minFaceIndex = 0; + // Transform the center of the sphere into the local-space of the convex polyhedron + const Vector3 sphereCenter = sphereToPolyhedronSpaceTransform.getPosition(); - // For each face of the convex mesh - for (uint f = 0; f < polyhedron->getNbFaces(); f++) { + // Minimum penetration depth + decimal minPenetrationDepth = DECIMAL_LARGEST; + uint minFaceIndex = 0; + bool noContact = false; - // Compute the penetration depth of the shapes along the face normal direction - decimal penetrationDepth = computePolyhedronFaceVsSpherePenetrationDepth(f, polyhedron, sphere, sphereCenter); + // For each face of the convex mesh + for (uint f = 0; f < polyhedron->getNbFaces(); f++) { - // If the penetration depth is negative, we have found a separating axis - if (penetrationDepth <= decimal(0.0)) { + // Compute the penetration depth of the shapes along the face normal direction + decimal penetrationDepth = computePolyhedronFaceVsSpherePenetrationDepth(f, polyhedron, sphere, sphereCenter); - return false; + // If the penetration depth is negative, we have found a separating axis + if (penetrationDepth <= decimal(0.0)) { + + noContact = true; + break; + } + + // Check if we have found a new minimum penetration axis + if (penetrationDepth < minPenetrationDepth) { + minPenetrationDepth = penetrationDepth; + minFaceIndex = f; + } } - // Check if we have found a new minimum penetration axis - if (penetrationDepth < minPenetrationDepth) { - minPenetrationDepth = penetrationDepth; - minFaceIndex = f; + if (noContact) { + continue; } + + if (reportContacts) { + + const Vector3 minFaceNormal = polyhedron->getFaceNormal(minFaceIndex); + Vector3 minFaceNormalWorld = polyhedronToWorldTransform.getOrientation() * minFaceNormal; + Vector3 contactPointSphereLocal = sphereToWorldTransform.getInverse().getOrientation() * (-minFaceNormalWorld * sphere->getRadius()); + Vector3 contactPointPolyhedronLocal = sphereCenter + minFaceNormal * (minPenetrationDepth - sphere->getRadius()); + + Vector3 normalWorld = isSphereShape1 ? -minFaceNormalWorld : minFaceNormalWorld; + + // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle + TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfoBatch.collisionShapes1[batchIndex], narrowPhaseInfoBatch.collisionShapes2[batchIndex], + isSphereShape1 ? contactPointSphereLocal : contactPointPolyhedronLocal, + isSphereShape1 ? contactPointPolyhedronLocal : contactPointSphereLocal, + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex], narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex], + minPenetrationDepth, normalWorld); + + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, minPenetrationDepth, + isSphereShape1 ? contactPointSphereLocal : contactPointPolyhedronLocal, + isSphereShape1 ? contactPointPolyhedronLocal : contactPointSphereLocal); + } + + narrowPhaseInfoBatch.isColliding[batchIndex] = true; } - - if (reportContacts) { - - const Vector3 minFaceNormal = polyhedron->getFaceNormal(minFaceIndex); - Vector3 minFaceNormalWorld = polyhedronToWorldTransform.getOrientation() * minFaceNormal; - Vector3 contactPointSphereLocal = sphereToWorldTransform.getInverse().getOrientation() * (-minFaceNormalWorld * sphere->getRadius()); - Vector3 contactPointPolyhedronLocal = sphereCenter + minFaceNormal * (minPenetrationDepth - sphere->getRadius()); - - Vector3 normalWorld = isSphereShape1 ? -minFaceNormalWorld : minFaceNormalWorld; - - // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfo->collisionShape1, narrowPhaseInfo->collisionShape2, - isSphereShape1 ? contactPointSphereLocal : contactPointPolyhedronLocal, - isSphereShape1 ? contactPointPolyhedronLocal : contactPointSphereLocal, - narrowPhaseInfo->shape1ToWorldTransform, narrowPhaseInfo->shape2ToWorldTransform, - minPenetrationDepth, normalWorld); - - // Create the contact info object - narrowPhaseInfo->addContactPoint(normalWorld, minPenetrationDepth, - isSphereShape1 ? contactPointSphereLocal : contactPointPolyhedronLocal, - isSphereShape1 ? contactPointPolyhedronLocal : contactPointSphereLocal); - } - - return true; } // Compute the penetration depth between a face of the polyhedron and a sphere along the polyhedron face normal direction @@ -144,23 +155,23 @@ decimal SATAlgorithm::computePolyhedronFaceVsSpherePenetrationDepth(uint faceInd } // Test collision between a capsule and a convex mesh -bool SATAlgorithm::testCollisionCapsuleVsConvexPolyhedron(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) const { +bool SATAlgorithm::testCollisionCapsuleVsConvexPolyhedron(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex, bool reportContacts) const { RP3D_PROFILE("SATAlgorithm::testCollisionCapsuleVsConvexPolyhedron()", mProfiler); - bool isCapsuleShape1 = narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CAPSULE; + bool isCapsuleShape1 = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CAPSULE; - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CONVEX_POLYHEDRON || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CONVEX_POLYHEDRON); - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CAPSULE || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CAPSULE); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CAPSULE || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CAPSULE); // Get the collision shapes - const CapsuleShape* capsuleShape = static_cast(isCapsuleShape1 ? narrowPhaseInfo->collisionShape1 : narrowPhaseInfo->collisionShape2); - const ConvexPolyhedronShape* polyhedron = static_cast(isCapsuleShape1 ? narrowPhaseInfo->collisionShape2 : narrowPhaseInfo->collisionShape1); + const CapsuleShape* capsuleShape = static_cast(isCapsuleShape1 ? narrowPhaseInfoBatch.collisionShapes1[batchIndex] : narrowPhaseInfoBatch.collisionShapes2[batchIndex]); + const ConvexPolyhedronShape* polyhedron = static_cast(isCapsuleShape1 ? narrowPhaseInfoBatch.collisionShapes2[batchIndex] : narrowPhaseInfoBatch.collisionShapes1[batchIndex]); - const Transform capsuleToWorld = isCapsuleShape1 ? narrowPhaseInfo->shape1ToWorldTransform : narrowPhaseInfo->shape2ToWorldTransform; - const Transform polyhedronToWorld = isCapsuleShape1 ? narrowPhaseInfo->shape2ToWorldTransform : narrowPhaseInfo->shape1ToWorldTransform; + const Transform capsuleToWorld = isCapsuleShape1 ? narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; + const Transform polyhedronToWorld = isCapsuleShape1 ? narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; const Transform polyhedronToCapsuleTransform = capsuleToWorld.getInverse() * polyhedronToWorld; @@ -264,7 +275,7 @@ bool SATAlgorithm::testCollisionCapsuleVsConvexPolyhedron(NarrowPhaseInfo* narro return computeCapsulePolyhedronFaceContactPoints(minFaceIndex, capsuleRadius, polyhedron, minPenetrationDepth, polyhedronToCapsuleTransform, normalWorld, separatingAxisCapsuleSpace, capsuleSegAPolyhedronSpace, capsuleSegBPolyhedronSpace, - narrowPhaseInfo, isCapsuleShape1); + narrowPhaseInfoBatch, batchIndex, isCapsuleShape1); } } else { // The separating axis is the cross product of a polyhedron edge and the inner capsule segment @@ -282,14 +293,14 @@ bool SATAlgorithm::testCollisionCapsuleVsConvexPolyhedron(NarrowPhaseInfo* narro Vector3 contactPointCapsule = (polyhedronToCapsuleTransform * closestPointCapsuleInnerSegment) - separatingAxisCapsuleSpace * capsuleRadius; // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfo->collisionShape1, narrowPhaseInfo->collisionShape2, + TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfoBatch.collisionShapes1[batchIndex], narrowPhaseInfoBatch.collisionShapes2[batchIndex], isCapsuleShape1 ? contactPointCapsule : closestPointPolyhedronEdge, isCapsuleShape1 ? closestPointPolyhedronEdge : contactPointCapsule, - narrowPhaseInfo->shape1ToWorldTransform, narrowPhaseInfo->shape2ToWorldTransform, + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex], narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex], minPenetrationDepth, normalWorld); // Create the contact point - narrowPhaseInfo->addContactPoint(normalWorld, minPenetrationDepth, + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, minPenetrationDepth, isCapsuleShape1 ? contactPointCapsule : closestPointPolyhedronEdge, isCapsuleShape1 ? closestPointPolyhedronEdge : contactPointCapsule); } @@ -362,7 +373,7 @@ bool SATAlgorithm::computeCapsulePolyhedronFaceContactPoints(uint referenceFaceI decimal penetrationDepth, const Transform& polyhedronToCapsuleTransform, Vector3& normalWorld, const Vector3& separatingAxisCapsuleSpace, const Vector3& capsuleSegAPolyhedronSpace, const Vector3& capsuleSegBPolyhedronSpace, - NarrowPhaseInfo* narrowPhaseInfo, bool isCapsuleShape1) const { + NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex, bool isCapsuleShape1) const { RP3D_PROFILE("SATAlgorithm::computeCapsulePolyhedronFaceContactPoints", mProfiler); @@ -427,15 +438,15 @@ bool SATAlgorithm::computeCapsulePolyhedronFaceContactPoints(uint referenceFaceI // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfo->collisionShape1, narrowPhaseInfo->collisionShape2, + TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfoBatch.collisionShapes1[batchIndex], narrowPhaseInfoBatch.collisionShapes2[batchIndex], isCapsuleShape1 ? contactPointCapsule : contactPointPolyhedron, isCapsuleShape1 ? contactPointPolyhedron : contactPointCapsule, - narrowPhaseInfo->shape1ToWorldTransform, narrowPhaseInfo->shape2ToWorldTransform, + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex], narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex], penetrationDepth, normalWorld); // Create the contact point - narrowPhaseInfo->addContactPoint(normalWorld, penetrationDepth, + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, penetrationDepth, isCapsuleShape1 ? contactPointCapsule : contactPointPolyhedron, isCapsuleShape1 ? contactPointPolyhedron : contactPointCapsule); } @@ -457,377 +468,394 @@ bool SATAlgorithm::isMinkowskiFaceCapsuleVsEdge(const Vector3& capsuleSegment, c } // Test collision between two convex polyhedrons -bool SATAlgorithm::testCollisionConvexPolyhedronVsConvexPolyhedron(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) const { +void SATAlgorithm::testCollisionConvexPolyhedronVsConvexPolyhedron(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, uint batchNbItems, bool reportContacts) const { RP3D_PROFILE("SATAlgorithm::testCollisionConvexPolyhedronVsConvexPolyhedron()", mProfiler); - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CONVEX_POLYHEDRON); - assert(narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CONVEX_POLYHEDRON); + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - const ConvexPolyhedronShape* polyhedron1 = static_cast(narrowPhaseInfo->collisionShape1); - const ConvexPolyhedronShape* polyhedron2 = static_cast(narrowPhaseInfo->collisionShape2); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON); + assert(narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON); - const Transform polyhedron1ToPolyhedron2 = narrowPhaseInfo->shape2ToWorldTransform.getInverse() * narrowPhaseInfo->shape1ToWorldTransform; - const Transform polyhedron2ToPolyhedron1 = polyhedron1ToPolyhedron2.getInverse(); + const ConvexPolyhedronShape* polyhedron1 = static_cast(narrowPhaseInfoBatch.collisionShapes1[batchIndex]); + const ConvexPolyhedronShape* polyhedron2 = static_cast(narrowPhaseInfoBatch.collisionShapes2[batchIndex]); - decimal minPenetrationDepth = DECIMAL_LARGEST; - uint minFaceIndex = 0; - bool isMinPenetrationFaceNormal = false; - bool isMinPenetrationFaceNormalPolyhedron1 = false; - uint minSeparatingEdge1Index = 0; - uint minSeparatingEdge2Index = 0; - Vector3 separatingEdge1A, separatingEdge1B; - Vector3 separatingEdge2A, separatingEdge2B; - Vector3 minEdgeVsEdgeSeparatingAxisPolyhedron2Space; - bool isShape1Triangle = polyhedron1->getName() == CollisionShapeName::TRIANGLE; + const Transform polyhedron1ToPolyhedron2 = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getInverse() * narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; + const Transform polyhedron2ToPolyhedron1 = polyhedron1ToPolyhedron2.getInverse(); - LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfo->getLastFrameCollisionInfo(); + decimal minPenetrationDepth = DECIMAL_LARGEST; + uint minFaceIndex = 0; + bool isMinPenetrationFaceNormal = false; + bool isMinPenetrationFaceNormalPolyhedron1 = false; + uint minSeparatingEdge1Index = 0; + uint minSeparatingEdge2Index = 0; + Vector3 separatingEdge1A, separatingEdge1B; + Vector3 separatingEdge2A, separatingEdge2B; + Vector3 minEdgeVsEdgeSeparatingAxisPolyhedron2Space; + bool isShape1Triangle = polyhedron1->getName() == CollisionShapeName::TRIANGLE; - // If the last frame collision info is valid and was also using SAT algorithm - if (lastFrameCollisionInfo->isValid && lastFrameCollisionInfo->wasUsingSAT) { + LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfoBatch.getLastFrameCollisionInfo(batchIndex); - // We perform temporal coherence, we check if there is still an overlapping along the previous minimum separating - // axis. If it is the case, we directly report the collision without executing the whole SAT algorithm again. If - // the shapes are still separated along this axis, we directly exit with no collision. + // If the last frame collision info is valid and was also using SAT algorithm + if (lastFrameCollisionInfo->isValid && lastFrameCollisionInfo->wasUsingSAT) { - // If the previous separating axis (or axis with minimum penetration depth) - // was a face normal of polyhedron 1 - if (lastFrameCollisionInfo->satIsAxisFacePolyhedron1) { + // We perform temporal coherence, we check if there is still an overlapping along the previous minimum separating + // axis. If it is the case, we directly report the collision without executing the whole SAT algorithm again. If + // the shapes are still separated along this axis, we directly exit with no collision. - decimal penetrationDepth = testSingleFaceDirectionPolyhedronVsPolyhedron(polyhedron1, polyhedron2, polyhedron1ToPolyhedron2, - lastFrameCollisionInfo->satMinAxisFaceIndex); + // If the previous separating axis (or axis with minimum penetration depth) + // was a face normal of polyhedron 1 + if (lastFrameCollisionInfo->satIsAxisFacePolyhedron1) { - // If the previous axis was a separating axis and is still a separating axis in this frame - if (!lastFrameCollisionInfo->wasColliding && penetrationDepth <= decimal(0.0)) { + decimal penetrationDepth = testSingleFaceDirectionPolyhedronVsPolyhedron(polyhedron1, polyhedron2, polyhedron1ToPolyhedron2, + lastFrameCollisionInfo->satMinAxisFaceIndex); - // Return no collision without running the whole SAT algorithm - return false; - } - - // The two shapes were overlapping in the previous frame and still seem to overlap in this one - if (lastFrameCollisionInfo->wasColliding && penetrationDepth > decimal(0.0)) { - - minPenetrationDepth = penetrationDepth; - minFaceIndex = lastFrameCollisionInfo->satMinAxisFaceIndex; - isMinPenetrationFaceNormal = true; - isMinPenetrationFaceNormalPolyhedron1 = true; - - // Compute the contact points between two faces of two convex polyhedra. - if(computePolyhedronVsPolyhedronFaceContactPoints(isMinPenetrationFaceNormalPolyhedron1, polyhedron1, polyhedron2, - polyhedron1ToPolyhedron2, polyhedron2ToPolyhedron1, minFaceIndex, - narrowPhaseInfo, minPenetrationDepth)) { - - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; - - // The shapes are still overlapping in the previous axis (the contact manifold is not empty). - // Therefore, we can return without running the whole SAT algorithm - return true; - } - - // The contact manifold is empty. Therefore, we have to run the whole SAT algorithm again - } - } - else if (lastFrameCollisionInfo->satIsAxisFacePolyhedron2) { // If the previous separating axis (or axis with minimum penetration depth) - // was a face normal of polyhedron 2 - - decimal penetrationDepth = testSingleFaceDirectionPolyhedronVsPolyhedron(polyhedron2, polyhedron1, polyhedron2ToPolyhedron1, - lastFrameCollisionInfo->satMinAxisFaceIndex); - - // If the previous axis was a separating axis and is still a separating axis in this frame - if (!lastFrameCollisionInfo->wasColliding && penetrationDepth <= decimal(0.0)) { - - // Return no collision without running the whole SAT algorithm - return false; - } - - // The two shapes were overlapping in the previous frame and still seem to overlap in this one - if (lastFrameCollisionInfo->wasColliding && penetrationDepth > decimal(0.0)) { - - minPenetrationDepth = penetrationDepth; - minFaceIndex = lastFrameCollisionInfo->satMinAxisFaceIndex; - isMinPenetrationFaceNormal = true; - isMinPenetrationFaceNormalPolyhedron1 = false; - - // Compute the contact points between two faces of two convex polyhedra. - if(computePolyhedronVsPolyhedronFaceContactPoints(isMinPenetrationFaceNormalPolyhedron1, polyhedron1, polyhedron2, - polyhedron1ToPolyhedron2, polyhedron2ToPolyhedron1, minFaceIndex, - narrowPhaseInfo, minPenetrationDepth)) { - - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; - - // The shapes are still overlapping in the previous axis (the contact manifold is not empty). - // Therefore, we can return without running the whole SAT algorithm - return true; - } - - // The contact manifold is empty. Therefore, we have to run the whole SAT algorithm again - } - } - else { // If the previous separating axis (or axis with minimum penetration depth) was the cross product of two edges - - const HalfEdgeStructure::Edge& edge1 = polyhedron1->getHalfEdge(lastFrameCollisionInfo->satMinEdge1Index); - const HalfEdgeStructure::Edge& edge2 = polyhedron2->getHalfEdge(lastFrameCollisionInfo->satMinEdge2Index); - - const Vector3 edge1A = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(edge1.vertexIndex); - const Vector3 edge1B = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(polyhedron1->getHalfEdge(edge1.nextEdgeIndex).vertexIndex); - const Vector3 edge1Direction = edge1B - edge1A; - const Vector3 edge2A = polyhedron2->getVertexPosition(edge2.vertexIndex); - const Vector3 edge2B = polyhedron2->getVertexPosition(polyhedron2->getHalfEdge(edge2.nextEdgeIndex).vertexIndex); - const Vector3 edge2Direction = edge2B - edge2A; - - // If the two edges build a minkowski face (and the cross product is - // therefore a candidate for separating axis - if (testEdgesBuildMinkowskiFace(polyhedron1, edge1, polyhedron2, edge2, polyhedron1ToPolyhedron2)) { - - Vector3 separatingAxisPolyhedron2Space; - - // Compute the penetration depth along the previous axis - const Vector3 polyhedron1Centroid = polyhedron1ToPolyhedron2 * polyhedron1->getCentroid(); - decimal penetrationDepth = computeDistanceBetweenEdges(edge1A, edge2A, polyhedron1Centroid, polyhedron2->getCentroid(), - edge1Direction, edge2Direction, isShape1Triangle, separatingAxisPolyhedron2Space); - - // If the shapes were not overlapping in the previous frame and are still not - // overlapping in the current one + // If the previous axis was a separating axis and is still a separating axis in this frame if (!lastFrameCollisionInfo->wasColliding && penetrationDepth <= decimal(0.0)) { - // We have found a separating axis without running the whole SAT algorithm - return false; + // Return no collision without running the whole SAT algorithm + continue; } - // If the shapes were overlapping on the previous axis and still seem to overlap in this frame + // The two shapes were overlapping in the previous frame and still seem to overlap in this one if (lastFrameCollisionInfo->wasColliding && penetrationDepth > decimal(0.0)) { - // Compute the closest points between the two edges (in the local-space of poylhedron 2) - Vector3 closestPointPolyhedron1Edge, closestPointPolyhedron2Edge; - computeClosestPointBetweenTwoSegments(edge1A, edge1B, edge2A, edge2B, - closestPointPolyhedron1Edge, closestPointPolyhedron2Edge); + minPenetrationDepth = penetrationDepth; + minFaceIndex = lastFrameCollisionInfo->satMinAxisFaceIndex; + isMinPenetrationFaceNormal = true; + isMinPenetrationFaceNormalPolyhedron1 = true; - // Here we try to project the closest point on edge1 onto the segment of edge 2 to see if - // the projected point falls onto the segment. We also try to project the closest point - // on edge 2 to see if it falls onto the segment of edge 1. If one of the point does not - // fall onto the opposite segment, it means the edges are not colliding (the contact manifold - // is empty). Therefore, we need to run the whole SAT algorithm again. - const Vector3 vec1 = closestPointPolyhedron1Edge - edge2A; - const Vector3 vec2 = closestPointPolyhedron2Edge - edge1A; - const decimal edge1LengthSquare = edge1Direction.lengthSquare(); - const decimal edge2LengthSquare = edge2Direction.lengthSquare(); - decimal t1 = vec1.dot(edge2Direction) / edge2LengthSquare; - decimal t2 = vec2.dot(edge1Direction) / edge1LengthSquare; - if (t1 >= decimal(0.0) && t1 <= decimal(1) && t2 >= decimal(0.0) && t2 <= decimal(1.0)) { + // Compute the contact points between two faces of two convex polyhedra. + if(computePolyhedronVsPolyhedronFaceContactPoints(isMinPenetrationFaceNormalPolyhedron1, polyhedron1, polyhedron2, + polyhedron1ToPolyhedron2, polyhedron2ToPolyhedron1, minFaceIndex, + narrowPhaseInfoBatch, batchIndex, minPenetrationDepth)) { - // Compute the contact point on polyhedron 1 edge in the local-space of polyhedron 1 - Vector3 closestPointPolyhedron1EdgeLocalSpace = polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge; + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; - // Compute the world normal - // We use the direction from the centroid to the edge of the shape that is not a triangle - // to avoid possible degeneracies when axis direction is orthogonal to triangle normal - Vector3 normal; - if (isShape1Triangle) { - normal = polyhedron2->getCentroid() - closestPointPolyhedron2Edge; - } - else { - normal = polyhedron1ToPolyhedron2.getOrientation() * ((polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge) - polyhedron1->getCentroid()); - } - - //Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * minEdgeVsEdgeSeparatingAxisPolyhedron2Space; - Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * normal.getUnit(); - - // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfo->collisionShape1, narrowPhaseInfo->collisionShape2, - closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge, - narrowPhaseInfo->shape1ToWorldTransform, narrowPhaseInfo->shape2ToWorldTransform, - penetrationDepth, normalWorld); - - // Create the contact point - narrowPhaseInfo->addContactPoint(normalWorld, penetrationDepth, - closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge); - - // The shapes are overlapping on the previous axis (the contact manifold is not empty). Therefore - // we return without running the whole SAT algorithm - return true; + // The shapes are still overlapping in the previous axis (the contact manifold is not empty). + // Therefore, we can return without running the whole SAT algorithm + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; } // The contact manifold is empty. Therefore, we have to run the whole SAT algorithm again } } - } - } + else if (lastFrameCollisionInfo->satIsAxisFacePolyhedron2) { // If the previous separating axis (or axis with minimum penetration depth) + // was a face normal of polyhedron 2 - // Test all the face normals of the polyhedron 1 for separating axis - uint faceIndex; - decimal penetrationDepth = testFacesDirectionPolyhedronVsPolyhedron(polyhedron1, polyhedron2, polyhedron1ToPolyhedron2, faceIndex); - if (penetrationDepth <= decimal(0.0)) { + decimal penetrationDepth = testSingleFaceDirectionPolyhedronVsPolyhedron(polyhedron2, polyhedron1, polyhedron2ToPolyhedron1, + lastFrameCollisionInfo->satMinAxisFaceIndex); - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = true; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = false; - lastFrameCollisionInfo->satMinAxisFaceIndex = faceIndex; + // If the previous axis was a separating axis and is still a separating axis in this frame + if (!lastFrameCollisionInfo->wasColliding && penetrationDepth <= decimal(0.0)) { - // We have found a separating axis - return false; - } - if (penetrationDepth < minPenetrationDepth - SAME_SEPARATING_AXIS_BIAS) { - isMinPenetrationFaceNormal = true; - minPenetrationDepth = penetrationDepth; - minFaceIndex = faceIndex; - isMinPenetrationFaceNormalPolyhedron1 = true; - } - - // Test all the face normals of the polyhedron 2 for separating axis - penetrationDepth = testFacesDirectionPolyhedronVsPolyhedron(polyhedron2, polyhedron1, polyhedron2ToPolyhedron1, faceIndex); - if (penetrationDepth <= decimal(0.0)) { - - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = false; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = true; - lastFrameCollisionInfo->satMinAxisFaceIndex = faceIndex; - - // We have found a separating axis - return false; - } - if (penetrationDepth < minPenetrationDepth - SAME_SEPARATING_AXIS_BIAS) { - isMinPenetrationFaceNormal = true; - minPenetrationDepth = penetrationDepth; - minFaceIndex = faceIndex; - isMinPenetrationFaceNormalPolyhedron1 = false; - } - - // Test the cross products of edges of polyhedron 1 with edges of polyhedron 2 for separating axis - for (uint i=0; i < polyhedron1->getNbHalfEdges(); i += 2) { - - // Get an edge of polyhedron 1 - const HalfEdgeStructure::Edge& edge1 = polyhedron1->getHalfEdge(i); - - const Vector3 edge1A = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(edge1.vertexIndex); - const Vector3 edge1B = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(polyhedron1->getHalfEdge(edge1.nextEdgeIndex).vertexIndex); - const Vector3 edge1Direction = edge1B - edge1A; - - for (uint j=0; j < polyhedron2->getNbHalfEdges(); j += 2) { - - // Get an edge of polyhedron 2 - const HalfEdgeStructure::Edge& edge2 = polyhedron2->getHalfEdge(j); - - const Vector3 edge2A = polyhedron2->getVertexPosition(edge2.vertexIndex); - const Vector3 edge2B = polyhedron2->getVertexPosition(polyhedron2->getHalfEdge(edge2.nextEdgeIndex).vertexIndex); - const Vector3 edge2Direction = edge2B - edge2A; - - // If the two edges build a minkowski face (and the cross product is - // therefore a candidate for separating axis - if (testEdgesBuildMinkowskiFace(polyhedron1, edge1, polyhedron2, edge2, polyhedron1ToPolyhedron2)) { - - Vector3 separatingAxisPolyhedron2Space; - - // Compute the penetration depth - const Vector3 polyhedron1Centroid = polyhedron1ToPolyhedron2 * polyhedron1->getCentroid(); - decimal penetrationDepth = computeDistanceBetweenEdges(edge1A, edge2A, polyhedron1Centroid, polyhedron2->getCentroid(), - edge1Direction, edge2Direction, isShape1Triangle, separatingAxisPolyhedron2Space); - - if (penetrationDepth <= decimal(0.0)) { - - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = false; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = false; - lastFrameCollisionInfo->satMinEdge1Index = i; - lastFrameCollisionInfo->satMinEdge2Index = j; - - // We have found a separating axis - return false; + // Return no collision without running the whole SAT algorithm + continue; } - if (penetrationDepth < minPenetrationDepth - SAME_SEPARATING_AXIS_BIAS) { + // The two shapes were overlapping in the previous frame and still seem to overlap in this one + if (lastFrameCollisionInfo->wasColliding && penetrationDepth > decimal(0.0)) { minPenetrationDepth = penetrationDepth; + minFaceIndex = lastFrameCollisionInfo->satMinAxisFaceIndex; + isMinPenetrationFaceNormal = true; isMinPenetrationFaceNormalPolyhedron1 = false; - isMinPenetrationFaceNormal = false; - minSeparatingEdge1Index = i; - minSeparatingEdge2Index = j; - separatingEdge1A = edge1A; - separatingEdge1B = edge1B; - separatingEdge2A = edge2A; - separatingEdge2B = edge2B; - minEdgeVsEdgeSeparatingAxisPolyhedron2Space = separatingAxisPolyhedron2Space; + + // Compute the contact points between two faces of two convex polyhedra. + if(computePolyhedronVsPolyhedronFaceContactPoints(isMinPenetrationFaceNormalPolyhedron1, polyhedron1, polyhedron2, + polyhedron1ToPolyhedron2, polyhedron2ToPolyhedron1, minFaceIndex, + narrowPhaseInfoBatch, batchIndex, minPenetrationDepth)) { + + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; + + // The shapes are still overlapping in the previous axis (the contact manifold is not empty). + // Therefore, we can return without running the whole SAT algorithm + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; + } + + // The contact manifold is empty. Therefore, we have to run the whole SAT algorithm again + } + } + else { // If the previous separating axis (or axis with minimum penetration depth) was the cross product of two edges + + const HalfEdgeStructure::Edge& edge1 = polyhedron1->getHalfEdge(lastFrameCollisionInfo->satMinEdge1Index); + const HalfEdgeStructure::Edge& edge2 = polyhedron2->getHalfEdge(lastFrameCollisionInfo->satMinEdge2Index); + + const Vector3 edge1A = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(edge1.vertexIndex); + const Vector3 edge1B = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(polyhedron1->getHalfEdge(edge1.nextEdgeIndex).vertexIndex); + const Vector3 edge1Direction = edge1B - edge1A; + const Vector3 edge2A = polyhedron2->getVertexPosition(edge2.vertexIndex); + const Vector3 edge2B = polyhedron2->getVertexPosition(polyhedron2->getHalfEdge(edge2.nextEdgeIndex).vertexIndex); + const Vector3 edge2Direction = edge2B - edge2A; + + // If the two edges build a minkowski face (and the cross product is + // therefore a candidate for separating axis + if (testEdgesBuildMinkowskiFace(polyhedron1, edge1, polyhedron2, edge2, polyhedron1ToPolyhedron2)) { + + Vector3 separatingAxisPolyhedron2Space; + + // Compute the penetration depth along the previous axis + const Vector3 polyhedron1Centroid = polyhedron1ToPolyhedron2 * polyhedron1->getCentroid(); + decimal penetrationDepth = computeDistanceBetweenEdges(edge1A, edge2A, polyhedron1Centroid, polyhedron2->getCentroid(), + edge1Direction, edge2Direction, isShape1Triangle, separatingAxisPolyhedron2Space); + + // If the shapes were not overlapping in the previous frame and are still not + // overlapping in the current one + if (!lastFrameCollisionInfo->wasColliding && penetrationDepth <= decimal(0.0)) { + + // We have found a separating axis without running the whole SAT algorithm + continue; + } + + // If the shapes were overlapping on the previous axis and still seem to overlap in this frame + if (lastFrameCollisionInfo->wasColliding && penetrationDepth > decimal(0.0)) { + + // Compute the closest points between the two edges (in the local-space of poylhedron 2) + Vector3 closestPointPolyhedron1Edge, closestPointPolyhedron2Edge; + computeClosestPointBetweenTwoSegments(edge1A, edge1B, edge2A, edge2B, + closestPointPolyhedron1Edge, closestPointPolyhedron2Edge); + + // Here we try to project the closest point on edge1 onto the segment of edge 2 to see if + // the projected point falls onto the segment. We also try to project the closest point + // on edge 2 to see if it falls onto the segment of edge 1. If one of the point does not + // fall onto the opposite segment, it means the edges are not colliding (the contact manifold + // is empty). Therefore, we need to run the whole SAT algorithm again. + const Vector3 vec1 = closestPointPolyhedron1Edge - edge2A; + const Vector3 vec2 = closestPointPolyhedron2Edge - edge1A; + const decimal edge1LengthSquare = edge1Direction.lengthSquare(); + const decimal edge2LengthSquare = edge2Direction.lengthSquare(); + decimal t1 = vec1.dot(edge2Direction) / edge2LengthSquare; + decimal t2 = vec2.dot(edge1Direction) / edge1LengthSquare; + if (t1 >= decimal(0.0) && t1 <= decimal(1) && t2 >= decimal(0.0) && t2 <= decimal(1.0)) { + + // Compute the contact point on polyhedron 1 edge in the local-space of polyhedron 1 + Vector3 closestPointPolyhedron1EdgeLocalSpace = polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge; + + // Compute the world normal + // We use the direction from the centroid to the edge of the shape that is not a triangle + // to avoid possible degeneracies when axis direction is orthogonal to triangle normal + Vector3 normal; + if (isShape1Triangle) { + normal = polyhedron2->getCentroid() - closestPointPolyhedron2Edge; + } + else { + normal = polyhedron1ToPolyhedron2.getOrientation() * ((polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge) - polyhedron1->getCentroid()); + } + + //Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * minEdgeVsEdgeSeparatingAxisPolyhedron2Space; + Vector3 normalWorld = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * normal.getUnit(); + + // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle + TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfoBatch.collisionShapes1[batchIndex], narrowPhaseInfoBatch.collisionShapes2[batchIndex], + closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge, + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex], narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex], + penetrationDepth, normalWorld); + + // Create the contact point + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, penetrationDepth, + closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge); + + // The shapes are overlapping on the previous axis (the contact manifold is not empty). Therefore + // we return without running the whole SAT algorithm + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; + } + + // The contact manifold is empty. Therefore, we have to run the whole SAT algorithm again + } } } } - } - // Here we know the shapes are overlapping on a given minimum separating axis. - // Now, we will clip the shapes along this axis to find the contact points + // Test all the face normals of the polyhedron 1 for separating axis + uint faceIndex; + decimal penetrationDepth = testFacesDirectionPolyhedronVsPolyhedron(polyhedron1, polyhedron2, polyhedron1ToPolyhedron2, faceIndex); + if (penetrationDepth <= decimal(0.0)) { - assert(minPenetrationDepth > decimal(0.0)); - assert((isMinPenetrationFaceNormal && minFaceIndex >= 0) || !isMinPenetrationFaceNormal); + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = true; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = false; + lastFrameCollisionInfo->satMinAxisFaceIndex = faceIndex; - // If the minimum separating axis is a face normal - if (isMinPenetrationFaceNormal) { - - if (reportContacts) { - - // Compute the contact points between two faces of two convex polyhedra. - bool contactsFound = computePolyhedronVsPolyhedronFaceContactPoints(isMinPenetrationFaceNormalPolyhedron1, polyhedron1, - polyhedron2, polyhedron1ToPolyhedron2, polyhedron2ToPolyhedron1, - minFaceIndex, narrowPhaseInfo, minPenetrationDepth); - - // There should be clipping points here. If it is not the case, it might be - // because of a numerical issue - if (!contactsFound) { - - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; - - // Return no collision - return false; - } + // We have found a separating axis + continue; + } + if (penetrationDepth < minPenetrationDepth - SAME_SEPARATING_AXIS_BIAS) { + isMinPenetrationFaceNormal = true; + minPenetrationDepth = penetrationDepth; + minFaceIndex = faceIndex; + isMinPenetrationFaceNormalPolyhedron1 = true; } - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; - lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; - } - else { // If we have an edge vs edge contact + // Test all the face normals of the polyhedron 2 for separating axis + penetrationDepth = testFacesDirectionPolyhedronVsPolyhedron(polyhedron2, polyhedron1, polyhedron2ToPolyhedron1, faceIndex); + if (penetrationDepth <= decimal(0.0)) { - if (reportContacts) { + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = false; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = true; + lastFrameCollisionInfo->satMinAxisFaceIndex = faceIndex; - // Compute the closest points between the two edges (in the local-space of poylhedron 2) - Vector3 closestPointPolyhedron1Edge, closestPointPolyhedron2Edge; - computeClosestPointBetweenTwoSegments(separatingEdge1A, separatingEdge1B, separatingEdge2A, separatingEdge2B, - closestPointPolyhedron1Edge, closestPointPolyhedron2Edge); - - // Compute the contact point on polyhedron 1 edge in the local-space of polyhedron 1 - Vector3 closestPointPolyhedron1EdgeLocalSpace = polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge; - - // Compute the world normal - // We use the direction from the centroid to the edge of the shape that is not a triangle - // to avoid possible degeneracies when axis direction is orthogonal to triangle normal - Vector3 normal; - if (isShape1Triangle) { - normal = polyhedron2->getCentroid() - closestPointPolyhedron2Edge; - } - else { - normal = polyhedron1ToPolyhedron2.getOrientation() * ((polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge) - polyhedron1->getCentroid()); - } - //Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * minEdgeVsEdgeSeparatingAxisPolyhedron2Space; - Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * normal.getUnit(); - - // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfo->collisionShape1, narrowPhaseInfo->collisionShape2, - closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge, - narrowPhaseInfo->shape1ToWorldTransform, narrowPhaseInfo->shape2ToWorldTransform, - minPenetrationDepth, normalWorld); - - // Create the contact point - narrowPhaseInfo->addContactPoint(normalWorld, minPenetrationDepth, - closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge); + // We have found a separating axis + continue; + } + if (penetrationDepth < minPenetrationDepth - SAME_SEPARATING_AXIS_BIAS) { + isMinPenetrationFaceNormal = true; + minPenetrationDepth = penetrationDepth; + minFaceIndex = faceIndex; + isMinPenetrationFaceNormalPolyhedron1 = false; } - lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = false; - lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = false; - lastFrameCollisionInfo->satMinEdge1Index = minSeparatingEdge1Index; - lastFrameCollisionInfo->satMinEdge2Index = minSeparatingEdge2Index; - } + bool separatingAxisFound = false; - return true; + // Test the cross products of edges of polyhedron 1 with edges of polyhedron 2 for separating axis + for (uint i=0; i < polyhedron1->getNbHalfEdges(); i += 2) { + + // Get an edge of polyhedron 1 + const HalfEdgeStructure::Edge& edge1 = polyhedron1->getHalfEdge(i); + + const Vector3 edge1A = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(edge1.vertexIndex); + const Vector3 edge1B = polyhedron1ToPolyhedron2 * polyhedron1->getVertexPosition(polyhedron1->getHalfEdge(edge1.nextEdgeIndex).vertexIndex); + const Vector3 edge1Direction = edge1B - edge1A; + + for (uint j=0; j < polyhedron2->getNbHalfEdges(); j += 2) { + + // Get an edge of polyhedron 2 + const HalfEdgeStructure::Edge& edge2 = polyhedron2->getHalfEdge(j); + + const Vector3 edge2A = polyhedron2->getVertexPosition(edge2.vertexIndex); + const Vector3 edge2B = polyhedron2->getVertexPosition(polyhedron2->getHalfEdge(edge2.nextEdgeIndex).vertexIndex); + const Vector3 edge2Direction = edge2B - edge2A; + + // If the two edges build a minkowski face (and the cross product is + // therefore a candidate for separating axis + if (testEdgesBuildMinkowskiFace(polyhedron1, edge1, polyhedron2, edge2, polyhedron1ToPolyhedron2)) { + + Vector3 separatingAxisPolyhedron2Space; + + // Compute the penetration depth + const Vector3 polyhedron1Centroid = polyhedron1ToPolyhedron2 * polyhedron1->getCentroid(); + decimal penetrationDepth = computeDistanceBetweenEdges(edge1A, edge2A, polyhedron1Centroid, polyhedron2->getCentroid(), + edge1Direction, edge2Direction, isShape1Triangle, separatingAxisPolyhedron2Space); + + if (penetrationDepth <= decimal(0.0)) { + + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = false; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = false; + lastFrameCollisionInfo->satMinEdge1Index = i; + lastFrameCollisionInfo->satMinEdge2Index = j; + + // We have found a separating axis + separatingAxisFound = true; + break; + } + + if (penetrationDepth < minPenetrationDepth - SAME_SEPARATING_AXIS_BIAS) { + + minPenetrationDepth = penetrationDepth; + isMinPenetrationFaceNormalPolyhedron1 = false; + isMinPenetrationFaceNormal = false; + minSeparatingEdge1Index = i; + minSeparatingEdge2Index = j; + separatingEdge1A = edge1A; + separatingEdge1B = edge1B; + separatingEdge2A = edge2A; + separatingEdge2B = edge2B; + minEdgeVsEdgeSeparatingAxisPolyhedron2Space = separatingAxisPolyhedron2Space; + } + } + } + + if (separatingAxisFound) { + break; + } + } + + if (separatingAxisFound) { + continue; + } + + // Here we know the shapes are overlapping on a given minimum separating axis. + // Now, we will clip the shapes along this axis to find the contact points + + assert(minPenetrationDepth > decimal(0.0)); + assert((isMinPenetrationFaceNormal && minFaceIndex >= 0) || !isMinPenetrationFaceNormal); + + // If the minimum separating axis is a face normal + if (isMinPenetrationFaceNormal) { + + if (reportContacts) { + + // Compute the contact points between two faces of two convex polyhedra. + bool contactsFound = computePolyhedronVsPolyhedronFaceContactPoints(isMinPenetrationFaceNormalPolyhedron1, polyhedron1, + polyhedron2, polyhedron1ToPolyhedron2, polyhedron2ToPolyhedron1, + minFaceIndex, narrowPhaseInfoBatch, batchIndex, minPenetrationDepth); + + // There should be clipping points here. If it is not the case, it might be + // because of a numerical issue + if (!contactsFound) { + + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; + + // Return no collision + continue; + } + } + + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = !isMinPenetrationFaceNormalPolyhedron1; + lastFrameCollisionInfo->satMinAxisFaceIndex = minFaceIndex; + } + else { // If we have an edge vs edge contact + + if (reportContacts) { + + // Compute the closest points between the two edges (in the local-space of poylhedron 2) + Vector3 closestPointPolyhedron1Edge, closestPointPolyhedron2Edge; + computeClosestPointBetweenTwoSegments(separatingEdge1A, separatingEdge1B, separatingEdge2A, separatingEdge2B, + closestPointPolyhedron1Edge, closestPointPolyhedron2Edge); + + // Compute the contact point on polyhedron 1 edge in the local-space of polyhedron 1 + Vector3 closestPointPolyhedron1EdgeLocalSpace = polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge; + + // Compute the world normal + // We use the direction from the centroid to the edge of the shape that is not a triangle + // to avoid possible degeneracies when axis direction is orthogonal to triangle normal + Vector3 normal; + if (isShape1Triangle) { + normal = polyhedron2->getCentroid() - closestPointPolyhedron2Edge; + } + else { + normal = polyhedron1ToPolyhedron2.getOrientation() * ((polyhedron2ToPolyhedron1 * closestPointPolyhedron1Edge) - polyhedron1->getCentroid()); + } + //Vector3 normalWorld = narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * minEdgeVsEdgeSeparatingAxisPolyhedron2Space; + Vector3 normalWorld = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * normal.getUnit(); + + // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle + TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfoBatch.collisionShapes1[batchIndex], narrowPhaseInfoBatch.collisionShapes2[batchIndex], + closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge, + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex], narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex], + minPenetrationDepth, normalWorld); + + // Create the contact point + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, minPenetrationDepth, + closestPointPolyhedron1EdgeLocalSpace, closestPointPolyhedron2Edge); + } + + lastFrameCollisionInfo->satIsAxisFacePolyhedron1 = false; + lastFrameCollisionInfo->satIsAxisFacePolyhedron2 = false; + lastFrameCollisionInfo->satMinEdge1Index = minSeparatingEdge1Index; + lastFrameCollisionInfo->satMinEdge2Index = minSeparatingEdge2Index; + } + + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + } } // Compute the contact points between two faces of two convex polyhedra. @@ -835,7 +863,8 @@ bool SATAlgorithm::testCollisionConvexPolyhedronVsConvexPolyhedron(NarrowPhaseIn bool SATAlgorithm::computePolyhedronVsPolyhedronFaceContactPoints(bool isMinPenetrationFaceNormalPolyhedron1, const ConvexPolyhedronShape* polyhedron1, const ConvexPolyhedronShape* polyhedron2, const Transform& polyhedron1ToPolyhedron2, const Transform& polyhedron2ToPolyhedron1, - uint minFaceIndex, NarrowPhaseInfo* narrowPhaseInfo, decimal minPenetrationDepth) const { + uint minFaceIndex, NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex, + decimal minPenetrationDepth) const { RP3D_PROFILE("SATAlgorithm::computePolyhedronVsPolyhedronFaceContactPoints", mProfiler); @@ -850,8 +879,8 @@ bool SATAlgorithm::computePolyhedronVsPolyhedronFaceContactPoints(bool isMinPene const Vector3 axisIncidentSpace = referenceToIncidentTransform.getOrientation() * axisReferenceSpace; // Compute the world normal - Vector3 normalWorld = isMinPenetrationFaceNormalPolyhedron1 ? narrowPhaseInfo->shape1ToWorldTransform.getOrientation() * axisReferenceSpace : - -(narrowPhaseInfo->shape2ToWorldTransform.getOrientation() * axisReferenceSpace); + Vector3 normalWorld = isMinPenetrationFaceNormalPolyhedron1 ? narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex].getOrientation() * axisReferenceSpace : + -(narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex].getOrientation() * axisReferenceSpace); // Get the reference face const HalfEdgeStructure::Face& referenceFace = referencePolyhedron->getFace(minFaceIndex); @@ -930,14 +959,14 @@ bool SATAlgorithm::computePolyhedronVsPolyhedronFaceContactPoints(bool isMinPene Vector3 contactPointReferencePolyhedron = projectPointOntoPlane(clipPolygonVertices[i], axisReferenceSpace, referenceFaceVertex); // Compute smooth triangle mesh contact if one of the two collision shapes is a triangle - TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfo->collisionShape1, narrowPhaseInfo->collisionShape2, + TriangleShape::computeSmoothTriangleMeshContact(narrowPhaseInfoBatch.collisionShapes1[batchIndex], narrowPhaseInfoBatch.collisionShapes2[batchIndex], isMinPenetrationFaceNormalPolyhedron1 ? contactPointReferencePolyhedron : contactPointIncidentPolyhedron, isMinPenetrationFaceNormalPolyhedron1 ? contactPointIncidentPolyhedron : contactPointReferencePolyhedron, - narrowPhaseInfo->shape1ToWorldTransform, narrowPhaseInfo->shape2ToWorldTransform, + narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex], narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex], penetrationDepth, outWorldNormal); // Create a new contact point - narrowPhaseInfo->addContactPoint(outWorldNormal, penetrationDepth, + narrowPhaseInfoBatch.addContactPoint(batchIndex, outWorldNormal, penetrationDepth, isMinPenetrationFaceNormalPolyhedron1 ? contactPointReferencePolyhedron : contactPointIncidentPolyhedron, isMinPenetrationFaceNormalPolyhedron1 ? contactPointIncidentPolyhedron : contactPointReferencePolyhedron); } diff --git a/src/collision/narrowphase/SAT/SATAlgorithm.h b/src/collision/narrowphase/SAT/SATAlgorithm.h index 5be84220..82bd680c 100644 --- a/src/collision/narrowphase/SAT/SATAlgorithm.h +++ b/src/collision/narrowphase/SAT/SATAlgorithm.h @@ -37,7 +37,7 @@ namespace reactphysics3d { class CapsuleShape; class SphereShape; class ContactManifoldInfo; -struct NarrowPhaseInfo; +struct NarrowPhaseInfoBatch; class ConvexPolyhedronShape; class MemoryAllocator; class Profiler; @@ -118,7 +118,7 @@ class SATAlgorithm { bool computePolyhedronVsPolyhedronFaceContactPoints(bool isMinPenetrationFaceNormalPolyhedron1, const ConvexPolyhedronShape* polyhedron1, const ConvexPolyhedronShape* polyhedron2, const Transform& polyhedron1ToPolyhedron2, const Transform& polyhedron2ToPolyhedron1, uint minFaceIndex, - NarrowPhaseInfo* narrowPhaseInfo, decimal minPenetrationDepth) const; + NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex, decimal minPenetrationDepth) const; public : @@ -138,24 +138,26 @@ class SATAlgorithm { SATAlgorithm& operator=(const SATAlgorithm& algorithm) = delete; /// Test collision between a sphere and a convex mesh - bool testCollisionSphereVsConvexPolyhedron(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) const; + void testCollisionSphereVsConvexPolyhedron(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, + uint batchStartIndex, uint batchNbItems, + bool reportContacts) const; /// Test collision between a capsule and a convex mesh - bool testCollisionCapsuleVsConvexPolyhedron(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) const; + bool testCollisionCapsuleVsConvexPolyhedron(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex, bool reportContacts) const; /// Compute the two contact points between a polyhedron and a capsule when the separating axis is a face normal of the polyhedron bool computeCapsulePolyhedronFaceContactPoints(uint referenceFaceIndex, decimal capsuleRadius, const ConvexPolyhedronShape* polyhedron, decimal penetrationDepth, const Transform& polyhedronToCapsuleTransform, Vector3& normalWorld, const Vector3& separatingAxisCapsuleSpace, const Vector3& capsuleSegAPolyhedronSpace, const Vector3& capsuleSegBPolyhedronSpace, - NarrowPhaseInfo* narrowPhaseInfo, bool isCapsuleShape1) const; + NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex, bool isCapsuleShape1) const; // This method returns true if an edge of a polyhedron and a capsule forms a face of the Minkowski Difference bool isMinkowskiFaceCapsuleVsEdge(const Vector3& capsuleSegment, const Vector3& edgeAdjacentFace1Normal, const Vector3& edgeAdjacentFace2Normal) const; /// Test collision between two convex meshes - bool testCollisionConvexPolyhedronVsConvexPolyhedron(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts) const; + void testCollisionConvexPolyhedronVsConvexPolyhedron(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, uint batchNbItems, bool reportContacts) const; #ifdef IS_PROFILING_ACTIVE diff --git a/src/collision/narrowphase/SphereVsCapsuleAlgorithm.cpp b/src/collision/narrowphase/SphereVsCapsuleAlgorithm.cpp index d1a00f66..c2bf193a 100755 --- a/src/collision/narrowphase/SphereVsCapsuleAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsCapsuleAlgorithm.cpp @@ -27,7 +27,7 @@ #include "SphereVsCapsuleAlgorithm.h" #include "collision/shapes/SphereShape.h" #include "collision/shapes/CapsuleShape.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; @@ -35,105 +35,110 @@ using namespace reactphysics3d; // Compute the narrow-phase collision detection between a sphere and a capsule // This technique is based on the "Robust Contact Creation for Physics Simulations" presentation // by Dirk Gregorius. -bool SphereVsCapsuleAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, - MemoryAllocator& memoryAllocator) { +void SphereVsCapsuleAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, uint batchNbItems, + bool reportContacts, MemoryAllocator& memoryAllocator) { - bool isSphereShape1 = narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::SPHERE; + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - assert(isSphereShape1 || narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CAPSULE); + bool isSphereShape1 = narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::SPHERE; - // Get the collision shapes - const SphereShape* sphereShape = static_cast(isSphereShape1 ? narrowPhaseInfo->collisionShape1 : narrowPhaseInfo->collisionShape2); - const CapsuleShape* capsuleShape = static_cast(isSphereShape1 ? narrowPhaseInfo->collisionShape2 : narrowPhaseInfo->collisionShape1); + assert(!narrowPhaseInfoBatch.isColliding[batchIndex]); + assert(isSphereShape1 || narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CAPSULE); - // Get the transform from sphere local-space to capsule local-space - const Transform& sphereToWorldTransform = isSphereShape1 ? narrowPhaseInfo->shape1ToWorldTransform : narrowPhaseInfo->shape2ToWorldTransform; - const Transform& capsuleToWorldTransform = isSphereShape1 ? narrowPhaseInfo->shape2ToWorldTransform : narrowPhaseInfo->shape1ToWorldTransform; - const Transform worldToCapsuleTransform = capsuleToWorldTransform.getInverse(); - const Transform sphereToCapsuleSpaceTransform = worldToCapsuleTransform * sphereToWorldTransform; + // Get the collision shapes + const SphereShape* sphereShape = static_cast(isSphereShape1 ? narrowPhaseInfoBatch.collisionShapes1[batchIndex] : narrowPhaseInfoBatch.collisionShapes2[batchIndex]); + const CapsuleShape* capsuleShape = static_cast(isSphereShape1 ? narrowPhaseInfoBatch.collisionShapes2[batchIndex] : narrowPhaseInfoBatch.collisionShapes1[batchIndex]); - // Transform the center of the sphere into the local-space of the capsule shape - const Vector3 sphereCenter = sphereToCapsuleSpaceTransform.getPosition(); + // Get the transform from sphere local-space to capsule local-space + const Transform& sphereToWorldTransform = isSphereShape1 ? narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; + const Transform& capsuleToWorldTransform = isSphereShape1 ? narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex] : narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; + const Transform worldToCapsuleTransform = capsuleToWorldTransform.getInverse(); + const Transform sphereToCapsuleSpaceTransform = worldToCapsuleTransform * sphereToWorldTransform; - // Compute the end-points of the inner segment of the capsule - const decimal capsuleHalfHeight = capsuleShape->getHeight() * decimal(0.5); - const Vector3 capsuleSegA(0, -capsuleHalfHeight, 0); - const Vector3 capsuleSegB(0, capsuleHalfHeight, 0); + // Transform the center of the sphere into the local-space of the capsule shape + const Vector3 sphereCenter = sphereToCapsuleSpaceTransform.getPosition(); - // Compute the point on the inner capsule segment that is the closes to center of sphere - const Vector3 closestPointOnSegment = computeClosestPointOnSegment(capsuleSegA, capsuleSegB, sphereCenter); + // Compute the end-points of the inner segment of the capsule + const decimal capsuleHalfHeight = capsuleShape->getHeight() * decimal(0.5); + const Vector3 capsuleSegA(0, -capsuleHalfHeight, 0); + const Vector3 capsuleSegB(0, capsuleHalfHeight, 0); - // Compute the distance between the sphere center and the closest point on the segment - Vector3 sphereCenterToSegment = (closestPointOnSegment - sphereCenter); - const decimal sphereSegmentDistanceSquare = sphereCenterToSegment.lengthSquare(); + // Compute the point on the inner capsule segment that is the closes to center of sphere + const Vector3 closestPointOnSegment = computeClosestPointOnSegment(capsuleSegA, capsuleSegB, sphereCenter); - // Compute the sum of the radius of the sphere and the capsule (virtual sphere) - decimal sumRadius = sphereShape->getRadius() + capsuleShape->getRadius(); - - // If the collision shapes overlap - if (sphereSegmentDistanceSquare < sumRadius * sumRadius) { + // Compute the distance between the sphere center and the closest point on the segment + Vector3 sphereCenterToSegment = (closestPointOnSegment - sphereCenter); + const decimal sphereSegmentDistanceSquare = sphereCenterToSegment.lengthSquare(); - decimal penetrationDepth; - Vector3 normalWorld; - Vector3 contactPointSphereLocal; - Vector3 contactPointCapsuleLocal; + // Compute the sum of the radius of the sphere and the capsule (virtual sphere) + decimal sumRadius = sphereShape->getRadius() + capsuleShape->getRadius(); - if (reportContacts) { + // If the collision shapes overlap + if (sphereSegmentDistanceSquare < sumRadius * sumRadius) { - // If the sphere center is not on the capsule inner segment - if (sphereSegmentDistanceSquare > MACHINE_EPSILON) { + decimal penetrationDepth; + Vector3 normalWorld; + Vector3 contactPointSphereLocal; + Vector3 contactPointCapsuleLocal; - decimal sphereSegmentDistance = std::sqrt(sphereSegmentDistanceSquare); - sphereCenterToSegment /= sphereSegmentDistance; + if (reportContacts) { - contactPointSphereLocal = sphereToCapsuleSpaceTransform.getInverse() * (sphereCenter + sphereCenterToSegment * sphereShape->getRadius()); - contactPointCapsuleLocal = closestPointOnSegment - sphereCenterToSegment * capsuleShape->getRadius(); + // If the sphere center is not on the capsule inner segment + if (sphereSegmentDistanceSquare > MACHINE_EPSILON) { - normalWorld = capsuleToWorldTransform.getOrientation() * sphereCenterToSegment; + decimal sphereSegmentDistance = std::sqrt(sphereSegmentDistanceSquare); + sphereCenterToSegment /= sphereSegmentDistance; - penetrationDepth = sumRadius - sphereSegmentDistance; + contactPointSphereLocal = sphereToCapsuleSpaceTransform.getInverse() * (sphereCenter + sphereCenterToSegment * sphereShape->getRadius()); + contactPointCapsuleLocal = closestPointOnSegment - sphereCenterToSegment * capsuleShape->getRadius(); - if (!isSphereShape1) { - normalWorld = -normalWorld; - } - } - else { // If the sphere center is on the capsule inner segment (degenerate case) + normalWorld = capsuleToWorldTransform.getOrientation() * sphereCenterToSegment; - // We take any direction that is orthogonal to the inner capsule segment as a contact normal + penetrationDepth = sumRadius - sphereSegmentDistance; - // Capsule inner segment - Vector3 capsuleSegment = (capsuleSegB - capsuleSegA).getUnit(); + if (!isSphereShape1) { + normalWorld = -normalWorld; + } + } + else { // If the sphere center is on the capsule inner segment (degenerate case) - Vector3 vec1(1, 0, 0); - Vector3 vec2(0, 1, 0); + // We take any direction that is orthogonal to the inner capsule segment as a contact normal - // Get the vectors (among vec1 and vec2) that is the most orthogonal to the capsule inner segment (smallest absolute dot product) - decimal cosA1 = std::abs(capsuleSegment.x); // abs(vec1.dot(seg2)) - decimal cosA2 = std::abs(capsuleSegment.y); // abs(vec2.dot(seg2)) + // Capsule inner segment + Vector3 capsuleSegment = (capsuleSegB - capsuleSegA).getUnit(); - penetrationDepth = sumRadius; + Vector3 vec1(1, 0, 0); + Vector3 vec2(0, 1, 0); - // We choose as a contact normal, any direction that is perpendicular to the inner capsule segment - Vector3 normalCapsuleSpace = cosA1 < cosA2 ? capsuleSegment.cross(vec1) : capsuleSegment.cross(vec2); - normalWorld = capsuleToWorldTransform.getOrientation() * normalCapsuleSpace; + // Get the vectors (among vec1 and vec2) that is the most orthogonal to the capsule inner segment (smallest absolute dot product) + decimal cosA1 = std::abs(capsuleSegment.x); // abs(vec1.dot(seg2)) + decimal cosA2 = std::abs(capsuleSegment.y); // abs(vec2.dot(seg2)) - // Compute the two local contact points - contactPointSphereLocal = sphereToCapsuleSpaceTransform.getInverse() * (sphereCenter + normalCapsuleSpace * sphereShape->getRadius()); - contactPointCapsuleLocal = sphereCenter - normalCapsuleSpace * capsuleShape->getRadius(); - } + penetrationDepth = sumRadius; - if (penetrationDepth <= decimal(0.0)) { - return false; + // We choose as a contact normal, any direction that is perpendicular to the inner capsule segment + Vector3 normalCapsuleSpace = cosA1 < cosA2 ? capsuleSegment.cross(vec1) : capsuleSegment.cross(vec2); + normalWorld = capsuleToWorldTransform.getOrientation() * normalCapsuleSpace; + + // Compute the two local contact points + contactPointSphereLocal = sphereToCapsuleSpaceTransform.getInverse() * (sphereCenter + normalCapsuleSpace * sphereShape->getRadius()); + contactPointCapsuleLocal = sphereCenter - normalCapsuleSpace * capsuleShape->getRadius(); + } + + if (penetrationDepth <= decimal(0.0)) { + + // No collision + continue; + } + + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normalWorld, penetrationDepth, + isSphereShape1 ? contactPointSphereLocal : contactPointCapsuleLocal, + isSphereShape1 ? contactPointCapsuleLocal : contactPointSphereLocal); } - // Create the contact info object - narrowPhaseInfo->addContactPoint(normalWorld, penetrationDepth, - isSphereShape1 ? contactPointSphereLocal : contactPointCapsuleLocal, - isSphereShape1 ? contactPointCapsuleLocal : contactPointSphereLocal); + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; } - - return true; } - - return false; } diff --git a/src/collision/narrowphase/SphereVsCapsuleAlgorithm.h b/src/collision/narrowphase/SphereVsCapsuleAlgorithm.h index 97bfc1ce..85abc219 100644 --- a/src/collision/narrowphase/SphereVsCapsuleAlgorithm.h +++ b/src/collision/narrowphase/SphereVsCapsuleAlgorithm.h @@ -66,7 +66,9 @@ class SphereVsCapsuleAlgorithm : public NarrowPhaseAlgorithm { SphereVsCapsuleAlgorithm& operator=(const SphereVsCapsuleAlgorithm& algorithm) = delete; /// Compute the narrow-phase collision detection between a sphere and a capsule - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, MemoryAllocator& memoryAllocator) override; + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, + MemoryAllocator& memoryAllocator) override; }; } diff --git a/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.cpp b/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.cpp index f6f4ab0f..fac9baf9 100644 --- a/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.cpp @@ -27,7 +27,7 @@ #include "SphereVsConvexPolyhedronAlgorithm.h" #include "GJK/GJKAlgorithm.h" #include "SAT/SATAlgorithm.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; @@ -35,57 +35,66 @@ using namespace reactphysics3d; // Compute the narrow-phase collision detection between a sphere and a convex polyhedron // This technique is based on the "Robust Contact Creation for Physics Simulations" presentation // by Dirk Gregorius. -bool SphereVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, - MemoryAllocator& memoryAllocator) { - - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::CONVEX_POLYHEDRON || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::CONVEX_POLYHEDRON); - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::SPHERE || - narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::SPHERE); - - // Get the last frame collision info - LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfo->getLastFrameCollisionInfo(); +void SphereVsConvexPolyhedronAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, uint batchNbItems, + bool reportContacts, MemoryAllocator& memoryAllocator) { // First, we run the GJK algorithm GJKAlgorithm gjkAlgorithm; #ifdef IS_PROFILING_ACTIVE - gjkAlgorithm.setProfiler(mProfiler); + gjkAlgorithm.setProfiler(mProfiler); #endif - GJKAlgorithm::GJKResult result = gjkAlgorithm.testCollision(narrowPhaseInfo, reportContacts); + List gjkResults(memoryAllocator); + gjkAlgorithm.testCollision(narrowPhaseInfoBatch, batchStartIndex, batchNbItems, gjkResults, reportContacts); - lastFrameCollisionInfo->wasUsingGJK = true; - lastFrameCollisionInfo->wasUsingSAT = false; + // For each item in the batch + uint resultIndex=0; + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - // If we have found a contact point inside the margins (shallow penetration) - if (result == GJKAlgorithm::GJKResult::COLLIDE_IN_MARGIN) { + assert(!narrowPhaseInfoBatch.isColliding[batchIndex]); - // Return true - return true; - } + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::CONVEX_POLYHEDRON); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::SPHERE || + narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::SPHERE); - // If we have overlap even without the margins (deep penetration) - if (result == GJKAlgorithm::GJKResult::INTERPENETRATE) { + // Get the last frame collision info + LastFrameCollisionInfo* lastFrameCollisionInfo = narrowPhaseInfoBatch.getLastFrameCollisionInfo(batchIndex); - // Run the SAT algorithm to find the separating axis and compute contact point - SATAlgorithm satAlgorithm(memoryAllocator); + lastFrameCollisionInfo->wasUsingGJK = true; + lastFrameCollisionInfo->wasUsingSAT = false; + + // If we have found a contact point inside the margins (shallow penetration) + if (gjkResults[resultIndex] == GJKAlgorithm::GJKResult::COLLIDE_IN_MARGIN) { + + // Return true + narrowPhaseInfoBatch.isColliding[batchIndex] = true; + continue; + } + + // If we have overlap even without the margins (deep penetration) + if (gjkResults[resultIndex] == GJKAlgorithm::GJKResult::INTERPENETRATE) { + + // Run the SAT algorithm to find the separating axis and compute contact point + SATAlgorithm satAlgorithm(memoryAllocator); #ifdef IS_PROFILING_ACTIVE - satAlgorithm.setProfiler(mProfiler); + satAlgorithm.setProfiler(mProfiler); #endif - bool isColliding = satAlgorithm.testCollisionSphereVsConvexPolyhedron(narrowPhaseInfo, reportContacts); + satAlgorithm.testCollisionSphereVsConvexPolyhedron(narrowPhaseInfoBatch, batchIndex, 1, reportContacts); - lastFrameCollisionInfo->wasUsingGJK = false; - lastFrameCollisionInfo->wasUsingSAT = true; + lastFrameCollisionInfo->wasUsingGJK = false; + lastFrameCollisionInfo->wasUsingSAT = true; - return isColliding; + continue; + } + + resultIndex++; } - - return false; } diff --git a/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.h b/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.h index c4782f9a..378255e0 100644 --- a/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.h +++ b/src/collision/narrowphase/SphereVsConvexPolyhedronAlgorithm.h @@ -71,7 +71,9 @@ class SphereVsConvexPolyhedronAlgorithm : public NarrowPhaseAlgorithm { SphereVsConvexPolyhedronAlgorithm& operator=(const SphereVsConvexPolyhedronAlgorithm& algorithm) = delete; /// Compute the narrow-phase collision detection between a sphere and a convex polyhedron - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, MemoryAllocator& memoryAllocator) override; + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, + MemoryAllocator& memoryAllocator) override; }; } diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp index f187bd1d..2c8704c7 100755 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.cpp @@ -26,66 +26,70 @@ // Libraries #include "SphereVsSphereAlgorithm.h" #include "collision/shapes/SphereShape.h" -#include "collision/NarrowPhaseInfo.h" +#include "collision/NarrowPhaseInfoBatch.h" // We want to use the ReactPhysics3D namespace using namespace reactphysics3d; -bool SphereVsSphereAlgorithm::testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, - MemoryAllocator& memoryAllocator) { - - assert(narrowPhaseInfo->collisionShape1->getType() == CollisionShapeType::SPHERE); - assert(narrowPhaseInfo->collisionShape2->getType() == CollisionShapeType::SPHERE); +void SphereVsSphereAlgorithm::testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, uint batchNbItems, + bool reportContacts, MemoryAllocator& memoryAllocator) { - // Get the sphere collision shapes - const SphereShape* sphereShape1 = static_cast(narrowPhaseInfo->collisionShape1); - const SphereShape* sphereShape2 = static_cast(narrowPhaseInfo->collisionShape2); + // For each item in the batch + for (uint batchIndex = batchStartIndex; batchIndex < batchStartIndex + batchNbItems; batchIndex++) { - // Get the local-space to world-space transforms - const Transform& transform1 = narrowPhaseInfo->shape1ToWorldTransform; - const Transform& transform2 = narrowPhaseInfo->shape2ToWorldTransform; + assert(!narrowPhaseInfoBatch.isColliding[batchIndex]); - // Compute the distance between the centers - Vector3 vectorBetweenCenters = transform2.getPosition() - transform1.getPosition(); - decimal squaredDistanceBetweenCenters = vectorBetweenCenters.lengthSquare(); + assert(narrowPhaseInfoBatch.collisionShapes1[batchIndex]->getType() == CollisionShapeType::SPHERE); + assert(narrowPhaseInfoBatch.collisionShapes2[batchIndex]->getType() == CollisionShapeType::SPHERE); - // Compute the sum of the radius - decimal sumRadius = sphereShape1->getRadius() + sphereShape2->getRadius(); - - // If the sphere collision shapes intersect - if (squaredDistanceBetweenCenters < sumRadius * sumRadius) { + // Get the sphere collision shapes + const SphereShape* sphereShape1 = static_cast(narrowPhaseInfoBatch.collisionShapes1[batchIndex]); + const SphereShape* sphereShape2 = static_cast(narrowPhaseInfoBatch.collisionShapes2[batchIndex]); - if (reportContacts) { + // Get the local-space to world-space transforms + const Transform& transform1 = narrowPhaseInfoBatch.shape1ToWorldTransforms[batchIndex]; + const Transform& transform2 = narrowPhaseInfoBatch.shape2ToWorldTransforms[batchIndex]; - Vector3 centerSphere2InBody1LocalSpace = transform1.getInverse() * transform2.getPosition(); - Vector3 centerSphere1InBody2LocalSpace = transform2.getInverse() * transform1.getPosition(); - decimal penetrationDepth = sumRadius - std::sqrt(squaredDistanceBetweenCenters); - Vector3 intersectionOnBody1; - Vector3 intersectionOnBody2; - Vector3 normal; + // Compute the distance between the centers + Vector3 vectorBetweenCenters = transform2.getPosition() - transform1.getPosition(); + decimal squaredDistanceBetweenCenters = vectorBetweenCenters.lengthSquare(); - // If the two sphere centers are not at the same position - if (squaredDistanceBetweenCenters > MACHINE_EPSILON) { + // Compute the sum of the radius + decimal sumRadius = sphereShape1->getRadius() + sphereShape2->getRadius(); - intersectionOnBody1 = sphereShape1->getRadius() * centerSphere2InBody1LocalSpace.getUnit(); - intersectionOnBody2 = sphereShape2->getRadius() * centerSphere1InBody2LocalSpace.getUnit(); - normal = vectorBetweenCenters.getUnit(); - } - else { // If the sphere centers are at the same position (degenerate case) + // If the sphere collision shapes intersect + if (squaredDistanceBetweenCenters < sumRadius * sumRadius) { - // Take any contact normal direction - normal.setAllValues(0, 1, 0); + if (reportContacts) { - intersectionOnBody1 = sphereShape1->getRadius() * (transform1.getInverse().getOrientation() * normal); - intersectionOnBody2 = sphereShape2->getRadius() * (transform2.getInverse().getOrientation() * normal); - } - - // Create the contact info object - narrowPhaseInfo->addContactPoint(normal, penetrationDepth, intersectionOnBody1, intersectionOnBody2); + Vector3 centerSphere2InBody1LocalSpace = transform1.getInverse() * transform2.getPosition(); + Vector3 centerSphere1InBody2LocalSpace = transform2.getInverse() * transform1.getPosition(); + decimal penetrationDepth = sumRadius - std::sqrt(squaredDistanceBetweenCenters); + Vector3 intersectionOnBody1; + Vector3 intersectionOnBody2; + Vector3 normal; + + // If the two sphere centers are not at the same position + if (squaredDistanceBetweenCenters > MACHINE_EPSILON) { + + intersectionOnBody1 = sphereShape1->getRadius() * centerSphere2InBody1LocalSpace.getUnit(); + intersectionOnBody2 = sphereShape2->getRadius() * centerSphere1InBody2LocalSpace.getUnit(); + normal = vectorBetweenCenters.getUnit(); + } + else { // If the sphere centers are at the same position (degenerate case) + + // Take any contact normal direction + normal.setAllValues(0, 1, 0); + + intersectionOnBody1 = sphereShape1->getRadius() * (transform1.getInverse().getOrientation() * normal); + intersectionOnBody2 = sphereShape2->getRadius() * (transform2.getInverse().getOrientation() * normal); + } + + // Create the contact info object + narrowPhaseInfoBatch.addContactPoint(batchIndex, normal, penetrationDepth, intersectionOnBody1, intersectionOnBody2); + } + + narrowPhaseInfoBatch.isColliding[batchIndex] = true; } - - return true; } - - return false; } diff --git a/src/collision/narrowphase/SphereVsSphereAlgorithm.h b/src/collision/narrowphase/SphereVsSphereAlgorithm.h index 3f1384ce..d5c7128f 100644 --- a/src/collision/narrowphase/SphereVsSphereAlgorithm.h +++ b/src/collision/narrowphase/SphereVsSphereAlgorithm.h @@ -65,7 +65,9 @@ class SphereVsSphereAlgorithm : public NarrowPhaseAlgorithm { SphereVsSphereAlgorithm& operator=(const SphereVsSphereAlgorithm& algorithm) = delete; /// Compute a contact info if the two bounding volume collide - virtual bool testCollision(NarrowPhaseInfo* narrowPhaseInfo, bool reportContacts, MemoryAllocator& memoryAllocator) override; + virtual void testCollision(NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchStartIndex, + uint batchNbItems, bool reportContacts, + MemoryAllocator& memoryAllocator) override; }; } diff --git a/src/constraint/ContactPoint.h b/src/constraint/ContactPoint.h index 0c1193fc..e705c7bf 100644 --- a/src/constraint/ContactPoint.h +++ b/src/constraint/ContactPoint.h @@ -37,8 +37,6 @@ namespace reactphysics3d { // Declarations class CollisionBody; -struct NarrowPhaseInfo; - // Class ContactPoint /** * This class represents a collision contact point between two diff --git a/src/containers/List.h b/src/containers/List.h index 9a7b1a3a..2e09887f 100755 --- a/src/containers/List.h +++ b/src/containers/List.h @@ -39,7 +39,7 @@ namespace reactphysics3d { // Class List /** * This class represents a simple generic list with custom memory allocator. - */ + */ template class List { diff --git a/src/engine/OverlappingPair.cpp b/src/engine/OverlappingPair.cpp index 863bd1a8..4498cbf4 100644 --- a/src/engine/OverlappingPair.cpp +++ b/src/engine/OverlappingPair.cpp @@ -26,7 +26,6 @@ // Libraries #include #include "OverlappingPair.h" -#include "collision/NarrowPhaseInfo.h" #include "containers/containers_common.h" #include "collision/ContactPointInfo.h" diff --git a/src/engine/OverlappingPair.h b/src/engine/OverlappingPair.h index 30d1ab94..2671edfd 100644 --- a/src/engine/OverlappingPair.h +++ b/src/engine/OverlappingPair.h @@ -38,7 +38,7 @@ namespace reactphysics3d { // Declarations -struct NarrowPhaseInfo; +struct NarrowPhaseInfoBatch; class CollisionShape; // Structure LastFrameCollisionInfo @@ -159,7 +159,7 @@ class OverlappingPair { const ContactManifoldSet& getContactManifoldSet(); /// Add potential contact-points from narrow-phase into potential contact manifolds - void addPotentialContactPoints(NarrowPhaseInfo* narrowPhaseInfo); + void addPotentialContactPoints(const NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex); /// Return a reference to the temporary memory allocator MemoryAllocator& getTemporaryAllocator(); @@ -289,8 +289,8 @@ inline LastFrameCollisionInfo* OverlappingPair::getLastFrameCollisionInfo(uint s } // Create a new potential contact manifold using contact-points from narrow-phase -inline void OverlappingPair::addPotentialContactPoints(NarrowPhaseInfo* narrowPhaseInfo) { - mContactManifoldSet.addContactPoints(narrowPhaseInfo); +inline void OverlappingPair::addPotentialContactPoints(const NarrowPhaseInfoBatch& narrowPhaseInfoBatch, uint batchIndex) { + mContactManifoldSet.addContactPoints(narrowPhaseInfoBatch, batchIndex); } }