b3GpuGridBroadphase.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. #include "b3GpuGridBroadphase.h"
  2. #include "Bullet3Geometry/b3AabbUtil.h"
  3. #include "kernels/gridBroadphaseKernels.h"
  4. #include "kernels/sapKernels.h"
  5. //#include "kernels/gridBroadphase.cl"
  6. #include "Bullet3OpenCL/Initialize/b3OpenCLUtils.h"
  7. #include "Bullet3OpenCL/ParallelPrimitives/b3LauncherCL.h"
  8. #define B3_BROADPHASE_SAP_PATH "src/Bullet3OpenCL/BroadphaseCollision/kernels/sap.cl"
  9. #define B3_GRID_BROADPHASE_PATH "src/Bullet3OpenCL/BroadphaseCollision/kernels/gridBroadphase.cl"
  10. cl_kernel kCalcHashAABB;
  11. cl_kernel kClearCellStart;
  12. cl_kernel kFindCellStart;
  13. cl_kernel kFindOverlappingPairs;
  14. cl_kernel m_copyAabbsKernel;
  15. cl_kernel m_sap2Kernel;
  16. //int maxPairsPerBody = 64;
  17. int maxBodiesPerCell = 256; //??
  18. b3GpuGridBroadphase::b3GpuGridBroadphase(cl_context ctx, cl_device_id device, cl_command_queue q)
  19. : m_context(ctx),
  20. m_device(device),
  21. m_queue(q),
  22. m_allAabbsGPU1(ctx, q),
  23. m_smallAabbsMappingGPU(ctx, q),
  24. m_largeAabbsMappingGPU(ctx, q),
  25. m_gpuPairs(ctx, q),
  26. m_hashGpu(ctx, q),
  27. m_cellStartGpu(ctx, q),
  28. m_paramsGPU(ctx, q)
  29. {
  30. b3Vector3 gridSize = b3MakeVector3(3, 3, 3);
  31. b3Vector3 invGridSize = b3MakeVector3(1.f / gridSize[0], 1.f / gridSize[1], 1.f / gridSize[2]);
  32. m_paramsCPU.m_gridSize[0] = 128;
  33. m_paramsCPU.m_gridSize[1] = 128;
  34. m_paramsCPU.m_gridSize[2] = 128;
  35. m_paramsCPU.m_gridSize[3] = maxBodiesPerCell;
  36. m_paramsCPU.setMaxBodiesPerCell(maxBodiesPerCell);
  37. m_paramsCPU.m_invCellSize[0] = invGridSize[0];
  38. m_paramsCPU.m_invCellSize[1] = invGridSize[1];
  39. m_paramsCPU.m_invCellSize[2] = invGridSize[2];
  40. m_paramsCPU.m_invCellSize[3] = 0.f;
  41. m_paramsGPU.push_back(m_paramsCPU);
  42. cl_int errNum = 0;
  43. {
  44. const char* sapSrc = sapCL;
  45. cl_program sapProg = b3OpenCLUtils::compileCLProgramFromString(m_context, m_device, sapSrc, &errNum, "", B3_BROADPHASE_SAP_PATH);
  46. b3Assert(errNum == CL_SUCCESS);
  47. m_copyAabbsKernel = b3OpenCLUtils::compileCLKernelFromString(m_context, m_device, sapSrc, "copyAabbsKernel", &errNum, sapProg);
  48. m_sap2Kernel = b3OpenCLUtils::compileCLKernelFromString(m_context, m_device, sapSrc, "computePairsKernelTwoArrays", &errNum, sapProg);
  49. b3Assert(errNum == CL_SUCCESS);
  50. }
  51. {
  52. cl_program gridProg = b3OpenCLUtils::compileCLProgramFromString(m_context, m_device, gridBroadphaseCL, &errNum, "", B3_GRID_BROADPHASE_PATH);
  53. b3Assert(errNum == CL_SUCCESS);
  54. kCalcHashAABB = b3OpenCLUtils::compileCLKernelFromString(m_context, m_device, gridBroadphaseCL, "kCalcHashAABB", &errNum, gridProg);
  55. b3Assert(errNum == CL_SUCCESS);
  56. kClearCellStart = b3OpenCLUtils::compileCLKernelFromString(m_context, m_device, gridBroadphaseCL, "kClearCellStart", &errNum, gridProg);
  57. b3Assert(errNum == CL_SUCCESS);
  58. kFindCellStart = b3OpenCLUtils::compileCLKernelFromString(m_context, m_device, gridBroadphaseCL, "kFindCellStart", &errNum, gridProg);
  59. b3Assert(errNum == CL_SUCCESS);
  60. kFindOverlappingPairs = b3OpenCLUtils::compileCLKernelFromString(m_context, m_device, gridBroadphaseCL, "kFindOverlappingPairs", &errNum, gridProg);
  61. b3Assert(errNum == CL_SUCCESS);
  62. }
  63. m_sorter = new b3RadixSort32CL(m_context, m_device, m_queue);
  64. }
  65. b3GpuGridBroadphase::~b3GpuGridBroadphase()
  66. {
  67. clReleaseKernel(kCalcHashAABB);
  68. clReleaseKernel(kClearCellStart);
  69. clReleaseKernel(kFindCellStart);
  70. clReleaseKernel(kFindOverlappingPairs);
  71. clReleaseKernel(m_sap2Kernel);
  72. clReleaseKernel(m_copyAabbsKernel);
  73. delete m_sorter;
  74. }
  75. void b3GpuGridBroadphase::createProxy(const b3Vector3& aabbMin, const b3Vector3& aabbMax, int userPtr, int collisionFilterGroup, int collisionFilterMask)
  76. {
  77. b3SapAabb aabb;
  78. aabb.m_minVec = aabbMin;
  79. aabb.m_maxVec = aabbMax;
  80. aabb.m_minIndices[3] = userPtr;
  81. aabb.m_signedMaxIndices[3] = m_allAabbsCPU1.size(); //NOT userPtr;
  82. m_smallAabbsMappingCPU.push_back(m_allAabbsCPU1.size());
  83. m_allAabbsCPU1.push_back(aabb);
  84. }
  85. void b3GpuGridBroadphase::createLargeProxy(const b3Vector3& aabbMin, const b3Vector3& aabbMax, int userPtr, int collisionFilterGroup, int collisionFilterMask)
  86. {
  87. b3SapAabb aabb;
  88. aabb.m_minVec = aabbMin;
  89. aabb.m_maxVec = aabbMax;
  90. aabb.m_minIndices[3] = userPtr;
  91. aabb.m_signedMaxIndices[3] = m_allAabbsCPU1.size(); //NOT userPtr;
  92. m_largeAabbsMappingCPU.push_back(m_allAabbsCPU1.size());
  93. m_allAabbsCPU1.push_back(aabb);
  94. }
  95. void b3GpuGridBroadphase::calculateOverlappingPairs(int maxPairs)
  96. {
  97. B3_PROFILE("b3GpuGridBroadphase::calculateOverlappingPairs");
  98. if (0)
  99. {
  100. calculateOverlappingPairsHost(maxPairs);
  101. /*
  102. b3AlignedObjectArray<b3Int4> cpuPairs;
  103. m_gpuPairs.copyToHost(cpuPairs);
  104. printf("host m_gpuPairs.size()=%d\n",m_gpuPairs.size());
  105. for (int i=0;i<m_gpuPairs.size();i++)
  106. {
  107. printf("host pair %d = %d,%d\n",i,cpuPairs[i].x,cpuPairs[i].y);
  108. }
  109. */
  110. return;
  111. }
  112. int numSmallAabbs = m_smallAabbsMappingGPU.size();
  113. b3OpenCLArray<int> pairCount(m_context, m_queue);
  114. pairCount.push_back(0);
  115. m_gpuPairs.resize(maxPairs); //numSmallAabbs*maxPairsPerBody);
  116. {
  117. int numLargeAabbs = m_largeAabbsMappingGPU.size();
  118. if (numLargeAabbs && numSmallAabbs)
  119. {
  120. B3_PROFILE("sap2Kernel");
  121. b3BufferInfoCL bInfo[] = {
  122. b3BufferInfoCL(m_allAabbsGPU1.getBufferCL()),
  123. b3BufferInfoCL(m_largeAabbsMappingGPU.getBufferCL()),
  124. b3BufferInfoCL(m_smallAabbsMappingGPU.getBufferCL()),
  125. b3BufferInfoCL(m_gpuPairs.getBufferCL()),
  126. b3BufferInfoCL(pairCount.getBufferCL())};
  127. b3LauncherCL launcher(m_queue, m_sap2Kernel, "m_sap2Kernel");
  128. launcher.setBuffers(bInfo, sizeof(bInfo) / sizeof(b3BufferInfoCL));
  129. launcher.setConst(numLargeAabbs);
  130. launcher.setConst(numSmallAabbs);
  131. launcher.setConst(0); //axis is not used
  132. launcher.setConst(maxPairs);
  133. //@todo: use actual maximum work item sizes of the device instead of hardcoded values
  134. launcher.launch2D(numLargeAabbs, numSmallAabbs, 4, 64);
  135. int numPairs = pairCount.at(0);
  136. if (numPairs > maxPairs)
  137. {
  138. b3Error("Error running out of pairs: numPairs = %d, maxPairs = %d.\n", numPairs, maxPairs);
  139. numPairs = maxPairs;
  140. }
  141. }
  142. }
  143. if (numSmallAabbs)
  144. {
  145. B3_PROFILE("gridKernel");
  146. m_hashGpu.resize(numSmallAabbs);
  147. {
  148. B3_PROFILE("kCalcHashAABB");
  149. b3LauncherCL launch(m_queue, kCalcHashAABB, "kCalcHashAABB");
  150. launch.setConst(numSmallAabbs);
  151. launch.setBuffer(m_allAabbsGPU1.getBufferCL());
  152. launch.setBuffer(m_smallAabbsMappingGPU.getBufferCL());
  153. launch.setBuffer(m_hashGpu.getBufferCL());
  154. launch.setBuffer(this->m_paramsGPU.getBufferCL());
  155. launch.launch1D(numSmallAabbs);
  156. }
  157. m_sorter->execute(m_hashGpu);
  158. int numCells = this->m_paramsCPU.m_gridSize[0] * this->m_paramsCPU.m_gridSize[1] * this->m_paramsCPU.m_gridSize[2];
  159. m_cellStartGpu.resize(numCells);
  160. //b3AlignedObjectArray<int > cellStartCpu;
  161. {
  162. B3_PROFILE("kClearCellStart");
  163. b3LauncherCL launch(m_queue, kClearCellStart, "kClearCellStart");
  164. launch.setConst(numCells);
  165. launch.setBuffer(m_cellStartGpu.getBufferCL());
  166. launch.launch1D(numCells);
  167. //m_cellStartGpu.copyToHost(cellStartCpu);
  168. //printf("??\n");
  169. }
  170. {
  171. B3_PROFILE("kFindCellStart");
  172. b3LauncherCL launch(m_queue, kFindCellStart, "kFindCellStart");
  173. launch.setConst(numSmallAabbs);
  174. launch.setBuffer(m_hashGpu.getBufferCL());
  175. launch.setBuffer(m_cellStartGpu.getBufferCL());
  176. launch.launch1D(numSmallAabbs);
  177. //m_cellStartGpu.copyToHost(cellStartCpu);
  178. //printf("??\n");
  179. }
  180. {
  181. B3_PROFILE("kFindOverlappingPairs");
  182. b3LauncherCL launch(m_queue, kFindOverlappingPairs, "kFindOverlappingPairs");
  183. launch.setConst(numSmallAabbs);
  184. launch.setBuffer(m_allAabbsGPU1.getBufferCL());
  185. launch.setBuffer(m_smallAabbsMappingGPU.getBufferCL());
  186. launch.setBuffer(m_hashGpu.getBufferCL());
  187. launch.setBuffer(m_cellStartGpu.getBufferCL());
  188. launch.setBuffer(m_paramsGPU.getBufferCL());
  189. //launch.setBuffer(0);
  190. launch.setBuffer(pairCount.getBufferCL());
  191. launch.setBuffer(m_gpuPairs.getBufferCL());
  192. launch.setConst(maxPairs);
  193. launch.launch1D(numSmallAabbs);
  194. int numPairs = pairCount.at(0);
  195. if (numPairs > maxPairs)
  196. {
  197. b3Error("Error running out of pairs: numPairs = %d, maxPairs = %d.\n", numPairs, maxPairs);
  198. numPairs = maxPairs;
  199. }
  200. m_gpuPairs.resize(numPairs);
  201. if (0)
  202. {
  203. b3AlignedObjectArray<b3Int4> pairsCpu;
  204. m_gpuPairs.copyToHost(pairsCpu);
  205. int sz = m_gpuPairs.size();
  206. printf("m_gpuPairs.size()=%d\n", sz);
  207. for (int i = 0; i < m_gpuPairs.size(); i++)
  208. {
  209. printf("pair %d = %d,%d\n", i, pairsCpu[i].x, pairsCpu[i].y);
  210. }
  211. printf("?!?\n");
  212. }
  213. }
  214. }
  215. //calculateOverlappingPairsHost(maxPairs);
  216. }
  217. void b3GpuGridBroadphase::calculateOverlappingPairsHost(int maxPairs)
  218. {
  219. m_hostPairs.resize(0);
  220. m_allAabbsGPU1.copyToHost(m_allAabbsCPU1);
  221. for (int i = 0; i < m_allAabbsCPU1.size(); i++)
  222. {
  223. for (int j = i + 1; j < m_allAabbsCPU1.size(); j++)
  224. {
  225. if (b3TestAabbAgainstAabb2(m_allAabbsCPU1[i].m_minVec, m_allAabbsCPU1[i].m_maxVec,
  226. m_allAabbsCPU1[j].m_minVec, m_allAabbsCPU1[j].m_maxVec))
  227. {
  228. b3Int4 pair;
  229. int a = m_allAabbsCPU1[j].m_minIndices[3];
  230. int b = m_allAabbsCPU1[i].m_minIndices[3];
  231. if (a <= b)
  232. {
  233. pair.x = a;
  234. pair.y = b; //store the original index in the unsorted aabb array
  235. }
  236. else
  237. {
  238. pair.x = b;
  239. pair.y = a; //store the original index in the unsorted aabb array
  240. }
  241. if (m_hostPairs.size() < maxPairs)
  242. {
  243. m_hostPairs.push_back(pair);
  244. }
  245. }
  246. }
  247. }
  248. m_gpuPairs.copyFromHost(m_hostPairs);
  249. }
  250. //call writeAabbsToGpu after done making all changes (createProxy etc)
  251. void b3GpuGridBroadphase::writeAabbsToGpu()
  252. {
  253. m_allAabbsGPU1.copyFromHost(m_allAabbsCPU1);
  254. m_smallAabbsMappingGPU.copyFromHost(m_smallAabbsMappingCPU);
  255. m_largeAabbsMappingGPU.copyFromHost(m_largeAabbsMappingCPU);
  256. }
  257. cl_mem b3GpuGridBroadphase::getAabbBufferWS()
  258. {
  259. return this->m_allAabbsGPU1.getBufferCL();
  260. }
  261. int b3GpuGridBroadphase::getNumOverlap()
  262. {
  263. return m_gpuPairs.size();
  264. }
  265. cl_mem b3GpuGridBroadphase::getOverlappingPairBuffer()
  266. {
  267. return m_gpuPairs.getBufferCL();
  268. }
  269. b3OpenCLArray<b3SapAabb>& b3GpuGridBroadphase::getAllAabbsGPU()
  270. {
  271. return m_allAabbsGPU1;
  272. }
  273. b3AlignedObjectArray<b3SapAabb>& b3GpuGridBroadphase::getAllAabbsCPU()
  274. {
  275. return m_allAabbsCPU1;
  276. }
  277. b3OpenCLArray<b3Int4>& b3GpuGridBroadphase::getOverlappingPairsGPU()
  278. {
  279. return m_gpuPairs;
  280. }
  281. b3OpenCLArray<int>& b3GpuGridBroadphase::getSmallAabbIndicesGPU()
  282. {
  283. return m_smallAabbsMappingGPU;
  284. }
  285. b3OpenCLArray<int>& b3GpuGridBroadphase::getLargeAabbIndicesGPU()
  286. {
  287. return m_largeAabbsMappingGPU;
  288. }