Home | History | Annotate | Download | only in NarrowPhaseCollision
      1 /*
      2 Bullet Continuous Collision Detection and Physics Library
      3 Copyright (c) 2003-2006 Erwin Coumans  http://continuousphysics.com/Bullet/
      4 
      5 This software is provided 'as-is', without any express or implied warranty.
      6 In no event will the authors be held liable for any damages arising from the use of this software.
      7 Permission is granted to anyone to use this software for any purpose,
      8 including commercial applications, and to alter it and redistribute it freely,
      9 subject to the following restrictions:
     10 
     11 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.
     12 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
     13 3. This notice may not be removed or altered from any source distribution.
     14 */
     15 
     16 #ifndef BT_PERSISTENT_MANIFOLD_H
     17 #define BT_PERSISTENT_MANIFOLD_H
     18 
     19 
     20 #include "LinearMath/btVector3.h"
     21 #include "LinearMath/btTransform.h"
     22 #include "btManifoldPoint.h"
     23 class btCollisionObject;
     24 #include "LinearMath/btAlignedAllocator.h"
     25 
     26 struct btCollisionResult;
     27 
     28 ///maximum contact breaking and merging threshold
     29 extern btScalar gContactBreakingThreshold;
     30 
     31 #ifndef SWIG
     32 class btPersistentManifold;
     33 
     34 typedef bool (*ContactDestroyedCallback)(void* userPersistentData);
     35 typedef bool (*ContactProcessedCallback)(btManifoldPoint& cp,void* body0,void* body1);
     36 typedef void(*ContactStartedCallback)(btPersistentManifold* const &manifold);
     37 typedef void(*ContactEndedCallback)(btPersistentManifold* const &manifold);
     38 extern ContactDestroyedCallback	gContactDestroyedCallback;
     39 extern ContactProcessedCallback gContactProcessedCallback;
     40 extern ContactStartedCallback gContactStartedCallback;
     41 extern ContactEndedCallback gContactEndedCallback;
     42 #endif //SWIG
     43 
     44 //the enum starts at 1024 to avoid type conflicts with btTypedConstraint
     45 enum btContactManifoldTypes
     46 {
     47 	MIN_CONTACT_MANIFOLD_TYPE = 1024,
     48 	BT_PERSISTENT_MANIFOLD_TYPE
     49 };
     50 
     51 #define MANIFOLD_CACHE_SIZE 4
     52 
     53 ///btPersistentManifold is a contact point cache, it stays persistent as long as objects are overlapping in the broadphase.
     54 ///Those contact points are created by the collision narrow phase.
     55 ///The cache can be empty, or hold 1,2,3 or 4 points. Some collision algorithms (GJK) might only add one point at a time.
     56 ///updates/refreshes old contact points, and throw them away if necessary (distance becomes too large)
     57 ///reduces the cache to 4 points, when more then 4 points are added, using following rules:
     58 ///the contact point with deepest penetration is always kept, and it tries to maximuze the area covered by the points
     59 ///note that some pairs of objects might have more then one contact manifold.
     60 
     61 
     62 ATTRIBUTE_ALIGNED128( class) btPersistentManifold : public btTypedObject
     63 //ATTRIBUTE_ALIGNED16( class) btPersistentManifold : public btTypedObject
     64 {
     65 
     66 	btManifoldPoint m_pointCache[MANIFOLD_CACHE_SIZE];
     67 
     68 	/// this two body pointers can point to the physics rigidbody class.
     69 	const btCollisionObject* m_body0;
     70 	const btCollisionObject* m_body1;
     71 
     72 	int	m_cachedPoints;
     73 
     74 	btScalar	m_contactBreakingThreshold;
     75 	btScalar	m_contactProcessingThreshold;
     76 
     77 
     78 	/// sort cached points so most isolated points come first
     79 	int	sortCachedPoints(const btManifoldPoint& pt);
     80 
     81 	int		findContactPoint(const btManifoldPoint* unUsed, int numUnused,const btManifoldPoint& pt);
     82 
     83 public:
     84 
     85 	BT_DECLARE_ALIGNED_ALLOCATOR();
     86 
     87 	int	m_companionIdA;
     88 	int	m_companionIdB;
     89 
     90 	int m_index1a;
     91 
     92 	btPersistentManifold();
     93 
     94 	btPersistentManifold(const btCollisionObject* body0,const btCollisionObject* body1,int , btScalar contactBreakingThreshold,btScalar contactProcessingThreshold)
     95 		: btTypedObject(BT_PERSISTENT_MANIFOLD_TYPE),
     96 	m_body0(body0),m_body1(body1),m_cachedPoints(0),
     97 		m_contactBreakingThreshold(contactBreakingThreshold),
     98 		m_contactProcessingThreshold(contactProcessingThreshold)
     99 	{
    100 	}
    101 
    102 	SIMD_FORCE_INLINE const btCollisionObject* getBody0() const { return m_body0;}
    103 	SIMD_FORCE_INLINE const btCollisionObject* getBody1() const { return m_body1;}
    104 
    105 	void	setBodies(const btCollisionObject* body0,const btCollisionObject* body1)
    106 	{
    107 		m_body0 = body0;
    108 		m_body1 = body1;
    109 	}
    110 
    111 	void clearUserCache(btManifoldPoint& pt);
    112 
    113 #ifdef DEBUG_PERSISTENCY
    114 	void	DebugPersistency();
    115 #endif //
    116 
    117 	SIMD_FORCE_INLINE int	getNumContacts() const { return m_cachedPoints;}
    118 	/// the setNumContacts API is usually not used, except when you gather/fill all contacts manually
    119 	void setNumContacts(int cachedPoints)
    120 	{
    121 		m_cachedPoints = cachedPoints;
    122 	}
    123 
    124 
    125 	SIMD_FORCE_INLINE const btManifoldPoint& getContactPoint(int index) const
    126 	{
    127 		btAssert(index < m_cachedPoints);
    128 		return m_pointCache[index];
    129 	}
    130 
    131 	SIMD_FORCE_INLINE btManifoldPoint& getContactPoint(int index)
    132 	{
    133 		btAssert(index < m_cachedPoints);
    134 		return m_pointCache[index];
    135 	}
    136 
    137 	///@todo: get this margin from the current physics / collision environment
    138 	btScalar	getContactBreakingThreshold() const;
    139 
    140 	btScalar	getContactProcessingThreshold() const
    141 	{
    142 		return m_contactProcessingThreshold;
    143 	}
    144 
    145 	void setContactBreakingThreshold(btScalar contactBreakingThreshold)
    146 	{
    147 		m_contactBreakingThreshold = contactBreakingThreshold;
    148 	}
    149 
    150 	void setContactProcessingThreshold(btScalar	contactProcessingThreshold)
    151 	{
    152 		m_contactProcessingThreshold = contactProcessingThreshold;
    153 	}
    154 
    155 
    156 
    157 
    158 	int getCacheEntry(const btManifoldPoint& newPoint) const;
    159 
    160 	int addManifoldPoint( const btManifoldPoint& newPoint, bool isPredictive=false);
    161 
    162 	void removeContactPoint (int index)
    163 	{
    164 		clearUserCache(m_pointCache[index]);
    165 
    166 		int lastUsedIndex = getNumContacts() - 1;
    167 //		m_pointCache[index] = m_pointCache[lastUsedIndex];
    168 		if(index != lastUsedIndex)
    169 		{
    170 			m_pointCache[index] = m_pointCache[lastUsedIndex];
    171 			//get rid of duplicated userPersistentData pointer
    172 			m_pointCache[lastUsedIndex].m_userPersistentData = 0;
    173 			m_pointCache[lastUsedIndex].m_appliedImpulse = 0.f;
    174 			m_pointCache[lastUsedIndex].m_lateralFrictionInitialized = false;
    175 			m_pointCache[lastUsedIndex].m_appliedImpulseLateral1 = 0.f;
    176 			m_pointCache[lastUsedIndex].m_appliedImpulseLateral2 = 0.f;
    177 			m_pointCache[lastUsedIndex].m_lifeTime = 0;
    178 		}
    179 
    180 		btAssert(m_pointCache[lastUsedIndex].m_userPersistentData==0);
    181 		m_cachedPoints--;
    182 
    183 		if (gContactEndedCallback && m_cachedPoints == 0)
    184 		{
    185 			gContactEndedCallback(this);
    186 		}
    187 	}
    188 	void replaceContactPoint(const btManifoldPoint& newPoint,int insertIndex)
    189 	{
    190 		btAssert(validContactDistance(newPoint));
    191 
    192 #define MAINTAIN_PERSISTENCY 1
    193 #ifdef MAINTAIN_PERSISTENCY
    194 		int	lifeTime = m_pointCache[insertIndex].getLifeTime();
    195 		btScalar	appliedImpulse = m_pointCache[insertIndex].m_appliedImpulse;
    196 		btScalar	appliedLateralImpulse1 = m_pointCache[insertIndex].m_appliedImpulseLateral1;
    197 		btScalar	appliedLateralImpulse2 = m_pointCache[insertIndex].m_appliedImpulseLateral2;
    198 //		bool isLateralFrictionInitialized = m_pointCache[insertIndex].m_lateralFrictionInitialized;
    199 
    200 
    201 
    202 		btAssert(lifeTime>=0);
    203 		void* cache = m_pointCache[insertIndex].m_userPersistentData;
    204 
    205 		m_pointCache[insertIndex] = newPoint;
    206 
    207 		m_pointCache[insertIndex].m_userPersistentData = cache;
    208 		m_pointCache[insertIndex].m_appliedImpulse = appliedImpulse;
    209 		m_pointCache[insertIndex].m_appliedImpulseLateral1 = appliedLateralImpulse1;
    210 		m_pointCache[insertIndex].m_appliedImpulseLateral2 = appliedLateralImpulse2;
    211 
    212 		m_pointCache[insertIndex].m_appliedImpulse =  appliedImpulse;
    213 		m_pointCache[insertIndex].m_appliedImpulseLateral1 = appliedLateralImpulse1;
    214 		m_pointCache[insertIndex].m_appliedImpulseLateral2 = appliedLateralImpulse2;
    215 
    216 
    217 		m_pointCache[insertIndex].m_lifeTime = lifeTime;
    218 #else
    219 		clearUserCache(m_pointCache[insertIndex]);
    220 		m_pointCache[insertIndex] = newPoint;
    221 
    222 #endif
    223 	}
    224 
    225 
    226 	bool validContactDistance(const btManifoldPoint& pt) const
    227 	{
    228 		return pt.m_distance1 <= getContactBreakingThreshold();
    229 	}
    230 	/// calculated new worldspace coordinates and depth, and reject points that exceed the collision margin
    231 	void	refreshContactPoints(  const btTransform& trA,const btTransform& trB);
    232 
    233 
    234 	SIMD_FORCE_INLINE	void	clearManifold()
    235 	{
    236 		int i;
    237 		for (i=0;i<m_cachedPoints;i++)
    238 		{
    239 			clearUserCache(m_pointCache[i]);
    240 		}
    241 
    242 		if (gContactEndedCallback && m_cachedPoints)
    243 		{
    244 			gContactEndedCallback(this);
    245 		}
    246 		m_cachedPoints = 0;
    247 	}
    248 
    249 
    250 
    251 }
    252 ;
    253 
    254 
    255 
    256 
    257 
    258 #endif //BT_PERSISTENT_MANIFOLD_H
    259