2
0

btBatchedConstraints.cpp 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080
  1. /*
  2. Bullet Continuous Collision Detection and Physics Library
  3. Copyright (c) 2003-2006 Erwin Coumans http://continuousphysics.com/Bullet/
  4. This software is provided 'as-is', without any express or implied warranty.
  5. In no event will the authors be held liable for any damages arising from the use of this software.
  6. Permission is granted to anyone to use this software for any purpose,
  7. including commercial applications, and to alter it and redistribute it freely,
  8. subject to the following restrictions:
  9. 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.
  10. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
  11. 3. This notice may not be removed or altered from any source distribution.
  12. */
  13. #include "btBatchedConstraints.h"
  14. #include "LinearMath/btIDebugDraw.h"
  15. #include "LinearMath/btMinMax.h"
  16. #include "LinearMath/btStackAlloc.h"
  17. #include "LinearMath/btQuickprof.h"
  18. #include <string.h> //for memset
  19. const int kNoMerge = -1;
  20. bool btBatchedConstraints::s_debugDrawBatches = false;
  21. struct btBatchedConstraintInfo
  22. {
  23. int constraintIndex;
  24. int numConstraintRows;
  25. int bodyIds[2];
  26. };
  27. struct btBatchInfo
  28. {
  29. int numConstraints;
  30. int mergeIndex;
  31. btBatchInfo() : numConstraints(0), mergeIndex(kNoMerge) {}
  32. };
  33. bool btBatchedConstraints::validate(btConstraintArray* constraints, const btAlignedObjectArray<btSolverBody>& bodies) const
  34. {
  35. //
  36. // validate: for debugging only. Verify coloring of bodies, that no body is touched by more than one batch in any given phase
  37. //
  38. int errors = 0;
  39. const int kUnassignedBatch = -1;
  40. btAlignedObjectArray<int> bodyBatchId;
  41. for (int iPhase = 0; iPhase < m_phases.size(); ++iPhase)
  42. {
  43. bodyBatchId.resizeNoInitialize(0);
  44. bodyBatchId.resize(bodies.size(), kUnassignedBatch);
  45. const Range& phase = m_phases[iPhase];
  46. for (int iBatch = phase.begin; iBatch < phase.end; ++iBatch)
  47. {
  48. const Range& batch = m_batches[iBatch];
  49. for (int iiCons = batch.begin; iiCons < batch.end; ++iiCons)
  50. {
  51. int iCons = m_constraintIndices[iiCons];
  52. const btSolverConstraint& cons = constraints->at(iCons);
  53. const btSolverBody& bodyA = bodies[cons.m_solverBodyIdA];
  54. const btSolverBody& bodyB = bodies[cons.m_solverBodyIdB];
  55. if (!bodyA.internalGetInvMass().isZero())
  56. {
  57. int thisBodyBatchId = bodyBatchId[cons.m_solverBodyIdA];
  58. if (thisBodyBatchId == kUnassignedBatch)
  59. {
  60. bodyBatchId[cons.m_solverBodyIdA] = iBatch;
  61. }
  62. else if (thisBodyBatchId != iBatch)
  63. {
  64. btAssert(!"dynamic body is used in 2 different batches in the same phase");
  65. errors++;
  66. }
  67. }
  68. if (!bodyB.internalGetInvMass().isZero())
  69. {
  70. int thisBodyBatchId = bodyBatchId[cons.m_solverBodyIdB];
  71. if (thisBodyBatchId == kUnassignedBatch)
  72. {
  73. bodyBatchId[cons.m_solverBodyIdB] = iBatch;
  74. }
  75. else if (thisBodyBatchId != iBatch)
  76. {
  77. btAssert(!"dynamic body is used in 2 different batches in the same phase");
  78. errors++;
  79. }
  80. }
  81. }
  82. }
  83. }
  84. return errors == 0;
  85. }
  86. static void debugDrawSingleBatch(const btBatchedConstraints* bc,
  87. btConstraintArray* constraints,
  88. const btAlignedObjectArray<btSolverBody>& bodies,
  89. int iBatch,
  90. const btVector3& color,
  91. const btVector3& offset)
  92. {
  93. if (bc && bc->m_debugDrawer && iBatch < bc->m_batches.size())
  94. {
  95. const btBatchedConstraints::Range& b = bc->m_batches[iBatch];
  96. for (int iiCon = b.begin; iiCon < b.end; ++iiCon)
  97. {
  98. int iCon = bc->m_constraintIndices[iiCon];
  99. const btSolverConstraint& con = constraints->at(iCon);
  100. int iBody0 = con.m_solverBodyIdA;
  101. int iBody1 = con.m_solverBodyIdB;
  102. btVector3 pos0 = bodies[iBody0].getWorldTransform().getOrigin() + offset;
  103. btVector3 pos1 = bodies[iBody1].getWorldTransform().getOrigin() + offset;
  104. bc->m_debugDrawer->drawLine(pos0, pos1, color);
  105. }
  106. }
  107. }
  108. static void debugDrawPhase(const btBatchedConstraints* bc,
  109. btConstraintArray* constraints,
  110. const btAlignedObjectArray<btSolverBody>& bodies,
  111. int iPhase,
  112. const btVector3& color0,
  113. const btVector3& color1,
  114. const btVector3& offset)
  115. {
  116. BT_PROFILE("debugDrawPhase");
  117. if (bc && bc->m_debugDrawer && iPhase < bc->m_phases.size())
  118. {
  119. const btBatchedConstraints::Range& phase = bc->m_phases[iPhase];
  120. for (int iBatch = phase.begin; iBatch < phase.end; ++iBatch)
  121. {
  122. float tt = float(iBatch - phase.begin) / float(btMax(1, phase.end - phase.begin - 1));
  123. btVector3 col = lerp(color0, color1, tt);
  124. debugDrawSingleBatch(bc, constraints, bodies, iBatch, col, offset);
  125. }
  126. }
  127. }
  128. static void debugDrawAllBatches(const btBatchedConstraints* bc,
  129. btConstraintArray* constraints,
  130. const btAlignedObjectArray<btSolverBody>& bodies)
  131. {
  132. BT_PROFILE("debugDrawAllBatches");
  133. if (bc && bc->m_debugDrawer && bc->m_phases.size() > 0)
  134. {
  135. btVector3 bboxMin(BT_LARGE_FLOAT, BT_LARGE_FLOAT, BT_LARGE_FLOAT);
  136. btVector3 bboxMax = -bboxMin;
  137. for (int iBody = 0; iBody < bodies.size(); ++iBody)
  138. {
  139. const btVector3& pos = bodies[iBody].getWorldTransform().getOrigin();
  140. bboxMin.setMin(pos);
  141. bboxMax.setMax(pos);
  142. }
  143. btVector3 bboxExtent = bboxMax - bboxMin;
  144. btVector3 offsetBase = btVector3(0, bboxExtent.y() * 1.1f, 0);
  145. btVector3 offsetStep = btVector3(0, 0, bboxExtent.z() * 1.1f);
  146. int numPhases = bc->m_phases.size();
  147. for (int iPhase = 0; iPhase < numPhases; ++iPhase)
  148. {
  149. float b = float(iPhase) / float(numPhases - 1);
  150. btVector3 color0 = btVector3(1, 0, b);
  151. btVector3 color1 = btVector3(0, 1, b);
  152. btVector3 offset = offsetBase + offsetStep * (float(iPhase) - float(numPhases - 1) * 0.5);
  153. debugDrawPhase(bc, constraints, bodies, iPhase, color0, color1, offset);
  154. }
  155. }
  156. }
  157. static void initBatchedBodyDynamicFlags(btAlignedObjectArray<bool>* outBodyDynamicFlags, const btAlignedObjectArray<btSolverBody>& bodies)
  158. {
  159. BT_PROFILE("initBatchedBodyDynamicFlags");
  160. btAlignedObjectArray<bool>& bodyDynamicFlags = *outBodyDynamicFlags;
  161. bodyDynamicFlags.resizeNoInitialize(bodies.size());
  162. for (int i = 0; i < bodies.size(); ++i)
  163. {
  164. const btSolverBody& body = bodies[i];
  165. bodyDynamicFlags[i] = (body.internalGetInvMass().x() > btScalar(0));
  166. }
  167. }
  168. static int runLengthEncodeConstraintInfo(btBatchedConstraintInfo* outConInfos, int numConstraints)
  169. {
  170. BT_PROFILE("runLengthEncodeConstraintInfo");
  171. // detect and run-length encode constraint rows that repeat the same bodies
  172. int iDest = 0;
  173. int iSrc = 0;
  174. while (iSrc < numConstraints)
  175. {
  176. const btBatchedConstraintInfo& srcConInfo = outConInfos[iSrc];
  177. btBatchedConstraintInfo& conInfo = outConInfos[iDest];
  178. conInfo.constraintIndex = iSrc;
  179. conInfo.bodyIds[0] = srcConInfo.bodyIds[0];
  180. conInfo.bodyIds[1] = srcConInfo.bodyIds[1];
  181. while (iSrc < numConstraints && outConInfos[iSrc].bodyIds[0] == srcConInfo.bodyIds[0] && outConInfos[iSrc].bodyIds[1] == srcConInfo.bodyIds[1])
  182. {
  183. ++iSrc;
  184. }
  185. conInfo.numConstraintRows = iSrc - conInfo.constraintIndex;
  186. ++iDest;
  187. }
  188. return iDest;
  189. }
  190. struct ReadSolverConstraintsLoop : public btIParallelForBody
  191. {
  192. btBatchedConstraintInfo* m_outConInfos;
  193. btConstraintArray* m_constraints;
  194. ReadSolverConstraintsLoop(btBatchedConstraintInfo* outConInfos, btConstraintArray* constraints)
  195. {
  196. m_outConInfos = outConInfos;
  197. m_constraints = constraints;
  198. }
  199. void forLoop(int iBegin, int iEnd) const BT_OVERRIDE
  200. {
  201. for (int i = iBegin; i < iEnd; ++i)
  202. {
  203. btBatchedConstraintInfo& conInfo = m_outConInfos[i];
  204. const btSolverConstraint& con = m_constraints->at(i);
  205. conInfo.bodyIds[0] = con.m_solverBodyIdA;
  206. conInfo.bodyIds[1] = con.m_solverBodyIdB;
  207. conInfo.constraintIndex = i;
  208. conInfo.numConstraintRows = 1;
  209. }
  210. }
  211. };
  212. static int initBatchedConstraintInfo(btBatchedConstraintInfo* outConInfos, btConstraintArray* constraints)
  213. {
  214. BT_PROFILE("initBatchedConstraintInfo");
  215. int numConstraints = constraints->size();
  216. bool inParallel = true;
  217. if (inParallel)
  218. {
  219. ReadSolverConstraintsLoop loop(outConInfos, constraints);
  220. int grainSize = 1200;
  221. btParallelFor(0, numConstraints, grainSize, loop);
  222. }
  223. else
  224. {
  225. for (int i = 0; i < numConstraints; ++i)
  226. {
  227. btBatchedConstraintInfo& conInfo = outConInfos[i];
  228. const btSolverConstraint& con = constraints->at(i);
  229. conInfo.bodyIds[0] = con.m_solverBodyIdA;
  230. conInfo.bodyIds[1] = con.m_solverBodyIdB;
  231. conInfo.constraintIndex = i;
  232. conInfo.numConstraintRows = 1;
  233. }
  234. }
  235. bool useRunLengthEncoding = true;
  236. if (useRunLengthEncoding)
  237. {
  238. numConstraints = runLengthEncodeConstraintInfo(outConInfos, numConstraints);
  239. }
  240. return numConstraints;
  241. }
  242. static void expandConstraintRowsInPlace(int* constraintBatchIds, const btBatchedConstraintInfo* conInfos, int numConstraints, int numConstraintRows)
  243. {
  244. BT_PROFILE("expandConstraintRowsInPlace");
  245. if (numConstraintRows > numConstraints)
  246. {
  247. // we walk the array in reverse to avoid overwriteing
  248. for (int iCon = numConstraints - 1; iCon >= 0; --iCon)
  249. {
  250. const btBatchedConstraintInfo& conInfo = conInfos[iCon];
  251. int iBatch = constraintBatchIds[iCon];
  252. for (int i = conInfo.numConstraintRows - 1; i >= 0; --i)
  253. {
  254. int iDest = conInfo.constraintIndex + i;
  255. btAssert(iDest >= iCon);
  256. btAssert(iDest >= 0 && iDest < numConstraintRows);
  257. constraintBatchIds[iDest] = iBatch;
  258. }
  259. }
  260. }
  261. }
  262. static void expandConstraintRows(int* destConstraintBatchIds, const int* srcConstraintBatchIds, const btBatchedConstraintInfo* conInfos, int numConstraints, int numConstraintRows)
  263. {
  264. BT_PROFILE("expandConstraintRows");
  265. for (int iCon = 0; iCon < numConstraints; ++iCon)
  266. {
  267. const btBatchedConstraintInfo& conInfo = conInfos[iCon];
  268. int iBatch = srcConstraintBatchIds[iCon];
  269. for (int i = 0; i < conInfo.numConstraintRows; ++i)
  270. {
  271. int iDest = conInfo.constraintIndex + i;
  272. btAssert(iDest >= iCon);
  273. btAssert(iDest >= 0 && iDest < numConstraintRows);
  274. destConstraintBatchIds[iDest] = iBatch;
  275. }
  276. }
  277. }
  278. struct ExpandConstraintRowsLoop : public btIParallelForBody
  279. {
  280. int* m_destConstraintBatchIds;
  281. const int* m_srcConstraintBatchIds;
  282. const btBatchedConstraintInfo* m_conInfos;
  283. int m_numConstraintRows;
  284. ExpandConstraintRowsLoop(int* destConstraintBatchIds, const int* srcConstraintBatchIds, const btBatchedConstraintInfo* conInfos, int numConstraintRows)
  285. {
  286. m_destConstraintBatchIds = destConstraintBatchIds;
  287. m_srcConstraintBatchIds = srcConstraintBatchIds;
  288. m_conInfos = conInfos;
  289. m_numConstraintRows = numConstraintRows;
  290. }
  291. void forLoop(int iBegin, int iEnd) const BT_OVERRIDE
  292. {
  293. expandConstraintRows(m_destConstraintBatchIds, m_srcConstraintBatchIds + iBegin, m_conInfos + iBegin, iEnd - iBegin, m_numConstraintRows);
  294. }
  295. };
  296. static void expandConstraintRowsMt(int* destConstraintBatchIds, const int* srcConstraintBatchIds, const btBatchedConstraintInfo* conInfos, int numConstraints, int numConstraintRows)
  297. {
  298. BT_PROFILE("expandConstraintRowsMt");
  299. ExpandConstraintRowsLoop loop(destConstraintBatchIds, srcConstraintBatchIds, conInfos, numConstraintRows);
  300. int grainSize = 600;
  301. btParallelFor(0, numConstraints, grainSize, loop);
  302. }
  303. static void initBatchedConstraintInfoArray(btAlignedObjectArray<btBatchedConstraintInfo>* outConInfos, btConstraintArray* constraints)
  304. {
  305. BT_PROFILE("initBatchedConstraintInfoArray");
  306. btAlignedObjectArray<btBatchedConstraintInfo>& conInfos = *outConInfos;
  307. int numConstraints = constraints->size();
  308. conInfos.resizeNoInitialize(numConstraints);
  309. int newSize = initBatchedConstraintInfo(&outConInfos->at(0), constraints);
  310. conInfos.resizeNoInitialize(newSize);
  311. }
  312. static void mergeSmallBatches(btBatchInfo* batches, int iBeginBatch, int iEndBatch, int minBatchSize, int maxBatchSize)
  313. {
  314. BT_PROFILE("mergeSmallBatches");
  315. for (int iBatch = iEndBatch - 1; iBatch >= iBeginBatch; --iBatch)
  316. {
  317. btBatchInfo& batch = batches[iBatch];
  318. if (batch.mergeIndex == kNoMerge && batch.numConstraints > 0 && batch.numConstraints < minBatchSize)
  319. {
  320. for (int iDestBatch = iBatch - 1; iDestBatch >= iBeginBatch; --iDestBatch)
  321. {
  322. btBatchInfo& destBatch = batches[iDestBatch];
  323. if (destBatch.mergeIndex == kNoMerge && (destBatch.numConstraints + batch.numConstraints) < maxBatchSize)
  324. {
  325. destBatch.numConstraints += batch.numConstraints;
  326. batch.numConstraints = 0;
  327. batch.mergeIndex = iDestBatch;
  328. break;
  329. }
  330. }
  331. }
  332. }
  333. // flatten mergeIndexes
  334. // e.g. in case where A was merged into B and then B was merged into C, we need A to point to C instead of B
  335. // Note: loop goes forward through batches because batches always merge from higher indexes to lower,
  336. // so by going from low to high it reduces the amount of trail-following
  337. for (int iBatch = iBeginBatch; iBatch < iEndBatch; ++iBatch)
  338. {
  339. btBatchInfo& batch = batches[iBatch];
  340. if (batch.mergeIndex != kNoMerge)
  341. {
  342. int iMergeDest = batches[batch.mergeIndex].mergeIndex;
  343. // follow trail of merges to the end
  344. while (iMergeDest != kNoMerge)
  345. {
  346. int iNext = batches[iMergeDest].mergeIndex;
  347. if (iNext == kNoMerge)
  348. {
  349. batch.mergeIndex = iMergeDest;
  350. break;
  351. }
  352. iMergeDest = iNext;
  353. }
  354. }
  355. }
  356. }
  357. static void updateConstraintBatchIdsForMerges(int* constraintBatchIds, int numConstraints, const btBatchInfo* batches, int numBatches)
  358. {
  359. BT_PROFILE("updateConstraintBatchIdsForMerges");
  360. // update batchIds to account for merges
  361. for (int i = 0; i < numConstraints; ++i)
  362. {
  363. int iBatch = constraintBatchIds[i];
  364. btAssert(iBatch < numBatches);
  365. // if this constraint references a batch that was merged into another batch
  366. if (batches[iBatch].mergeIndex != kNoMerge)
  367. {
  368. // update batchId
  369. constraintBatchIds[i] = batches[iBatch].mergeIndex;
  370. }
  371. }
  372. }
  373. struct UpdateConstraintBatchIdsForMergesLoop : public btIParallelForBody
  374. {
  375. int* m_constraintBatchIds;
  376. const btBatchInfo* m_batches;
  377. int m_numBatches;
  378. UpdateConstraintBatchIdsForMergesLoop(int* constraintBatchIds, const btBatchInfo* batches, int numBatches)
  379. {
  380. m_constraintBatchIds = constraintBatchIds;
  381. m_batches = batches;
  382. m_numBatches = numBatches;
  383. }
  384. void forLoop(int iBegin, int iEnd) const BT_OVERRIDE
  385. {
  386. BT_PROFILE("UpdateConstraintBatchIdsForMergesLoop");
  387. updateConstraintBatchIdsForMerges(m_constraintBatchIds + iBegin, iEnd - iBegin, m_batches, m_numBatches);
  388. }
  389. };
  390. static void updateConstraintBatchIdsForMergesMt(int* constraintBatchIds, int numConstraints, const btBatchInfo* batches, int numBatches)
  391. {
  392. BT_PROFILE("updateConstraintBatchIdsForMergesMt");
  393. UpdateConstraintBatchIdsForMergesLoop loop(constraintBatchIds, batches, numBatches);
  394. int grainSize = 800;
  395. btParallelFor(0, numConstraints, grainSize, loop);
  396. }
  397. inline bool BatchCompare(const btBatchedConstraints::Range& a, const btBatchedConstraints::Range& b)
  398. {
  399. int lenA = a.end - a.begin;
  400. int lenB = b.end - b.begin;
  401. return lenA > lenB;
  402. }
  403. static void writeOutConstraintIndicesForRangeOfBatches(btBatchedConstraints* bc,
  404. const int* constraintBatchIds,
  405. int numConstraints,
  406. int* constraintIdPerBatch,
  407. int batchBegin,
  408. int batchEnd)
  409. {
  410. BT_PROFILE("writeOutConstraintIndicesForRangeOfBatches");
  411. for (int iCon = 0; iCon < numConstraints; ++iCon)
  412. {
  413. int iBatch = constraintBatchIds[iCon];
  414. if (iBatch >= batchBegin && iBatch < batchEnd)
  415. {
  416. int iDestCon = constraintIdPerBatch[iBatch];
  417. constraintIdPerBatch[iBatch] = iDestCon + 1;
  418. bc->m_constraintIndices[iDestCon] = iCon;
  419. }
  420. }
  421. }
  422. struct WriteOutConstraintIndicesLoop : public btIParallelForBody
  423. {
  424. btBatchedConstraints* m_batchedConstraints;
  425. const int* m_constraintBatchIds;
  426. int m_numConstraints;
  427. int* m_constraintIdPerBatch;
  428. int m_maxNumBatchesPerPhase;
  429. WriteOutConstraintIndicesLoop(btBatchedConstraints* bc, const int* constraintBatchIds, int numConstraints, int* constraintIdPerBatch, int maxNumBatchesPerPhase)
  430. {
  431. m_batchedConstraints = bc;
  432. m_constraintBatchIds = constraintBatchIds;
  433. m_numConstraints = numConstraints;
  434. m_constraintIdPerBatch = constraintIdPerBatch;
  435. m_maxNumBatchesPerPhase = maxNumBatchesPerPhase;
  436. }
  437. void forLoop(int iBegin, int iEnd) const BT_OVERRIDE
  438. {
  439. BT_PROFILE("WriteOutConstraintIndicesLoop");
  440. int batchBegin = iBegin * m_maxNumBatchesPerPhase;
  441. int batchEnd = iEnd * m_maxNumBatchesPerPhase;
  442. writeOutConstraintIndicesForRangeOfBatches(m_batchedConstraints,
  443. m_constraintBatchIds,
  444. m_numConstraints,
  445. m_constraintIdPerBatch,
  446. batchBegin,
  447. batchEnd);
  448. }
  449. };
  450. static void writeOutConstraintIndicesMt(btBatchedConstraints* bc,
  451. const int* constraintBatchIds,
  452. int numConstraints,
  453. int* constraintIdPerBatch,
  454. int maxNumBatchesPerPhase,
  455. int numPhases)
  456. {
  457. BT_PROFILE("writeOutConstraintIndicesMt");
  458. bool inParallel = true;
  459. if (inParallel)
  460. {
  461. WriteOutConstraintIndicesLoop loop(bc, constraintBatchIds, numConstraints, constraintIdPerBatch, maxNumBatchesPerPhase);
  462. btParallelFor(0, numPhases, 1, loop);
  463. }
  464. else
  465. {
  466. for (int iCon = 0; iCon < numConstraints; ++iCon)
  467. {
  468. int iBatch = constraintBatchIds[iCon];
  469. int iDestCon = constraintIdPerBatch[iBatch];
  470. constraintIdPerBatch[iBatch] = iDestCon + 1;
  471. bc->m_constraintIndices[iDestCon] = iCon;
  472. }
  473. }
  474. }
  475. static void writeGrainSizes(btBatchedConstraints* bc)
  476. {
  477. typedef btBatchedConstraints::Range Range;
  478. int numPhases = bc->m_phases.size();
  479. bc->m_phaseGrainSize.resizeNoInitialize(numPhases);
  480. int numThreads = btGetTaskScheduler()->getNumThreads();
  481. for (int iPhase = 0; iPhase < numPhases; ++iPhase)
  482. {
  483. const Range& phase = bc->m_phases[iPhase];
  484. int numBatches = phase.end - phase.begin;
  485. float grainSize = floor((0.25f * numBatches / float(numThreads)) + 0.0f);
  486. bc->m_phaseGrainSize[iPhase] = btMax(1, int(grainSize));
  487. }
  488. }
  489. static void writeOutBatches(btBatchedConstraints* bc,
  490. const int* constraintBatchIds,
  491. int numConstraints,
  492. const btBatchInfo* batches,
  493. int* batchWork,
  494. int maxNumBatchesPerPhase,
  495. int numPhases)
  496. {
  497. BT_PROFILE("writeOutBatches");
  498. typedef btBatchedConstraints::Range Range;
  499. bc->m_constraintIndices.reserve(numConstraints);
  500. bc->m_batches.resizeNoInitialize(0);
  501. bc->m_phases.resizeNoInitialize(0);
  502. //int maxNumBatches = numPhases * maxNumBatchesPerPhase;
  503. {
  504. int* constraintIdPerBatch = batchWork; // for each batch, keep an index into the next available slot in the m_constraintIndices array
  505. int iConstraint = 0;
  506. for (int iPhase = 0; iPhase < numPhases; ++iPhase)
  507. {
  508. int curPhaseBegin = bc->m_batches.size();
  509. int iBegin = iPhase * maxNumBatchesPerPhase;
  510. int iEnd = iBegin + maxNumBatchesPerPhase;
  511. for (int i = iBegin; i < iEnd; ++i)
  512. {
  513. const btBatchInfo& batch = batches[i];
  514. int curBatchBegin = iConstraint;
  515. constraintIdPerBatch[i] = curBatchBegin; // record the start of each batch in m_constraintIndices array
  516. int numConstraints = batch.numConstraints;
  517. iConstraint += numConstraints;
  518. if (numConstraints > 0)
  519. {
  520. bc->m_batches.push_back(Range(curBatchBegin, iConstraint));
  521. }
  522. }
  523. // if any batches were emitted this phase,
  524. if (bc->m_batches.size() > curPhaseBegin)
  525. {
  526. // output phase
  527. bc->m_phases.push_back(Range(curPhaseBegin, bc->m_batches.size()));
  528. }
  529. }
  530. btAssert(iConstraint == numConstraints);
  531. bc->m_constraintIndices.resizeNoInitialize(numConstraints);
  532. writeOutConstraintIndicesMt(bc, constraintBatchIds, numConstraints, constraintIdPerBatch, maxNumBatchesPerPhase, numPhases);
  533. }
  534. // for each phase
  535. for (int iPhase = 0; iPhase < bc->m_phases.size(); ++iPhase)
  536. {
  537. // sort the batches from largest to smallest (can be helpful to some task schedulers)
  538. const Range& curBatches = bc->m_phases[iPhase];
  539. bc->m_batches.quickSortInternal(BatchCompare, curBatches.begin, curBatches.end - 1);
  540. }
  541. bc->m_phaseOrder.resize(bc->m_phases.size());
  542. for (int i = 0; i < bc->m_phases.size(); ++i)
  543. {
  544. bc->m_phaseOrder[i] = i;
  545. }
  546. writeGrainSizes(bc);
  547. }
  548. //
  549. // PreallocatedMemoryHelper -- helper object for allocating a number of chunks of memory in a single contiguous block.
  550. // It is generally more efficient to do a single larger allocation than many smaller allocations.
  551. //
  552. // Example Usage:
  553. //
  554. // btVector3* bodyPositions = NULL;
  555. // btBatchedConstraintInfo* conInfos = NULL;
  556. // {
  557. // PreallocatedMemoryHelper<8> memHelper;
  558. // memHelper.addChunk( (void**) &bodyPositions, sizeof( btVector3 ) * bodies.size() );
  559. // memHelper.addChunk( (void**) &conInfos, sizeof( btBatchedConstraintInfo ) * numConstraints );
  560. // void* memPtr = malloc( memHelper.getSizeToAllocate() ); // allocate the memory
  561. // memHelper.setChunkPointers( memPtr ); // update pointers to chunks
  562. // }
  563. template <int N>
  564. class PreallocatedMemoryHelper
  565. {
  566. struct Chunk
  567. {
  568. void** ptr;
  569. size_t size;
  570. };
  571. Chunk m_chunks[N];
  572. int m_numChunks;
  573. public:
  574. PreallocatedMemoryHelper() { m_numChunks = 0; }
  575. void addChunk(void** ptr, size_t sz)
  576. {
  577. btAssert(m_numChunks < N);
  578. if (m_numChunks < N)
  579. {
  580. Chunk& chunk = m_chunks[m_numChunks];
  581. chunk.ptr = ptr;
  582. chunk.size = sz;
  583. m_numChunks++;
  584. }
  585. }
  586. size_t getSizeToAllocate() const
  587. {
  588. size_t totalSize = 0;
  589. for (int i = 0; i < m_numChunks; ++i)
  590. {
  591. totalSize += m_chunks[i].size;
  592. }
  593. return totalSize;
  594. }
  595. void setChunkPointers(void* mem) const
  596. {
  597. size_t totalSize = 0;
  598. for (int i = 0; i < m_numChunks; ++i)
  599. {
  600. const Chunk& chunk = m_chunks[i];
  601. char* chunkPtr = static_cast<char*>(mem) + totalSize;
  602. *chunk.ptr = chunkPtr;
  603. totalSize += chunk.size;
  604. }
  605. }
  606. };
  607. static btVector3 findMaxDynamicConstraintExtent(
  608. btVector3* bodyPositions,
  609. bool* bodyDynamicFlags,
  610. btBatchedConstraintInfo* conInfos,
  611. int numConstraints,
  612. int numBodies)
  613. {
  614. BT_PROFILE("findMaxDynamicConstraintExtent");
  615. btVector3 consExtent = btVector3(1, 1, 1) * 0.001;
  616. for (int iCon = 0; iCon < numConstraints; ++iCon)
  617. {
  618. const btBatchedConstraintInfo& con = conInfos[iCon];
  619. int iBody0 = con.bodyIds[0];
  620. int iBody1 = con.bodyIds[1];
  621. btAssert(iBody0 >= 0 && iBody0 < numBodies);
  622. btAssert(iBody1 >= 0 && iBody1 < numBodies);
  623. // is it a dynamic constraint?
  624. if (bodyDynamicFlags[iBody0] && bodyDynamicFlags[iBody1])
  625. {
  626. btVector3 delta = bodyPositions[iBody1] - bodyPositions[iBody0];
  627. consExtent.setMax(delta.absolute());
  628. }
  629. }
  630. return consExtent;
  631. }
  632. struct btIntVec3
  633. {
  634. int m_ints[3];
  635. SIMD_FORCE_INLINE const int& operator[](int i) const { return m_ints[i]; }
  636. SIMD_FORCE_INLINE int& operator[](int i) { return m_ints[i]; }
  637. };
  638. struct AssignConstraintsToGridBatchesParams
  639. {
  640. bool* bodyDynamicFlags;
  641. btIntVec3* bodyGridCoords;
  642. int numBodies;
  643. btBatchedConstraintInfo* conInfos;
  644. int* constraintBatchIds;
  645. btIntVec3 gridChunkDim;
  646. int maxNumBatchesPerPhase;
  647. int numPhases;
  648. int phaseMask;
  649. AssignConstraintsToGridBatchesParams()
  650. {
  651. memset(this, 0, sizeof(*this));
  652. }
  653. };
  654. static void assignConstraintsToGridBatches(const AssignConstraintsToGridBatchesParams& params, int iConBegin, int iConEnd)
  655. {
  656. BT_PROFILE("assignConstraintsToGridBatches");
  657. // (can be done in parallel)
  658. for (int iCon = iConBegin; iCon < iConEnd; ++iCon)
  659. {
  660. const btBatchedConstraintInfo& con = params.conInfos[iCon];
  661. int iBody0 = con.bodyIds[0];
  662. int iBody1 = con.bodyIds[1];
  663. int iPhase = iCon; //iBody0; // pseudorandom choice to distribute evenly amongst phases
  664. iPhase &= params.phaseMask;
  665. int gridCoord[3];
  666. // is it a dynamic constraint?
  667. if (params.bodyDynamicFlags[iBody0] && params.bodyDynamicFlags[iBody1])
  668. {
  669. const btIntVec3& body0Coords = params.bodyGridCoords[iBody0];
  670. const btIntVec3& body1Coords = params.bodyGridCoords[iBody1];
  671. // for each dimension x,y,z,
  672. for (int i = 0; i < 3; ++i)
  673. {
  674. int coordMin = btMin(body0Coords.m_ints[i], body1Coords.m_ints[i]);
  675. int coordMax = btMax(body0Coords.m_ints[i], body1Coords.m_ints[i]);
  676. if (coordMin != coordMax)
  677. {
  678. btAssert(coordMax == coordMin + 1);
  679. if ((coordMin & 1) == 0)
  680. {
  681. iPhase &= ~(1 << i); // force bit off
  682. }
  683. else
  684. {
  685. iPhase |= (1 << i); // force bit on
  686. iPhase &= params.phaseMask;
  687. }
  688. }
  689. gridCoord[i] = coordMin;
  690. }
  691. }
  692. else
  693. {
  694. if (!params.bodyDynamicFlags[iBody0])
  695. {
  696. iBody0 = con.bodyIds[1];
  697. }
  698. btAssert(params.bodyDynamicFlags[iBody0]);
  699. const btIntVec3& body0Coords = params.bodyGridCoords[iBody0];
  700. // for each dimension x,y,z,
  701. for (int i = 0; i < 3; ++i)
  702. {
  703. gridCoord[i] = body0Coords.m_ints[i];
  704. }
  705. }
  706. // calculate chunk coordinates
  707. int chunkCoord[3];
  708. btIntVec3 gridChunkDim = params.gridChunkDim;
  709. // for each dimension x,y,z,
  710. for (int i = 0; i < 3; ++i)
  711. {
  712. int coordOffset = (iPhase >> i) & 1;
  713. chunkCoord[i] = (gridCoord[i] - coordOffset) / 2;
  714. btClamp(chunkCoord[i], 0, gridChunkDim[i] - 1);
  715. btAssert(chunkCoord[i] < gridChunkDim[i]);
  716. }
  717. int iBatch = iPhase * params.maxNumBatchesPerPhase + chunkCoord[0] + chunkCoord[1] * gridChunkDim[0] + chunkCoord[2] * gridChunkDim[0] * gridChunkDim[1];
  718. btAssert(iBatch >= 0 && iBatch < params.maxNumBatchesPerPhase * params.numPhases);
  719. params.constraintBatchIds[iCon] = iBatch;
  720. }
  721. }
  722. struct AssignConstraintsToGridBatchesLoop : public btIParallelForBody
  723. {
  724. const AssignConstraintsToGridBatchesParams* m_params;
  725. AssignConstraintsToGridBatchesLoop(const AssignConstraintsToGridBatchesParams& params)
  726. {
  727. m_params = &params;
  728. }
  729. void forLoop(int iBegin, int iEnd) const BT_OVERRIDE
  730. {
  731. assignConstraintsToGridBatches(*m_params, iBegin, iEnd);
  732. }
  733. };
  734. //
  735. // setupSpatialGridBatchesMt -- generate batches using a uniform 3D grid
  736. //
  737. /*
  738. Bodies are treated as 3D points at their center of mass. We only consider dynamic bodies at this stage,
  739. because only dynamic bodies are mutated when a constraint is solved, thus subject to race conditions.
  740. 1. Compute a bounding box around all dynamic bodies
  741. 2. Compute the maximum extent of all dynamic constraints. Each dynamic constraint is treated as a line segment, and we need the size of
  742. box that will fully enclose any single dynamic constraint
  743. 3. Establish the cell size of our grid, the cell size in each dimension must be at least as large as the dynamic constraints max-extent,
  744. so that no dynamic constraint can span more than 2 cells of our grid on any axis of the grid. The cell size should be adjusted
  745. larger in order to keep the total number of cells from being excessively high
  746. Key idea: Given that each constraint spans 1 or 2 grid cells in each dimension, we can handle all constraints by processing
  747. in chunks of 2x2x2 cells with 8 different 1-cell offsets ((0,0,0),(0,0,1),(0,1,0),(0,1,1),(1,0,0)...).
  748. For each of the 8 offsets, we create a phase, and for each 2x2x2 chunk with dynamic constraints becomes a batch in that phase.
  749. 4. Once the grid is established, we can calculate for each constraint which phase and batch it belongs in.
  750. 5. Do a merge small batches on the batches of each phase separately, to try to even out the sizes of batches
  751. Optionally, we can "collapse" one dimension of our 3D grid to turn it into a 2D grid, which reduces the number of phases
  752. to 4. With fewer phases, there are more constraints per phase and this makes it easier to create batches of a useful size.
  753. */
  754. //
  755. static void setupSpatialGridBatchesMt(
  756. btBatchedConstraints* batchedConstraints,
  757. btAlignedObjectArray<char>* scratchMemory,
  758. btConstraintArray* constraints,
  759. const btAlignedObjectArray<btSolverBody>& bodies,
  760. int minBatchSize,
  761. int maxBatchSize,
  762. bool use2DGrid)
  763. {
  764. BT_PROFILE("setupSpatialGridBatchesMt");
  765. const int numPhases = 8;
  766. int numConstraints = constraints->size();
  767. int numConstraintRows = constraints->size();
  768. const int maxGridChunkCount = 128;
  769. int allocNumBatchesPerPhase = maxGridChunkCount;
  770. int minNumBatchesPerPhase = 16;
  771. int allocNumBatches = allocNumBatchesPerPhase * numPhases;
  772. btVector3* bodyPositions = NULL;
  773. bool* bodyDynamicFlags = NULL;
  774. btIntVec3* bodyGridCoords = NULL;
  775. btBatchInfo* batches = NULL;
  776. int* batchWork = NULL;
  777. btBatchedConstraintInfo* conInfos = NULL;
  778. int* constraintBatchIds = NULL;
  779. int* constraintRowBatchIds = NULL;
  780. {
  781. PreallocatedMemoryHelper<10> memHelper;
  782. memHelper.addChunk((void**)&bodyPositions, sizeof(btVector3) * bodies.size());
  783. memHelper.addChunk((void**)&bodyDynamicFlags, sizeof(bool) * bodies.size());
  784. memHelper.addChunk((void**)&bodyGridCoords, sizeof(btIntVec3) * bodies.size());
  785. memHelper.addChunk((void**)&batches, sizeof(btBatchInfo) * allocNumBatches);
  786. memHelper.addChunk((void**)&batchWork, sizeof(int) * allocNumBatches);
  787. memHelper.addChunk((void**)&conInfos, sizeof(btBatchedConstraintInfo) * numConstraints);
  788. memHelper.addChunk((void**)&constraintBatchIds, sizeof(int) * numConstraints);
  789. memHelper.addChunk((void**)&constraintRowBatchIds, sizeof(int) * numConstraintRows);
  790. size_t scratchSize = memHelper.getSizeToAllocate();
  791. // if we need to reallocate
  792. if (scratchMemory->capacity() < scratchSize)
  793. {
  794. // allocate 6.25% extra to avoid repeated reallocs
  795. scratchMemory->reserve(scratchSize + scratchSize / 16);
  796. }
  797. scratchMemory->resizeNoInitialize(scratchSize);
  798. char* memPtr = &scratchMemory->at(0);
  799. memHelper.setChunkPointers(memPtr);
  800. }
  801. numConstraints = initBatchedConstraintInfo(conInfos, constraints);
  802. // compute bounding box around all dynamic bodies
  803. // (could be done in parallel)
  804. btVector3 bboxMin(BT_LARGE_FLOAT, BT_LARGE_FLOAT, BT_LARGE_FLOAT);
  805. btVector3 bboxMax = -bboxMin;
  806. //int dynamicBodyCount = 0;
  807. for (int i = 0; i < bodies.size(); ++i)
  808. {
  809. const btSolverBody& body = bodies[i];
  810. btVector3 bodyPos = body.getWorldTransform().getOrigin();
  811. bool isDynamic = (body.internalGetInvMass().x() > btScalar(0));
  812. bodyPositions[i] = bodyPos;
  813. bodyDynamicFlags[i] = isDynamic;
  814. if (isDynamic)
  815. {
  816. //dynamicBodyCount++;
  817. bboxMin.setMin(bodyPos);
  818. bboxMax.setMax(bodyPos);
  819. }
  820. }
  821. // find max extent of all dynamic constraints
  822. // (could be done in parallel)
  823. btVector3 consExtent = findMaxDynamicConstraintExtent(bodyPositions, bodyDynamicFlags, conInfos, numConstraints, bodies.size());
  824. btVector3 gridExtent = bboxMax - bboxMin;
  825. btVector3 gridCellSize = consExtent;
  826. int gridDim[3];
  827. gridDim[0] = int(1.0 + gridExtent.x() / gridCellSize.x());
  828. gridDim[1] = int(1.0 + gridExtent.y() / gridCellSize.y());
  829. gridDim[2] = int(1.0 + gridExtent.z() / gridCellSize.z());
  830. // if we can collapse an axis, it will cut our number of phases in half which could be more efficient
  831. int phaseMask = 7;
  832. bool collapseAxis = use2DGrid;
  833. if (collapseAxis)
  834. {
  835. // pick the smallest axis to collapse, leaving us with the greatest number of cells in our grid
  836. int iAxisToCollapse = 0;
  837. int axisDim = gridDim[iAxisToCollapse];
  838. //for each dimension
  839. for (int i = 0; i < 3; ++i)
  840. {
  841. if (gridDim[i] < axisDim)
  842. {
  843. iAxisToCollapse = i;
  844. axisDim = gridDim[i];
  845. }
  846. }
  847. // collapse it
  848. gridCellSize[iAxisToCollapse] = gridExtent[iAxisToCollapse] * 2.0f;
  849. phaseMask &= ~(1 << iAxisToCollapse);
  850. }
  851. int numGridChunks = 0;
  852. btIntVec3 gridChunkDim; // each chunk is 2x2x2 group of cells
  853. while (true)
  854. {
  855. gridDim[0] = int(1.0 + gridExtent.x() / gridCellSize.x());
  856. gridDim[1] = int(1.0 + gridExtent.y() / gridCellSize.y());
  857. gridDim[2] = int(1.0 + gridExtent.z() / gridCellSize.z());
  858. gridChunkDim[0] = btMax(1, (gridDim[0] + 0) / 2);
  859. gridChunkDim[1] = btMax(1, (gridDim[1] + 0) / 2);
  860. gridChunkDim[2] = btMax(1, (gridDim[2] + 0) / 2);
  861. numGridChunks = gridChunkDim[0] * gridChunkDim[1] * gridChunkDim[2];
  862. float nChunks = float(gridChunkDim[0]) * float(gridChunkDim[1]) * float(gridChunkDim[2]); // suceptible to integer overflow
  863. if (numGridChunks <= maxGridChunkCount && nChunks <= maxGridChunkCount)
  864. {
  865. break;
  866. }
  867. gridCellSize *= 1.25; // should roughly cut numCells in half
  868. }
  869. btAssert(numGridChunks <= maxGridChunkCount);
  870. int maxNumBatchesPerPhase = numGridChunks;
  871. // for each dynamic body, compute grid coords
  872. btVector3 invGridCellSize = btVector3(1, 1, 1) / gridCellSize;
  873. // (can be done in parallel)
  874. for (int iBody = 0; iBody < bodies.size(); ++iBody)
  875. {
  876. btIntVec3& coords = bodyGridCoords[iBody];
  877. if (bodyDynamicFlags[iBody])
  878. {
  879. btVector3 v = (bodyPositions[iBody] - bboxMin) * invGridCellSize;
  880. coords.m_ints[0] = int(v.x());
  881. coords.m_ints[1] = int(v.y());
  882. coords.m_ints[2] = int(v.z());
  883. btAssert(coords.m_ints[0] >= 0 && coords.m_ints[0] < gridDim[0]);
  884. btAssert(coords.m_ints[1] >= 0 && coords.m_ints[1] < gridDim[1]);
  885. btAssert(coords.m_ints[2] >= 0 && coords.m_ints[2] < gridDim[2]);
  886. }
  887. else
  888. {
  889. coords.m_ints[0] = -1;
  890. coords.m_ints[1] = -1;
  891. coords.m_ints[2] = -1;
  892. }
  893. }
  894. for (int iPhase = 0; iPhase < numPhases; ++iPhase)
  895. {
  896. int batchBegin = iPhase * maxNumBatchesPerPhase;
  897. int batchEnd = batchBegin + maxNumBatchesPerPhase;
  898. for (int iBatch = batchBegin; iBatch < batchEnd; ++iBatch)
  899. {
  900. btBatchInfo& batch = batches[iBatch];
  901. batch = btBatchInfo();
  902. }
  903. }
  904. {
  905. AssignConstraintsToGridBatchesParams params;
  906. params.bodyDynamicFlags = bodyDynamicFlags;
  907. params.bodyGridCoords = bodyGridCoords;
  908. params.numBodies = bodies.size();
  909. params.conInfos = conInfos;
  910. params.constraintBatchIds = constraintBatchIds;
  911. params.gridChunkDim = gridChunkDim;
  912. params.maxNumBatchesPerPhase = maxNumBatchesPerPhase;
  913. params.numPhases = numPhases;
  914. params.phaseMask = phaseMask;
  915. bool inParallel = true;
  916. if (inParallel)
  917. {
  918. AssignConstraintsToGridBatchesLoop loop(params);
  919. int grainSize = 250;
  920. btParallelFor(0, numConstraints, grainSize, loop);
  921. }
  922. else
  923. {
  924. assignConstraintsToGridBatches(params, 0, numConstraints);
  925. }
  926. }
  927. for (int iCon = 0; iCon < numConstraints; ++iCon)
  928. {
  929. const btBatchedConstraintInfo& con = conInfos[iCon];
  930. int iBatch = constraintBatchIds[iCon];
  931. btBatchInfo& batch = batches[iBatch];
  932. batch.numConstraints += con.numConstraintRows;
  933. }
  934. for (int iPhase = 0; iPhase < numPhases; ++iPhase)
  935. {
  936. // if phase is legit,
  937. if (iPhase == (iPhase & phaseMask))
  938. {
  939. int iBeginBatch = iPhase * maxNumBatchesPerPhase;
  940. int iEndBatch = iBeginBatch + maxNumBatchesPerPhase;
  941. mergeSmallBatches(batches, iBeginBatch, iEndBatch, minBatchSize, maxBatchSize);
  942. }
  943. }
  944. // all constraints have been assigned a batchId
  945. updateConstraintBatchIdsForMergesMt(constraintBatchIds, numConstraints, batches, maxNumBatchesPerPhase * numPhases);
  946. if (numConstraintRows > numConstraints)
  947. {
  948. expandConstraintRowsMt(&constraintRowBatchIds[0], &constraintBatchIds[0], &conInfos[0], numConstraints, numConstraintRows);
  949. }
  950. else
  951. {
  952. constraintRowBatchIds = constraintBatchIds;
  953. }
  954. writeOutBatches(batchedConstraints, constraintRowBatchIds, numConstraintRows, batches, batchWork, maxNumBatchesPerPhase, numPhases);
  955. btAssert(batchedConstraints->validate(constraints, bodies));
  956. }
  957. static void setupSingleBatch(
  958. btBatchedConstraints* bc,
  959. int numConstraints)
  960. {
  961. BT_PROFILE("setupSingleBatch");
  962. typedef btBatchedConstraints::Range Range;
  963. bc->m_constraintIndices.resize(numConstraints);
  964. for (int i = 0; i < numConstraints; ++i)
  965. {
  966. bc->m_constraintIndices[i] = i;
  967. }
  968. bc->m_batches.resizeNoInitialize(0);
  969. bc->m_phases.resizeNoInitialize(0);
  970. bc->m_phaseOrder.resizeNoInitialize(0);
  971. bc->m_phaseGrainSize.resizeNoInitialize(0);
  972. if (numConstraints > 0)
  973. {
  974. bc->m_batches.push_back(Range(0, numConstraints));
  975. bc->m_phases.push_back(Range(0, 1));
  976. bc->m_phaseOrder.push_back(0);
  977. bc->m_phaseGrainSize.push_back(1);
  978. }
  979. }
  980. void btBatchedConstraints::setup(
  981. btConstraintArray* constraints,
  982. const btAlignedObjectArray<btSolverBody>& bodies,
  983. BatchingMethod batchingMethod,
  984. int minBatchSize,
  985. int maxBatchSize,
  986. btAlignedObjectArray<char>* scratchMemory)
  987. {
  988. if (constraints->size() >= minBatchSize * 4)
  989. {
  990. bool use2DGrid = batchingMethod == BATCHING_METHOD_SPATIAL_GRID_2D;
  991. setupSpatialGridBatchesMt(this, scratchMemory, constraints, bodies, minBatchSize, maxBatchSize, use2DGrid);
  992. if (s_debugDrawBatches)
  993. {
  994. debugDrawAllBatches(this, constraints, bodies);
  995. }
  996. }
  997. else
  998. {
  999. setupSingleBatch(this, constraints->size());
  1000. }
  1001. }