389 lines
14 KiB
C++
389 lines
14 KiB
C++
/********************************************************************************
|
|
* 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 <cstddef>
|
|
|
|
/// 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<uint, uint>;
|
|
|
|
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<uint64, uint64> 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<ShapeIdPair, LastFrameCollisionInfo*>* 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<bodypair>& 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<bodypair>& 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<ShapeIdPair, LastFrameCollisionInfo*>::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
|
|
|