/******************************************************************************** * ReactPhysics3D physics library, http://www.reactphysics3d.com * * Copyright (c) 2010-2018 Daniel Chappuis * ********************************************************************************* * * * This software is provided 'as-is', without any express or implied warranty. * * In no event will the authors be held liable for any damages arising from the * * use of this software. * * * * Permission is granted to anyone to use this software for any purpose, * * including commercial applications, and to alter it and redistribute it * * freely, subject to the following restrictions: * * * * 1. The origin of this software must not be misrepresented; you must not claim * * that you wrote the original software. If you use this software in a * * product, an acknowledgment in the product documentation would be * * appreciated but is not required. * * * * 2. Altered source versions must be plainly marked as such, and must not be * * misrepresented as being the original software. * * * * 3. This notice may not be removed or altered from any source distribution. * * * ********************************************************************************/ #ifndef REACTPHYSICS3D_OVERLAPPING_PAIR_H #define REACTPHYSICS3D_OVERLAPPING_PAIR_H // Libraries #include "collision/ProxyShape.h" #include "containers/Map.h" #include "containers/Pair.h" #include "containers/Set.h" #include "containers/containers_common.h" #include "utils/Profiler.h" #include "components/ProxyShapeComponents.h" #include "components/CollisionBodyComponents.h" #include "components/RigidBodyComponents.h" #include /// ReactPhysics3D namespace namespace reactphysics3d { // Declarations struct NarrowPhaseInfoBatch; enum class NarrowPhaseAlgorithmType; class CollisionShape; class CollisionDispatch; // Structure LastFrameCollisionInfo /** * This structure contains collision info about the last frame. * This is used for temporal coherence between frames. */ struct LastFrameCollisionInfo { /// True if we have information about the previous frame bool isValid; /// True if the frame info is obsolete (the collision shape are not overlapping in middle phase) bool isObsolete; /// True if the two shapes were colliding in the previous frame bool wasColliding; /// True if we were using GJK algorithm to check for collision in the previous frame bool wasUsingGJK; /// True if we were using SAT algorithm to check for collision in the previous frame bool wasUsingSAT; // ----- GJK Algorithm ----- /// Previous separating axis Vector3 gjkSeparatingAxis; // SAT Algorithm bool satIsAxisFacePolyhedron1; bool satIsAxisFacePolyhedron2; uint satMinAxisFaceIndex; uint satMinEdge1Index; uint satMinEdge2Index; /// Constructor LastFrameCollisionInfo() { isValid = false; isObsolete = false; wasColliding = false; wasUsingSAT = false; wasUsingGJK = false; gjkSeparatingAxis = Vector3(0, 1, 0); } }; // Class OverlappingPairs /** * This class contains pairs of two proxy collision shapes that are overlapping * during the broad-phase collision detection. A pair is created when * the two proxy collision shapes start to overlap and is destroyed when they do not * overlap anymore. Each contains a contact manifold that * store all the contact points between the two bodies. */ class OverlappingPairs { public: // TODO : Try to use a pairing function like pairNumbers() here using ShapeIdPair = Pair; private: // -------------------- Constants -------------------- // /// Number of pairs to allocated at the beginning const uint32 INIT_NB_ALLOCATED_PAIRS = 10; // -------------------- Attributes -------------------- // /// Persistent memory allocator MemoryAllocator& mPersistentAllocator; /// Memory allocator used to allocated memory for the ContactManifoldInfo and ContactPointInfo // TODO : Do we need to keep this ? MemoryAllocator& mTempMemoryAllocator; /// Current number of components uint64 mNbPairs; /// Index in the array of the first convex vs concave pair uint64 mConcavePairsStartIndex; /// Size (in bytes) of a single pair size_t mPairDataSize; /// Number of allocated pairs uint64 mNbAllocatedPairs; /// Allocated memory for all the data of the pairs void* mBuffer; /// Map a pair id to the internal array index Map mMapPairIdToPairIndex; /// Ids of the convex vs convex pairs // TODO : Check if we need this array uint64* mPairIds; /// Array with the broad-phase Ids of the first shape int32* mPairBroadPhaseId1; /// Array with the broad-phase Ids of the second shape int32* mPairBroadPhaseId2; /// Array of Entity of the first proxy-shape of the convex vs convex pairs Entity* mProxyShapes1; /// Array of Entity of the second proxy-shape of the convex vs convex pairs Entity* mProxyShapes2; /// Temporal coherence collision data for each overlapping collision shapes of this pair. /// Temporal coherence data store collision information about the last frame. /// If two convex shapes overlap, we have a single collision data but if one shape is concave, /// we might have collision data for several overlapping triangles. The key in the map is the /// shape Ids of the two collision shapes. Map* mLastFrameCollisionInfos; /// True if we need to test if the convex vs convex overlapping pairs of shapes still overlap bool* mNeedToTestOverlap; /// True if the overlapping pair is active (at least one body of the pair is active and not static) bool* mIsActive; /// Array with the pointer to the narrow-phase algorithm for each overlapping pair NarrowPhaseAlgorithmType* mNarrowPhaseAlgorithmType; /// Reference to the proxy-shapes components ProxyShapeComponents& mProxyShapeComponents; /// Reference to the collision body components CollisionBodyComponents& mCollisionBodyComponents; /// Reference to the rigid bodies components RigidBodyComponents& mRigidBodyComponents; /// Reference to the set of bodies that cannot collide with each others Set& mNoCollisionPairs; /// Reference to the collision dispatch CollisionDispatch& mCollisionDispatch; #ifdef IS_PROFILING_ACTIVE /// Pointer to the profiler Profiler* mProfiler; #endif // -------------------- Methods -------------------- // /// Allocate memory for a given number of pairs void allocate(uint64 nbPairsToAllocate); /// Compute the index where we need to insert the new pair uint64 prepareAddPair(bool isConvexVsConvex); /// Destroy a pair at a given index void destroyPair(uint64 index); // Move a pair from a source to a destination index in the pairs array void movePairToIndex(uint64 srcIndex, uint64 destIndex); /// Swap two pairs in the array void swapPairs(uint64 index1, uint64 index2); public: // -------------------- Methods -------------------- // /// Constructor OverlappingPairs(MemoryAllocator& persistentMemoryAllocator, MemoryAllocator& temporaryMemoryAllocator, ProxyShapeComponents& proxyShapeComponents, CollisionBodyComponents& collisionBodyComponents, RigidBodyComponents& rigidBodyComponents, Set& noCollisionPairs, CollisionDispatch& collisionDispatch); /// Destructor ~OverlappingPairs(); /// Deleted copy-constructor OverlappingPairs(const OverlappingPairs& pair) = delete; /// Deleted assignment operator OverlappingPairs& operator=(const OverlappingPairs& pair) = delete; /// Add an overlapping pair uint64 addPair(ProxyShape* shape1, ProxyShape* shape2, bool isActive); /// Remove a component at a given index void removePair(uint64 pairId); /// Return the number of pairs uint64 getNbPairs() const; /// Return the number of convex vs convex pairs uint64 getNbConvexVsConvexPairs() const; /// Return the number of convex vs concave pairs uint64 getNbConvexVsConcavePairs() const; /// Return the starting index of the convex vs concave pairs uint64 getConvexVsConcavePairsStartIndex() const; /// Return the entity of the first proxy-shape Entity getProxyShape1(uint64 pairId) const; /// Return the entity of the second proxy-shape Entity getProxyShape2(uint64 pairId) const; /// Notify if a given pair is active or not void setIsPairActive(uint64 pairId, bool isActive); /// Return the last frame collision info LastFrameCollisionInfo* getLastFrameCollisionInfo(uint64, ShapeIdPair& shapeIds); /// Return a reference to the temporary memory allocator MemoryAllocator& getTemporaryAllocator(); /// Add a new last frame collision info if it does not exist for the given shapes already LastFrameCollisionInfo* addLastFrameInfoIfNecessary(uint64 pairId, uint shapeId1, uint shapeId2); /// Delete all the obsolete last frame collision info void clearObsoleteLastFrameCollisionInfos(); /// Return the pair of bodies index of the pair static bodypair computeBodiesIndexPair(Entity body1Entity, Entity body2Entity); /// Return true if the overlapping pair between two shapes is active bool computeIsPairActive(ProxyShape* shape1, ProxyShape* shape2); /// Set if we need to test a given pair for overlap void setNeedToTestOverlap(uint64 pairId, bool needToTestOverlap); #ifdef IS_PROFILING_ACTIVE /// Set the profiler void setProfiler(Profiler* profiler); #endif // -------------------- Friendship -------------------- // friend class DynamicsWorld; friend class CollisionDetectionSystem; }; // Return the entity of the first proxy-shape inline Entity OverlappingPairs::getProxyShape1(uint64 pairId) const { assert(mMapPairIdToPairIndex.containsKey(pairId)); assert(mMapPairIdToPairIndex[pairId] < mNbPairs); return mProxyShapes1[mMapPairIdToPairIndex[pairId]]; } // Return the entity of the second proxy-shape inline Entity OverlappingPairs::getProxyShape2(uint64 pairId) const { assert(mMapPairIdToPairIndex.containsKey(pairId)); assert(mMapPairIdToPairIndex[pairId] < mNbPairs); return mProxyShapes2[mMapPairIdToPairIndex[pairId]]; } // Notify if a given pair is active or not inline void OverlappingPairs::setIsPairActive(uint64 pairId, bool isActive) { assert(mMapPairIdToPairIndex.containsKey(pairId)); assert(mMapPairIdToPairIndex[pairId] < mNbPairs); mIsActive[mMapPairIdToPairIndex[pairId]] = isActive; } // Return the last frame collision info for a given shape id or nullptr if none is found inline LastFrameCollisionInfo* OverlappingPairs::getLastFrameCollisionInfo(uint64 pairId, ShapeIdPair& shapeIds) { assert(mMapPairIdToPairIndex.containsKey(pairId)); const uint64 index = mMapPairIdToPairIndex[pairId]; assert(index < mNbPairs); Map::Iterator it = mLastFrameCollisionInfos[index].find(shapeIds); if (it != mLastFrameCollisionInfos[index].end()) { return it->second; } return nullptr; } // Return the pair of bodies index inline bodypair OverlappingPairs::computeBodiesIndexPair(Entity body1Entity, Entity body2Entity) { // Construct the pair of body index bodypair indexPair = body1Entity.id < body2Entity.id ? bodypair(body1Entity, body2Entity) : bodypair(body2Entity, body1Entity); assert(indexPair.first != indexPair.second); return indexPair; } // Return the number of pairs inline uint64 OverlappingPairs::getNbPairs() const { return mNbPairs; } // Return the number of convex vs convex pairs inline uint64 OverlappingPairs::getNbConvexVsConvexPairs() const { return mConcavePairsStartIndex; } // Return the number of convex vs concave pairs inline uint64 OverlappingPairs::getNbConvexVsConcavePairs() const { return mNbPairs - mConcavePairsStartIndex; } // Return the starting index of the convex vs concave pairs inline uint64 OverlappingPairs::getConvexVsConcavePairsStartIndex() const { return mConcavePairsStartIndex; } // Return a reference to the temporary memory allocator inline MemoryAllocator& OverlappingPairs::getTemporaryAllocator() { return mTempMemoryAllocator; } // Set if we need to test a given pair for overlap inline void OverlappingPairs::setNeedToTestOverlap(uint64 pairId, bool needToTestOverlap) { assert(mMapPairIdToPairIndex.containsKey(pairId)); mNeedToTestOverlap[mMapPairIdToPairIndex[pairId]] = needToTestOverlap; } #ifdef IS_PROFILING_ACTIVE // Set the profiler inline void OverlappingPairs::setProfiler(Profiler* profiler) { mProfiler = profiler; } #endif } #endif