Bullet Collision Detection & Physics Library
btCompoundCollisionAlgorithm.cpp
Go to the documentation of this file.
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 
22 #include "LinearMath/btAabbUtil2.h"
23 #include "btManifoldResult.h"
25 
27 
29 :btActivatingCollisionAlgorithm(ci,body0Wrap,body1Wrap),
30 m_isSwapped(isSwapped),
31 m_sharedManifold(ci.m_manifold)
32 {
33  m_ownsManifold = false;
34 
35  const btCollisionObjectWrapper* colObjWrap = m_isSwapped? body1Wrap : body0Wrap;
36  btAssert (colObjWrap->getCollisionShape()->isCompound());
37 
38  const btCompoundShape* compoundShape = static_cast<const btCompoundShape*>(colObjWrap->getCollisionShape());
39  m_compoundShapeRevision = compoundShape->getUpdateRevision();
40 
41 
42  preallocateChildAlgorithms(body0Wrap,body1Wrap);
43 }
44 
46 {
47  const btCollisionObjectWrapper* colObjWrap = m_isSwapped? body1Wrap : body0Wrap;
48  const btCollisionObjectWrapper* otherObjWrap = m_isSwapped? body0Wrap : body1Wrap;
49  btAssert (colObjWrap->getCollisionShape()->isCompound());
50 
51  const btCompoundShape* compoundShape = static_cast<const btCompoundShape*>(colObjWrap->getCollisionShape());
52 
53  int numChildren = compoundShape->getNumChildShapes();
54  int i;
55 
57  for (i=0;i<numChildren;i++)
58  {
59  if (compoundShape->getDynamicAabbTree())
60  {
62  } else
63  {
64 
65  const btCollisionShape* childShape = compoundShape->getChildShape(i);
66 
67  btCollisionObjectWrapper childWrap(colObjWrap,childShape,colObjWrap->getCollisionObject(),colObjWrap->getWorldTransform(),-1,i);//wrong child trans, but unused (hopefully)
69  }
70  }
71 }
72 
74 {
75  int numChildren = m_childCollisionAlgorithms.size();
76  int i;
77  for (i=0;i<numChildren;i++)
78  {
80  {
81  m_childCollisionAlgorithms[i]->~btCollisionAlgorithm();
83  }
84  }
85 }
86 
88 {
90 }
91 
92 
93 
94 
96 {
97 
98 public:
99 
107 
108  btCompoundLeafCallback (const btCollisionObjectWrapper* compoundObjWrap,const btCollisionObjectWrapper* otherObjWrap,btDispatcher* dispatcher,const btDispatcherInfo& dispatchInfo,btManifoldResult* resultOut,btCollisionAlgorithm** childCollisionAlgorithms,btPersistentManifold* sharedManifold)
109  :m_compoundColObjWrap(compoundObjWrap),m_otherObjWrap(otherObjWrap),m_dispatcher(dispatcher),m_dispatchInfo(dispatchInfo),m_resultOut(resultOut),
110  m_childCollisionAlgorithms(childCollisionAlgorithms),
111  m_sharedManifold(sharedManifold)
112  {
113 
114  }
115 
116 
117  void ProcessChildShape(const btCollisionShape* childShape,int index)
118  {
119  btAssert(index>=0);
120  const btCompoundShape* compoundShape = static_cast<const btCompoundShape*>(m_compoundColObjWrap->getCollisionShape());
121  btAssert(index<compoundShape->getNumChildShapes());
122 
123 
124  //backup
126  btTransform orgInterpolationTrans = m_compoundColObjWrap->getWorldTransform();
127  const btTransform& childTrans = compoundShape->getChildTransform(index);
128  btTransform newChildWorldTrans = orgTrans*childTrans ;
129 
130  //perform an AABB check first
131  btVector3 aabbMin0,aabbMax0,aabbMin1,aabbMax1;
132  childShape->getAabb(newChildWorldTrans,aabbMin0,aabbMax0);
134 
136  {
138  return;
139  }
140 
141  if (TestAabbAgainstAabb2(aabbMin0,aabbMax0,aabbMin1,aabbMax1))
142  {
143 
144  btCollisionObjectWrapper compoundWrap(this->m_compoundColObjWrap,childShape,m_compoundColObjWrap->getCollisionObject(),newChildWorldTrans,-1,index);
145 
146 
147  //the contactpoint is still projected back using the original inverted worldtrans
148  if (!m_childCollisionAlgorithms[index])
150 
151 
152  const btCollisionObjectWrapper* tmpWrap = 0;
153 
156  {
157  tmpWrap = m_resultOut->getBody0Wrap();
158  m_resultOut->setBody0Wrap(&compoundWrap);
160  } else
161  {
162  tmpWrap = m_resultOut->getBody1Wrap();
163  m_resultOut->setBody1Wrap(&compoundWrap);
165  }
166 
167 
169 
170 #if 0
172  {
173  btVector3 worldAabbMin,worldAabbMax;
174  m_dispatchInfo.m_debugDraw->drawAabb(aabbMin0,aabbMax0,btVector3(1,1,1));
175  m_dispatchInfo.m_debugDraw->drawAabb(aabbMin1,aabbMax1,btVector3(1,1,1));
176  }
177 #endif
178 
180  {
181  m_resultOut->setBody0Wrap(tmpWrap);
182  } else
183  {
184  m_resultOut->setBody1Wrap(tmpWrap);
185  }
186 
187  }
188  }
189  void Process(const btDbvtNode* leaf)
190  {
191  int index = leaf->dataAsInt;
192 
193  const btCompoundShape* compoundShape = static_cast<const btCompoundShape*>(m_compoundColObjWrap->getCollisionShape());
194  const btCollisionShape* childShape = compoundShape->getChildShape(index);
195 
196 #if 0
198  {
199  btVector3 worldAabbMin,worldAabbMax;
201  btTransformAabb(leaf->volume.Mins(),leaf->volume.Maxs(),0.,orgTrans,worldAabbMin,worldAabbMax);
202  m_dispatchInfo.m_debugDraw->drawAabb(worldAabbMin,worldAabbMax,btVector3(1,0,0));
203  }
204 #endif
205 
206  ProcessChildShape(childShape,index);
207 
208  }
209 };
210 
211 
212 
213 
214 
215 
217 {
218  const btCollisionObjectWrapper* colObjWrap = m_isSwapped? body1Wrap : body0Wrap;
219  const btCollisionObjectWrapper* otherObjWrap = m_isSwapped? body0Wrap : body1Wrap;
220 
221  btAssert (colObjWrap->getCollisionShape()->isCompound());
222  const btCompoundShape* compoundShape = static_cast<const btCompoundShape*>(colObjWrap->getCollisionShape());
223 
226  if (compoundShape->getUpdateRevision() != m_compoundShapeRevision)
227  {
230 
231  preallocateChildAlgorithms(body0Wrap,body1Wrap);
232  }
233 
234 
235  const btDbvt* tree = compoundShape->getDynamicAabbTree();
236  //use a dynamic aabb tree to cull potential child-overlaps
237  btCompoundLeafCallback callback(colObjWrap,otherObjWrap,m_dispatcher,dispatchInfo,resultOut,&m_childCollisionAlgorithms[0],m_sharedManifold);
238 
242  {
243  int i;
244  btManifoldArray manifoldArray;
245  for (i=0;i<m_childCollisionAlgorithms.size();i++)
246  {
248  {
249  m_childCollisionAlgorithms[i]->getAllContactManifolds(manifoldArray);
250  for (int m=0;m<manifoldArray.size();m++)
251  {
252  if (manifoldArray[m]->getNumContacts())
253  {
254  resultOut->setPersistentManifold(manifoldArray[m]);
255  resultOut->refreshContactPoints();
256  resultOut->setPersistentManifold(0);//??necessary?
257  }
258  }
259  manifoldArray.resize(0);
260  }
261  }
262  }
263 
264  if (tree)
265  {
266 
267  btVector3 localAabbMin,localAabbMax;
268  btTransform otherInCompoundSpace;
269  otherInCompoundSpace = colObjWrap->getWorldTransform().inverse() * otherObjWrap->getWorldTransform();
270  otherObjWrap->getCollisionShape()->getAabb(otherInCompoundSpace,localAabbMin,localAabbMax);
271 
272  const ATTRIBUTE_ALIGNED16(btDbvtVolume) bounds=btDbvtVolume::FromMM(localAabbMin,localAabbMax);
273  //process all children, that overlap with the given AABB bounds
274  tree->collideTV(tree->m_root,bounds,callback);
275 
276  } else
277  {
278  //iterate over all children, perform an AABB check inside ProcessChildShape
279  int numChildren = m_childCollisionAlgorithms.size();
280  int i;
281  for (i=0;i<numChildren;i++)
282  {
283  callback.ProcessChildShape(compoundShape->getChildShape(i),i);
284  }
285  }
286 
287  {
288  //iterate over all children, perform an AABB check inside ProcessChildShape
289  int numChildren = m_childCollisionAlgorithms.size();
290  int i;
291  btManifoldArray manifoldArray;
292  const btCollisionShape* childShape = 0;
293  btTransform orgTrans;
294  btTransform orgInterpolationTrans;
295  btTransform newChildWorldTrans;
296  btVector3 aabbMin0,aabbMax0,aabbMin1,aabbMax1;
297 
298  for (i=0;i<numChildren;i++)
299  {
301  {
302  childShape = compoundShape->getChildShape(i);
303  //if not longer overlapping, remove the algorithm
304  orgTrans = colObjWrap->getWorldTransform();
305  orgInterpolationTrans = colObjWrap->getWorldTransform();
306  const btTransform& childTrans = compoundShape->getChildTransform(i);
307  newChildWorldTrans = orgTrans*childTrans ;
308 
309  //perform an AABB check first
310  childShape->getAabb(newChildWorldTrans,aabbMin0,aabbMax0);
311  otherObjWrap->getCollisionShape()->getAabb(otherObjWrap->getWorldTransform(),aabbMin1,aabbMax1);
312 
313  if (!TestAabbAgainstAabb2(aabbMin0,aabbMax0,aabbMin1,aabbMax1))
314  {
315  m_childCollisionAlgorithms[i]->~btCollisionAlgorithm();
318  }
319  }
320  }
321  }
322 }
323 
325 {
326  btAssert(0);
327  //needs to be fixed, using btCollisionObjectWrapper and NOT modifying internal data structures
328  btCollisionObject* colObj = m_isSwapped? body1 : body0;
329  btCollisionObject* otherObj = m_isSwapped? body0 : body1;
330 
331  btAssert (colObj->getCollisionShape()->isCompound());
332 
333  btCompoundShape* compoundShape = static_cast<btCompoundShape*>(colObj->getCollisionShape());
334 
335  //We will use the OptimizedBVH, AABB tree to cull potential child-overlaps
336  //If both proxies are Compound, we will deal with that directly, by performing sequential/parallel tree traversals
337  //given Proxy0 and Proxy1, if both have a tree, Tree0 and Tree1, this means:
338  //determine overlapping nodes of Proxy1 using Proxy0 AABB against Tree1
339  //then use each overlapping node AABB against Tree0
340  //and vise versa.
341 
342  btScalar hitFraction = btScalar(1.);
343 
344  int numChildren = m_childCollisionAlgorithms.size();
345  int i;
346  btTransform orgTrans;
347  btScalar frac;
348  for (i=0;i<numChildren;i++)
349  {
350  //btCollisionShape* childShape = compoundShape->getChildShape(i);
351 
352  //backup
353  orgTrans = colObj->getWorldTransform();
354 
355  const btTransform& childTrans = compoundShape->getChildTransform(i);
356  //btTransform newChildWorldTrans = orgTrans*childTrans ;
357  colObj->setWorldTransform( orgTrans*childTrans );
358 
359  //btCollisionShape* tmpShape = colObj->getCollisionShape();
360  //colObj->internalSetTemporaryCollisionShape( childShape );
361  frac = m_childCollisionAlgorithms[i]->calculateTimeOfImpact(colObj,otherObj,dispatchInfo,resultOut);
362  if (frac<hitFraction)
363  {
364  hitFraction = frac;
365  }
366  //revert back
367  //colObj->internalSetTemporaryCollisionShape( tmpShape);
368  colObj->setWorldTransform( orgTrans);
369  }
370  return hitFraction;
371 
372 }
373 
374 
375 
btPersistentManifold is a contact point cache, it stays persistent as long as objects are overlapping...
virtual btCollisionAlgorithm * findAlgorithm(const btCollisionObjectWrapper *body0Wrap, const btCollisionObjectWrapper *body1Wrap, btPersistentManifold *sharedManifold=0)=0
virtual void processCollision(const btCollisionObjectWrapper *body0Wrap, const btCollisionObjectWrapper *body1Wrap, const btDispatcherInfo &dispatchInfo, btManifoldResult *resultOut)=0
class btPersistentManifold * m_sharedManifold
void setPersistentManifold(btPersistentManifold *manifoldPtr)
btCompoundLeafCallback(const btCollisionObjectWrapper *compoundObjWrap, const btCollisionObjectWrapper *otherObjWrap, btDispatcher *dispatcher, const btDispatcherInfo &dispatchInfo, btManifoldResult *resultOut, btCollisionAlgorithm **childCollisionAlgorithms, btPersistentManifold *sharedManifold)
#define btAssert(x)
Definition: btScalar.h:101
This class is not enabled yet (work-in-progress) to more aggressively activate objects.
The btDbvt class implements a fast dynamic bounding volume tree based on axis aligned bounding boxes ...
Definition: btDbvt.h:194
The btCollisionShape class provides an interface for collision shapes that can be shared among btColl...
btCompoundCollisionAlgorithm(const btCollisionAlgorithmConstructionInfo &ci, const btCollisionObjectWrapper *body0Wrap, const btCollisionObjectWrapper *body1Wrap, bool isSwapped)
btDbvtNode * m_root
Definition: btDbvt.h:258
btManifoldResult is a helper class to manage contact results.
class btIDebugDraw * m_debugDraw
Definition: btDispatcher.h:58
const btCollisionObjectWrapper * getBody1Wrap() const
bool TestAabbAgainstAabb2(const btVector3 &aabbMin1, const btVector3 &aabbMax1, const btVector3 &aabbMin2, const btVector3 &aabbMax2)
conservative test for overlap between two aabbs
Definition: btAabbUtil2.h:48
const btDispatcherInfo & m_dispatchInfo
virtual void getAabb(const btTransform &t, btVector3 &aabbMin, btVector3 &aabbMax) const =0
getAabb returns the axis aligned bounding box in the coordinate frame of the given transform t...
virtual void setShapeIdentifiersA(int partId0, int index0)
setShapeIdentifiersA/B provides experimental support for per-triangle material / custom material comb...
btAlignedObjectArray< btCollisionAlgorithm * > m_childCollisionAlgorithms
virtual void processCollision(const btCollisionObjectWrapper *body0Wrap, const btCollisionObjectWrapper *body1Wrap, const btDispatcherInfo &dispatchInfo, btManifoldResult *resultOut)
btPersistentManifold * m_sharedManifold
btTransform & getWorldTransform()
int size() const
return the number of elements in the array
const btCollisionObjectWrapper * m_compoundColObjWrap
bool(* btShapePairCallback)(const btCollisionShape *pShape0, const btCollisionShape *pShape1)
const btCollisionObjectWrapper * getBody0Wrap() const
btTransform & getChildTransform(int index)
static btDbvtAabbMm FromMM(const btVector3 &mi, const btVector3 &mx)
Definition: btDbvt.h:411
const btTransform & getWorldTransform() const
btCollisionObject can be used to manage collision detection objects.
virtual void drawAabb(const btVector3 &from, const btVector3 &to, const btVector3 &color)
Definition: btIDebugDraw.h:107
const btCollisionObject * getBody0Internal() const
virtual void freeCollisionAlgorithm(void *ptr)=0
btTransform inverse() const
Return the inverse of this transform.
Definition: btTransform.h:188
virtual void setShapeIdentifiersB(int partId1, int index1)
void setWorldTransform(const btTransform &worldTrans)
const btCollisionShape * getCollisionShape() const
btVector3 can be used to represent 3D points and vectors.
Definition: btVector3.h:83
#define ATTRIBUTE_ALIGNED16(a)
Definition: btScalar.h:59
DBVT_INLINE const btVector3 & Maxs() const
Definition: btDbvt.h:136
The btTransform class supports rigid transforms with only translation and rotation and no scaling/she...
Definition: btTransform.h:34
void Process(const btDbvtNode *leaf)
bool isCompound() const
void resize(int newsize, const T &fillData=T())
btDbvtVolume volume
Definition: btDbvt.h:179
virtual int getDebugMode() const =0
void setBody1Wrap(const btCollisionObjectWrapper *obj1Wrap)
int dataAsInt
Definition: btDbvt.h:187
void preallocateChildAlgorithms(const btCollisionObjectWrapper *body0Wrap, const btCollisionObjectWrapper *body1Wrap)
btShapePairCallback gCompoundChildShapePairCallback
void btTransformAabb(const btVector3 &halfExtents, btScalar margin, const btTransform &t, btVector3 &aabbMinOut, btVector3 &aabbMaxOut)
Definition: btAabbUtil2.h:182
The btCompoundShape allows to store multiple other btCollisionShapes This allows for moving concave c...
btCollisionAlgorithm ** m_childCollisionAlgorithms
void setBody0Wrap(const btCollisionObjectWrapper *obj0Wrap)
DBVT_INLINE const btVector3 & Mins() const
Definition: btDbvt.h:135
btCollisionShape * getChildShape(int index)
The btDispatcher interface class can be used in combination with broadphase to dispatch calculations ...
Definition: btDispatcher.h:69
btCollisionAlgorithm is an collision interface that is compatible with the Broadphase and btDispatche...
const btCollisionShape * getCollisionShape() const
DBVT_PREFIX void collideTV(const btDbvtNode *root, const btDbvtVolume &volume, DBVT_IPOLICY) const
Definition: btDbvt.h:922
btScalar calculateTimeOfImpact(btCollisionObject *body0, btCollisionObject *body1, const btDispatcherInfo &dispatchInfo, btManifoldResult *resultOut)
void ProcessChildShape(const btCollisionShape *childShape, int index)
const btCollisionObjectWrapper * m_otherObjWrap
float btScalar
The btScalar type abstracts floating point numbers, to easily switch between double and single floati...
Definition: btScalar.h:266
static btDbvtVolume bounds(const tNodeArray &leaves)
Definition: btDbvt.cpp:249
const btCollisionObject * getCollisionObject() const
int getNumChildShapes() const