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 | ///btDbvt implementation by Nathanael Presson |
---|
16 | |
---|
17 | #include "btDbvt.h" |
---|
18 | |
---|
19 | // |
---|
20 | typedef btAlignedObjectArray<btDbvtNode*> tNodeArray; |
---|
21 | typedef btAlignedObjectArray<const btDbvtNode*> tConstNodeArray; |
---|
22 | |
---|
23 | // |
---|
24 | struct btDbvtNodeEnumerator : btDbvt::ICollide |
---|
25 | { |
---|
26 | tConstNodeArray nodes; |
---|
27 | void Process(const btDbvtNode* n) { nodes.push_back(n); } |
---|
28 | }; |
---|
29 | |
---|
30 | // |
---|
31 | static DBVT_INLINE int indexof(const btDbvtNode* node) |
---|
32 | { |
---|
33 | return(node->parent->childs[1]==node); |
---|
34 | } |
---|
35 | |
---|
36 | // |
---|
37 | static DBVT_INLINE btDbvtVolume merge( const btDbvtVolume& a, |
---|
38 | const btDbvtVolume& b) |
---|
39 | { |
---|
40 | #if DBVT_MERGE_IMPL==DBVT_IMPL_SSE |
---|
41 | DBVT_ALIGN char locals[sizeof(btDbvtAabbMm)]; |
---|
42 | btDbvtVolume& res=*(btDbvtVolume*)locals; |
---|
43 | #else |
---|
44 | btDbvtVolume res; |
---|
45 | #endif |
---|
46 | Merge(a,b,res); |
---|
47 | return(res); |
---|
48 | } |
---|
49 | |
---|
50 | // volume+edge lengths |
---|
51 | static DBVT_INLINE btScalar size(const btDbvtVolume& a) |
---|
52 | { |
---|
53 | const btVector3 edges=a.Lengths(); |
---|
54 | return( edges.x()*edges.y()*edges.z()+ |
---|
55 | edges.x()+edges.y()+edges.z()); |
---|
56 | } |
---|
57 | |
---|
58 | // |
---|
59 | static void getmaxdepth(const btDbvtNode* node,int depth,int& maxdepth) |
---|
60 | { |
---|
61 | if(node->isinternal()) |
---|
62 | { |
---|
63 | getmaxdepth(node->childs[0],depth+1,maxdepth); |
---|
64 | getmaxdepth(node->childs[0],depth+1,maxdepth); |
---|
65 | } else maxdepth=btMax(maxdepth,depth); |
---|
66 | } |
---|
67 | |
---|
68 | // |
---|
69 | static DBVT_INLINE void deletenode( btDbvt* pdbvt, |
---|
70 | btDbvtNode* node) |
---|
71 | { |
---|
72 | btAlignedFree(pdbvt->m_free); |
---|
73 | pdbvt->m_free=node; |
---|
74 | } |
---|
75 | |
---|
76 | // |
---|
77 | static void recursedeletenode( btDbvt* pdbvt, |
---|
78 | btDbvtNode* node) |
---|
79 | { |
---|
80 | if(!node->isleaf()) |
---|
81 | { |
---|
82 | recursedeletenode(pdbvt,node->childs[0]); |
---|
83 | recursedeletenode(pdbvt,node->childs[1]); |
---|
84 | } |
---|
85 | if(node==pdbvt->m_root) pdbvt->m_root=0; |
---|
86 | deletenode(pdbvt,node); |
---|
87 | } |
---|
88 | |
---|
89 | // |
---|
90 | static DBVT_INLINE btDbvtNode* createnode( btDbvt* pdbvt, |
---|
91 | btDbvtNode* parent, |
---|
92 | void* data) |
---|
93 | { |
---|
94 | btDbvtNode* node; |
---|
95 | if(pdbvt->m_free) |
---|
96 | { node=pdbvt->m_free;pdbvt->m_free=0; } |
---|
97 | else |
---|
98 | { node=new(btAlignedAlloc(sizeof(btDbvtNode),16)) btDbvtNode(); } |
---|
99 | node->parent = parent; |
---|
100 | node->data = data; |
---|
101 | node->childs[1] = 0; |
---|
102 | return(node); |
---|
103 | } |
---|
104 | |
---|
105 | // |
---|
106 | static DBVT_INLINE btDbvtNode* createnode( btDbvt* pdbvt, |
---|
107 | btDbvtNode* parent, |
---|
108 | const btDbvtVolume& volume, |
---|
109 | void* data) |
---|
110 | { |
---|
111 | btDbvtNode* node=createnode(pdbvt,parent,data); |
---|
112 | node->volume=volume; |
---|
113 | return(node); |
---|
114 | } |
---|
115 | |
---|
116 | // |
---|
117 | static DBVT_INLINE btDbvtNode* createnode( btDbvt* pdbvt, |
---|
118 | btDbvtNode* parent, |
---|
119 | const btDbvtVolume& volume0, |
---|
120 | const btDbvtVolume& volume1, |
---|
121 | void* data) |
---|
122 | { |
---|
123 | btDbvtNode* node=createnode(pdbvt,parent,data); |
---|
124 | Merge(volume0,volume1,node->volume); |
---|
125 | return(node); |
---|
126 | } |
---|
127 | |
---|
128 | // |
---|
129 | static void insertleaf( btDbvt* pdbvt, |
---|
130 | btDbvtNode* root, |
---|
131 | btDbvtNode* leaf) |
---|
132 | { |
---|
133 | if(!pdbvt->m_root) |
---|
134 | { |
---|
135 | pdbvt->m_root = leaf; |
---|
136 | leaf->parent = 0; |
---|
137 | } |
---|
138 | else |
---|
139 | { |
---|
140 | if(!root->isleaf()) |
---|
141 | { |
---|
142 | do { |
---|
143 | root=root->childs[Select( leaf->volume, |
---|
144 | root->childs[0]->volume, |
---|
145 | root->childs[1]->volume)]; |
---|
146 | } while(!root->isleaf()); |
---|
147 | } |
---|
148 | btDbvtNode* prev=root->parent; |
---|
149 | btDbvtNode* node=createnode(pdbvt,prev,leaf->volume,root->volume,0); |
---|
150 | if(prev) |
---|
151 | { |
---|
152 | prev->childs[indexof(root)] = node; |
---|
153 | node->childs[0] = root;root->parent=node; |
---|
154 | node->childs[1] = leaf;leaf->parent=node; |
---|
155 | do { |
---|
156 | if(!prev->volume.Contain(node->volume)) |
---|
157 | Merge(prev->childs[0]->volume,prev->childs[1]->volume,prev->volume); |
---|
158 | else |
---|
159 | break; |
---|
160 | node=prev; |
---|
161 | } while(0!=(prev=node->parent)); |
---|
162 | } |
---|
163 | else |
---|
164 | { |
---|
165 | node->childs[0] = root;root->parent=node; |
---|
166 | node->childs[1] = leaf;leaf->parent=node; |
---|
167 | pdbvt->m_root = node; |
---|
168 | } |
---|
169 | } |
---|
170 | } |
---|
171 | |
---|
172 | // |
---|
173 | static btDbvtNode* removeleaf( btDbvt* pdbvt, |
---|
174 | btDbvtNode* leaf) |
---|
175 | { |
---|
176 | if(leaf==pdbvt->m_root) |
---|
177 | { |
---|
178 | pdbvt->m_root=0; |
---|
179 | return(0); |
---|
180 | } |
---|
181 | else |
---|
182 | { |
---|
183 | btDbvtNode* parent=leaf->parent; |
---|
184 | btDbvtNode* prev=parent->parent; |
---|
185 | btDbvtNode* sibling=parent->childs[1-indexof(leaf)]; |
---|
186 | if(prev) |
---|
187 | { |
---|
188 | prev->childs[indexof(parent)]=sibling; |
---|
189 | sibling->parent=prev; |
---|
190 | deletenode(pdbvt,parent); |
---|
191 | while(prev) |
---|
192 | { |
---|
193 | const btDbvtVolume pb=prev->volume; |
---|
194 | Merge(prev->childs[0]->volume,prev->childs[1]->volume,prev->volume); |
---|
195 | if(NotEqual(pb,prev->volume)) |
---|
196 | { |
---|
197 | prev=prev->parent; |
---|
198 | } else break; |
---|
199 | } |
---|
200 | return(prev?prev:pdbvt->m_root); |
---|
201 | } |
---|
202 | else |
---|
203 | { |
---|
204 | pdbvt->m_root=sibling; |
---|
205 | sibling->parent=0; |
---|
206 | deletenode(pdbvt,parent); |
---|
207 | return(pdbvt->m_root); |
---|
208 | } |
---|
209 | } |
---|
210 | } |
---|
211 | |
---|
212 | // |
---|
213 | static void fetchleaves(btDbvt* pdbvt, |
---|
214 | btDbvtNode* root, |
---|
215 | tNodeArray& leaves, |
---|
216 | int depth=-1) |
---|
217 | { |
---|
218 | if(root->isinternal()&&depth) |
---|
219 | { |
---|
220 | fetchleaves(pdbvt,root->childs[0],leaves,depth-1); |
---|
221 | fetchleaves(pdbvt,root->childs[1],leaves,depth-1); |
---|
222 | deletenode(pdbvt,root); |
---|
223 | } |
---|
224 | else |
---|
225 | { |
---|
226 | leaves.push_back(root); |
---|
227 | } |
---|
228 | } |
---|
229 | |
---|
230 | // |
---|
231 | static void split( const tNodeArray& leaves, |
---|
232 | tNodeArray& left, |
---|
233 | tNodeArray& right, |
---|
234 | const btVector3& org, |
---|
235 | const btVector3& axis) |
---|
236 | { |
---|
237 | left.resize(0); |
---|
238 | right.resize(0); |
---|
239 | for(int i=0,ni=leaves.size();i<ni;++i) |
---|
240 | { |
---|
241 | if(dot(axis,leaves[i]->volume.Center()-org)<0) |
---|
242 | left.push_back(leaves[i]); |
---|
243 | else |
---|
244 | right.push_back(leaves[i]); |
---|
245 | } |
---|
246 | } |
---|
247 | |
---|
248 | // |
---|
249 | static btDbvtVolume bounds( const tNodeArray& leaves) |
---|
250 | { |
---|
251 | #if DBVT_MERGE_IMPL==DBVT_IMPL_SSE |
---|
252 | DBVT_ALIGN char locals[sizeof(btDbvtVolume)]; |
---|
253 | btDbvtVolume& volume=*(btDbvtVolume*)locals; |
---|
254 | volume=leaves[0]->volume; |
---|
255 | #else |
---|
256 | btDbvtVolume volume=leaves[0]->volume; |
---|
257 | #endif |
---|
258 | for(int i=1,ni=leaves.size();i<ni;++i) |
---|
259 | { |
---|
260 | Merge(volume,leaves[i]->volume,volume); |
---|
261 | } |
---|
262 | return(volume); |
---|
263 | } |
---|
264 | |
---|
265 | // |
---|
266 | static void bottomup( btDbvt* pdbvt, |
---|
267 | tNodeArray& leaves) |
---|
268 | { |
---|
269 | while(leaves.size()>1) |
---|
270 | { |
---|
271 | btScalar minsize=SIMD_INFINITY; |
---|
272 | int minidx[2]={-1,-1}; |
---|
273 | for(int i=0;i<leaves.size();++i) |
---|
274 | { |
---|
275 | for(int j=i+1;j<leaves.size();++j) |
---|
276 | { |
---|
277 | const btScalar sz=size(merge(leaves[i]->volume,leaves[j]->volume)); |
---|
278 | if(sz<minsize) |
---|
279 | { |
---|
280 | minsize = sz; |
---|
281 | minidx[0] = i; |
---|
282 | minidx[1] = j; |
---|
283 | } |
---|
284 | } |
---|
285 | } |
---|
286 | btDbvtNode* n[] = {leaves[minidx[0]],leaves[minidx[1]]}; |
---|
287 | btDbvtNode* p = createnode(pdbvt,0,n[0]->volume,n[1]->volume,0); |
---|
288 | p->childs[0] = n[0]; |
---|
289 | p->childs[1] = n[1]; |
---|
290 | n[0]->parent = p; |
---|
291 | n[1]->parent = p; |
---|
292 | leaves[minidx[0]] = p; |
---|
293 | leaves.swap(minidx[1],leaves.size()-1); |
---|
294 | leaves.pop_back(); |
---|
295 | } |
---|
296 | } |
---|
297 | |
---|
298 | // |
---|
299 | static btDbvtNode* topdown(btDbvt* pdbvt, |
---|
300 | tNodeArray& leaves, |
---|
301 | int bu_treshold) |
---|
302 | { |
---|
303 | static const btVector3 axis[]={btVector3(1,0,0), |
---|
304 | btVector3(0,1,0), |
---|
305 | btVector3(0,0,1)}; |
---|
306 | if(leaves.size()>1) |
---|
307 | { |
---|
308 | if(leaves.size()>bu_treshold) |
---|
309 | { |
---|
310 | const btDbvtVolume vol=bounds(leaves); |
---|
311 | const btVector3 org=vol.Center(); |
---|
312 | tNodeArray sets[2]; |
---|
313 | int bestaxis=-1; |
---|
314 | int bestmidp=leaves.size(); |
---|
315 | int splitcount[3][2]={{0,0},{0,0},{0,0}}; |
---|
316 | int i; |
---|
317 | for( i=0;i<leaves.size();++i) |
---|
318 | { |
---|
319 | const btVector3 x=leaves[i]->volume.Center()-org; |
---|
320 | for(int j=0;j<3;++j) |
---|
321 | { |
---|
322 | ++splitcount[j][dot(x,axis[j])>0?1:0]; |
---|
323 | } |
---|
324 | } |
---|
325 | for( i=0;i<3;++i) |
---|
326 | { |
---|
327 | if((splitcount[i][0]>0)&&(splitcount[i][1]>0)) |
---|
328 | { |
---|
329 | const int midp=(int)btFabs(btScalar(splitcount[i][0]-splitcount[i][1])); |
---|
330 | if(midp<bestmidp) |
---|
331 | { |
---|
332 | bestaxis=i; |
---|
333 | bestmidp=midp; |
---|
334 | } |
---|
335 | } |
---|
336 | } |
---|
337 | if(bestaxis>=0) |
---|
338 | { |
---|
339 | sets[0].reserve(splitcount[bestaxis][0]); |
---|
340 | sets[1].reserve(splitcount[bestaxis][1]); |
---|
341 | split(leaves,sets[0],sets[1],org,axis[bestaxis]); |
---|
342 | } |
---|
343 | else |
---|
344 | { |
---|
345 | sets[0].reserve(leaves.size()/2+1); |
---|
346 | sets[1].reserve(leaves.size()/2); |
---|
347 | for(int i=0,ni=leaves.size();i<ni;++i) |
---|
348 | { |
---|
349 | sets[i&1].push_back(leaves[i]); |
---|
350 | } |
---|
351 | } |
---|
352 | btDbvtNode* node=createnode(pdbvt,0,vol,0); |
---|
353 | node->childs[0]=topdown(pdbvt,sets[0],bu_treshold); |
---|
354 | node->childs[1]=topdown(pdbvt,sets[1],bu_treshold); |
---|
355 | node->childs[0]->parent=node; |
---|
356 | node->childs[1]->parent=node; |
---|
357 | return(node); |
---|
358 | } |
---|
359 | else |
---|
360 | { |
---|
361 | bottomup(pdbvt,leaves); |
---|
362 | return(leaves[0]); |
---|
363 | } |
---|
364 | } |
---|
365 | return(leaves[0]); |
---|
366 | } |
---|
367 | |
---|
368 | // |
---|
369 | static DBVT_INLINE btDbvtNode* sort(btDbvtNode* n,btDbvtNode*& r) |
---|
370 | { |
---|
371 | btDbvtNode* p=n->parent; |
---|
372 | btAssert(n->isinternal()); |
---|
373 | if(p>n) |
---|
374 | { |
---|
375 | const int i=indexof(n); |
---|
376 | const int j=1-i; |
---|
377 | btDbvtNode* s=p->childs[j]; |
---|
378 | btDbvtNode* q=p->parent; |
---|
379 | btAssert(n==p->childs[i]); |
---|
380 | if(q) q->childs[indexof(p)]=n; else r=n; |
---|
381 | s->parent=n; |
---|
382 | p->parent=n; |
---|
383 | n->parent=q; |
---|
384 | p->childs[0]=n->childs[0]; |
---|
385 | p->childs[1]=n->childs[1]; |
---|
386 | n->childs[0]->parent=p; |
---|
387 | n->childs[1]->parent=p; |
---|
388 | n->childs[i]=p; |
---|
389 | n->childs[j]=s; |
---|
390 | btSwap(p->volume,n->volume); |
---|
391 | return(p); |
---|
392 | } |
---|
393 | return(n); |
---|
394 | } |
---|
395 | |
---|
396 | // |
---|
397 | static DBVT_INLINE btDbvtNode* walkup(btDbvtNode* n,int count) |
---|
398 | { |
---|
399 | while(n&&(count--)) n=n->parent; |
---|
400 | return(n); |
---|
401 | } |
---|
402 | |
---|
403 | // |
---|
404 | // Api |
---|
405 | // |
---|
406 | |
---|
407 | // |
---|
408 | btDbvt::btDbvt() |
---|
409 | { |
---|
410 | m_root = 0; |
---|
411 | m_free = 0; |
---|
412 | m_lkhd = -1; |
---|
413 | m_leaves = 0; |
---|
414 | m_opath = 0; |
---|
415 | } |
---|
416 | |
---|
417 | // |
---|
418 | btDbvt::~btDbvt() |
---|
419 | { |
---|
420 | clear(); |
---|
421 | } |
---|
422 | |
---|
423 | // |
---|
424 | void btDbvt::clear() |
---|
425 | { |
---|
426 | if(m_root) recursedeletenode(this,m_root); |
---|
427 | btAlignedFree(m_free); |
---|
428 | m_free=0; |
---|
429 | } |
---|
430 | |
---|
431 | // |
---|
432 | void btDbvt::optimizeBottomUp() |
---|
433 | { |
---|
434 | if(m_root) |
---|
435 | { |
---|
436 | tNodeArray leaves; |
---|
437 | leaves.reserve(m_leaves); |
---|
438 | fetchleaves(this,m_root,leaves); |
---|
439 | bottomup(this,leaves); |
---|
440 | m_root=leaves[0]; |
---|
441 | } |
---|
442 | } |
---|
443 | |
---|
444 | // |
---|
445 | void btDbvt::optimizeTopDown(int bu_treshold) |
---|
446 | { |
---|
447 | if(m_root) |
---|
448 | { |
---|
449 | tNodeArray leaves; |
---|
450 | leaves.reserve(m_leaves); |
---|
451 | fetchleaves(this,m_root,leaves); |
---|
452 | m_root=topdown(this,leaves,bu_treshold); |
---|
453 | } |
---|
454 | } |
---|
455 | |
---|
456 | // |
---|
457 | void btDbvt::optimizeIncremental(int passes) |
---|
458 | { |
---|
459 | if(passes<0) passes=m_leaves; |
---|
460 | if(m_root&&(passes>0)) |
---|
461 | { |
---|
462 | do { |
---|
463 | btDbvtNode* node=m_root; |
---|
464 | unsigned bit=0; |
---|
465 | while(node->isinternal()) |
---|
466 | { |
---|
467 | node=sort(node,m_root)->childs[(m_opath>>bit)&1]; |
---|
468 | bit=(bit+1)&(sizeof(unsigned)*8-1); |
---|
469 | } |
---|
470 | update(node); |
---|
471 | ++m_opath; |
---|
472 | } while(--passes); |
---|
473 | } |
---|
474 | } |
---|
475 | |
---|
476 | // |
---|
477 | btDbvtNode* btDbvt::insert(const btDbvtVolume& volume,void* data) |
---|
478 | { |
---|
479 | btDbvtNode* leaf=createnode(this,0,volume,data); |
---|
480 | insertleaf(this,m_root,leaf); |
---|
481 | ++m_leaves; |
---|
482 | return(leaf); |
---|
483 | } |
---|
484 | |
---|
485 | // |
---|
486 | void btDbvt::update(btDbvtNode* leaf,int lookahead) |
---|
487 | { |
---|
488 | btDbvtNode* root=removeleaf(this,leaf); |
---|
489 | if(root) |
---|
490 | { |
---|
491 | if(lookahead>=0) |
---|
492 | { |
---|
493 | for(int i=0;(i<lookahead)&&root->parent;++i) |
---|
494 | { |
---|
495 | root=root->parent; |
---|
496 | } |
---|
497 | } else root=m_root; |
---|
498 | } |
---|
499 | insertleaf(this,root,leaf); |
---|
500 | } |
---|
501 | |
---|
502 | // |
---|
503 | void btDbvt::update(btDbvtNode* leaf,const btDbvtVolume& volume) |
---|
504 | { |
---|
505 | btDbvtNode* root=removeleaf(this,leaf); |
---|
506 | if(root) |
---|
507 | { |
---|
508 | if(m_lkhd>=0) |
---|
509 | { |
---|
510 | for(int i=0;(i<m_lkhd)&&root->parent;++i) |
---|
511 | { |
---|
512 | root=root->parent; |
---|
513 | } |
---|
514 | } else root=m_root; |
---|
515 | } |
---|
516 | leaf->volume=volume; |
---|
517 | insertleaf(this,root,leaf); |
---|
518 | } |
---|
519 | |
---|
520 | // |
---|
521 | bool btDbvt::update(btDbvtNode* leaf,btDbvtVolume volume,const btVector3& velocity,btScalar margin) |
---|
522 | { |
---|
523 | if(leaf->volume.Contain(volume)) return(false); |
---|
524 | volume.Expand(btVector3(margin,margin,margin)); |
---|
525 | volume.SignedExpand(velocity); |
---|
526 | update(leaf,volume); |
---|
527 | return(true); |
---|
528 | } |
---|
529 | |
---|
530 | // |
---|
531 | bool btDbvt::update(btDbvtNode* leaf,btDbvtVolume volume,const btVector3& velocity) |
---|
532 | { |
---|
533 | if(leaf->volume.Contain(volume)) return(false); |
---|
534 | volume.SignedExpand(velocity); |
---|
535 | update(leaf,volume); |
---|
536 | return(true); |
---|
537 | } |
---|
538 | |
---|
539 | // |
---|
540 | bool btDbvt::update(btDbvtNode* leaf,btDbvtVolume volume,btScalar margin) |
---|
541 | { |
---|
542 | if(leaf->volume.Contain(volume)) return(false); |
---|
543 | volume.Expand(btVector3(margin,margin,margin)); |
---|
544 | update(leaf,volume); |
---|
545 | return(true); |
---|
546 | } |
---|
547 | |
---|
548 | // |
---|
549 | void btDbvt::remove(btDbvtNode* leaf) |
---|
550 | { |
---|
551 | removeleaf(this,leaf); |
---|
552 | deletenode(this,leaf); |
---|
553 | --m_leaves; |
---|
554 | } |
---|
555 | |
---|
556 | // |
---|
557 | void btDbvt::write(IWriter* iwriter) const |
---|
558 | { |
---|
559 | btDbvtNodeEnumerator nodes; |
---|
560 | nodes.nodes.reserve(m_leaves*2); |
---|
561 | enumNodes(m_root,nodes); |
---|
562 | iwriter->Prepare(m_root,nodes.nodes.size()); |
---|
563 | for(int i=0;i<nodes.nodes.size();++i) |
---|
564 | { |
---|
565 | const btDbvtNode* n=nodes.nodes[i]; |
---|
566 | int p=-1; |
---|
567 | if(n->parent) p=nodes.nodes.findLinearSearch(n->parent); |
---|
568 | if(n->isinternal()) |
---|
569 | { |
---|
570 | const int c0=nodes.nodes.findLinearSearch(n->childs[0]); |
---|
571 | const int c1=nodes.nodes.findLinearSearch(n->childs[1]); |
---|
572 | iwriter->WriteNode(n,i,p,c0,c1); |
---|
573 | } |
---|
574 | else |
---|
575 | { |
---|
576 | iwriter->WriteLeaf(n,i,p); |
---|
577 | } |
---|
578 | } |
---|
579 | } |
---|
580 | |
---|
581 | // |
---|
582 | void btDbvt::clone(btDbvt& dest,IClone* iclone) const |
---|
583 | { |
---|
584 | dest.clear(); |
---|
585 | if(m_root!=0) |
---|
586 | { |
---|
587 | btAlignedObjectArray<sStkCLN> stack; |
---|
588 | stack.reserve(m_leaves); |
---|
589 | stack.push_back(sStkCLN(m_root,0)); |
---|
590 | do { |
---|
591 | const int i=stack.size()-1; |
---|
592 | const sStkCLN e=stack[i]; |
---|
593 | btDbvtNode* n=createnode(&dest,e.parent,e.node->volume,e.node->data); |
---|
594 | stack.pop_back(); |
---|
595 | if(e.parent!=0) |
---|
596 | e.parent->childs[i&1]=n; |
---|
597 | else |
---|
598 | dest.m_root=n; |
---|
599 | if(e.node->isinternal()) |
---|
600 | { |
---|
601 | stack.push_back(sStkCLN(e.node->childs[0],n)); |
---|
602 | stack.push_back(sStkCLN(e.node->childs[1],n)); |
---|
603 | } |
---|
604 | else |
---|
605 | { |
---|
606 | iclone->CloneLeaf(n); |
---|
607 | } |
---|
608 | } while(stack.size()>0); |
---|
609 | } |
---|
610 | } |
---|
611 | |
---|
612 | // |
---|
613 | int btDbvt::maxdepth(const btDbvtNode* node) |
---|
614 | { |
---|
615 | int depth=0; |
---|
616 | if(node) getmaxdepth(node,1,depth); |
---|
617 | return(depth); |
---|
618 | } |
---|
619 | |
---|
620 | // |
---|
621 | int btDbvt::countLeaves(const btDbvtNode* node) |
---|
622 | { |
---|
623 | if(node->isinternal()) |
---|
624 | return(countLeaves(node->childs[0])+countLeaves(node->childs[1])); |
---|
625 | else |
---|
626 | return(1); |
---|
627 | } |
---|
628 | |
---|
629 | // |
---|
630 | void btDbvt::extractLeaves(const btDbvtNode* node,btAlignedObjectArray<const btDbvtNode*>& leaves) |
---|
631 | { |
---|
632 | if(node->isinternal()) |
---|
633 | { |
---|
634 | extractLeaves(node->childs[0],leaves); |
---|
635 | extractLeaves(node->childs[1],leaves); |
---|
636 | } |
---|
637 | else |
---|
638 | { |
---|
639 | leaves.push_back(node); |
---|
640 | } |
---|
641 | } |
---|
642 | |
---|
643 | // |
---|
644 | #if DBVT_ENABLE_BENCHMARK |
---|
645 | |
---|
646 | #include <stdio.h> |
---|
647 | #include <stdlib.h> |
---|
648 | #include "LinearMath/btQuickProf.h" |
---|
649 | |
---|
650 | /* |
---|
651 | q6600,2.4ghz |
---|
652 | |
---|
653 | /Ox /Ob2 /Oi /Ot /I "." /I "..\.." /I "..\..\src" /D "NDEBUG" /D "_LIB" /D "_WINDOWS" /D "_CRT_SECURE_NO_DEPRECATE" /D "_CRT_NONSTDC_NO_DEPRECATE" /D "WIN32" |
---|
654 | /GF /FD /MT /GS- /Gy /arch:SSE2 /Zc:wchar_t- /Fp"..\..\out\release8\build\libbulletcollision\libbulletcollision.pch" |
---|
655 | /Fo"..\..\out\release8\build\libbulletcollision\\" |
---|
656 | /Fd"..\..\out\release8\build\libbulletcollision\bulletcollision.pdb" |
---|
657 | /W3 /nologo /c /Wp64 /Zi /errorReport:prompt |
---|
658 | |
---|
659 | Benchmarking dbvt... |
---|
660 | World scale: 100.000000 |
---|
661 | Extents base: 1.000000 |
---|
662 | Extents range: 4.000000 |
---|
663 | Leaves: 8192 |
---|
664 | sizeof(btDbvtVolume): 32 bytes |
---|
665 | sizeof(btDbvtNode): 44 bytes |
---|
666 | [1] btDbvtVolume intersections: 3499 ms (-1%) |
---|
667 | [2] btDbvtVolume merges: 1934 ms (0%) |
---|
668 | [3] btDbvt::collideTT: 5485 ms (-21%) |
---|
669 | [4] btDbvt::collideTT self: 2814 ms (-20%) |
---|
670 | [5] btDbvt::collideTT xform: 7379 ms (-1%) |
---|
671 | [6] btDbvt::collideTT xform,self: 7270 ms (-2%) |
---|
672 | [7] btDbvt::collideRAY: 6314 ms (0%),(332143 r/s) |
---|
673 | [8] insert/remove: 2093 ms (0%),(1001983 ir/s) |
---|
674 | [9] updates (teleport): 1879 ms (-3%),(1116100 u/s) |
---|
675 | [10] updates (jitter): 1244 ms (-4%),(1685813 u/s) |
---|
676 | [11] optimize (incremental): 2514 ms (0%),(1668000 o/s) |
---|
677 | [12] btDbvtVolume notequal: 3659 ms (0%) |
---|
678 | [13] culling(OCL+fullsort): 2218 ms (0%),(461 t/s) |
---|
679 | [14] culling(OCL+qsort): 3688 ms (5%),(2221 t/s) |
---|
680 | [15] culling(KDOP+qsort): 1139 ms (-1%),(7192 t/s) |
---|
681 | [16] insert/remove batch(256): 5092 ms (0%),(823704 bir/s) |
---|
682 | [17] btDbvtVolume select: 3419 ms (0%) |
---|
683 | */ |
---|
684 | |
---|
685 | struct btDbvtBenchmark |
---|
686 | { |
---|
687 | struct NilPolicy : btDbvt::ICollide |
---|
688 | { |
---|
689 | NilPolicy() : m_pcount(0),m_depth(-SIMD_INFINITY),m_checksort(true) {} |
---|
690 | void Process(const btDbvtNode*,const btDbvtNode*) { ++m_pcount; } |
---|
691 | void Process(const btDbvtNode*) { ++m_pcount; } |
---|
692 | void Process(const btDbvtNode*,btScalar depth) |
---|
693 | { |
---|
694 | ++m_pcount; |
---|
695 | if(m_checksort) |
---|
696 | { if(depth>=m_depth) m_depth=depth; else printf("wrong depth: %f (should be >= %f)\r\n",depth,m_depth); } |
---|
697 | } |
---|
698 | int m_pcount; |
---|
699 | btScalar m_depth; |
---|
700 | bool m_checksort; |
---|
701 | }; |
---|
702 | struct P14 : btDbvt::ICollide |
---|
703 | { |
---|
704 | struct Node |
---|
705 | { |
---|
706 | const btDbvtNode* leaf; |
---|
707 | btScalar depth; |
---|
708 | }; |
---|
709 | void Process(const btDbvtNode* leaf,btScalar depth) |
---|
710 | { |
---|
711 | Node n; |
---|
712 | n.leaf = leaf; |
---|
713 | n.depth = depth; |
---|
714 | } |
---|
715 | static int sortfnc(const Node& a,const Node& b) |
---|
716 | { |
---|
717 | if(a.depth<b.depth) return(+1); |
---|
718 | if(a.depth>b.depth) return(-1); |
---|
719 | return(0); |
---|
720 | } |
---|
721 | btAlignedObjectArray<Node> m_nodes; |
---|
722 | }; |
---|
723 | struct P15 : btDbvt::ICollide |
---|
724 | { |
---|
725 | struct Node |
---|
726 | { |
---|
727 | const btDbvtNode* leaf; |
---|
728 | btScalar depth; |
---|
729 | }; |
---|
730 | void Process(const btDbvtNode* leaf) |
---|
731 | { |
---|
732 | Node n; |
---|
733 | n.leaf = leaf; |
---|
734 | n.depth = dot(leaf->volume.Center(),m_axis); |
---|
735 | } |
---|
736 | static int sortfnc(const Node& a,const Node& b) |
---|
737 | { |
---|
738 | if(a.depth<b.depth) return(+1); |
---|
739 | if(a.depth>b.depth) return(-1); |
---|
740 | return(0); |
---|
741 | } |
---|
742 | btAlignedObjectArray<Node> m_nodes; |
---|
743 | btVector3 m_axis; |
---|
744 | }; |
---|
745 | static btScalar RandUnit() |
---|
746 | { |
---|
747 | return(rand()/(btScalar)RAND_MAX); |
---|
748 | } |
---|
749 | static btVector3 RandVector3() |
---|
750 | { |
---|
751 | return(btVector3(RandUnit(),RandUnit(),RandUnit())); |
---|
752 | } |
---|
753 | static btVector3 RandVector3(btScalar cs) |
---|
754 | { |
---|
755 | return(RandVector3()*cs-btVector3(cs,cs,cs)/2); |
---|
756 | } |
---|
757 | static btDbvtVolume RandVolume(btScalar cs,btScalar eb,btScalar es) |
---|
758 | { |
---|
759 | return(btDbvtVolume::FromCE(RandVector3(cs),btVector3(eb,eb,eb)+RandVector3()*es)); |
---|
760 | } |
---|
761 | static btTransform RandTransform(btScalar cs) |
---|
762 | { |
---|
763 | btTransform t; |
---|
764 | t.setOrigin(RandVector3(cs)); |
---|
765 | t.setRotation(btQuaternion(RandUnit()*SIMD_PI*2,RandUnit()*SIMD_PI*2,RandUnit()*SIMD_PI*2).normalized()); |
---|
766 | return(t); |
---|
767 | } |
---|
768 | static void RandTree(btScalar cs,btScalar eb,btScalar es,int leaves,btDbvt& dbvt) |
---|
769 | { |
---|
770 | dbvt.clear(); |
---|
771 | for(int i=0;i<leaves;++i) |
---|
772 | { |
---|
773 | dbvt.insert(RandVolume(cs,eb,es),0); |
---|
774 | } |
---|
775 | } |
---|
776 | }; |
---|
777 | |
---|
778 | void btDbvt::benchmark() |
---|
779 | { |
---|
780 | static const btScalar cfgVolumeCenterScale = 100; |
---|
781 | static const btScalar cfgVolumeExentsBase = 1; |
---|
782 | static const btScalar cfgVolumeExentsScale = 4; |
---|
783 | static const int cfgLeaves = 8192; |
---|
784 | static const bool cfgEnable = true; |
---|
785 | |
---|
786 | //[1] btDbvtVolume intersections |
---|
787 | bool cfgBenchmark1_Enable = cfgEnable; |
---|
788 | static const int cfgBenchmark1_Iterations = 8; |
---|
789 | static const int cfgBenchmark1_Reference = 3499; |
---|
790 | //[2] btDbvtVolume merges |
---|
791 | bool cfgBenchmark2_Enable = cfgEnable; |
---|
792 | static const int cfgBenchmark2_Iterations = 4; |
---|
793 | static const int cfgBenchmark2_Reference = 1945; |
---|
794 | //[3] btDbvt::collideTT |
---|
795 | bool cfgBenchmark3_Enable = cfgEnable; |
---|
796 | static const int cfgBenchmark3_Iterations = 512; |
---|
797 | static const int cfgBenchmark3_Reference = 5485; |
---|
798 | //[4] btDbvt::collideTT self |
---|
799 | bool cfgBenchmark4_Enable = cfgEnable; |
---|
800 | static const int cfgBenchmark4_Iterations = 512; |
---|
801 | static const int cfgBenchmark4_Reference = 2814; |
---|
802 | //[5] btDbvt::collideTT xform |
---|
803 | bool cfgBenchmark5_Enable = cfgEnable; |
---|
804 | static const int cfgBenchmark5_Iterations = 512; |
---|
805 | static const btScalar cfgBenchmark5_OffsetScale = 2; |
---|
806 | static const int cfgBenchmark5_Reference = 7379; |
---|
807 | //[6] btDbvt::collideTT xform,self |
---|
808 | bool cfgBenchmark6_Enable = cfgEnable; |
---|
809 | static const int cfgBenchmark6_Iterations = 512; |
---|
810 | static const btScalar cfgBenchmark6_OffsetScale = 2; |
---|
811 | static const int cfgBenchmark6_Reference = 7270; |
---|
812 | //[7] btDbvt::collideRAY |
---|
813 | bool cfgBenchmark7_Enable = cfgEnable; |
---|
814 | static const int cfgBenchmark7_Passes = 32; |
---|
815 | static const int cfgBenchmark7_Iterations = 65536; |
---|
816 | static const int cfgBenchmark7_Reference = 6307; |
---|
817 | //[8] insert/remove |
---|
818 | bool cfgBenchmark8_Enable = cfgEnable; |
---|
819 | static const int cfgBenchmark8_Passes = 32; |
---|
820 | static const int cfgBenchmark8_Iterations = 65536; |
---|
821 | static const int cfgBenchmark8_Reference = 2105; |
---|
822 | //[9] updates (teleport) |
---|
823 | bool cfgBenchmark9_Enable = cfgEnable; |
---|
824 | static const int cfgBenchmark9_Passes = 32; |
---|
825 | static const int cfgBenchmark9_Iterations = 65536; |
---|
826 | static const int cfgBenchmark9_Reference = 1879; |
---|
827 | //[10] updates (jitter) |
---|
828 | bool cfgBenchmark10_Enable = cfgEnable; |
---|
829 | static const btScalar cfgBenchmark10_Scale = cfgVolumeCenterScale/10000; |
---|
830 | static const int cfgBenchmark10_Passes = 32; |
---|
831 | static const int cfgBenchmark10_Iterations = 65536; |
---|
832 | static const int cfgBenchmark10_Reference = 1244; |
---|
833 | //[11] optimize (incremental) |
---|
834 | bool cfgBenchmark11_Enable = cfgEnable; |
---|
835 | static const int cfgBenchmark11_Passes = 64; |
---|
836 | static const int cfgBenchmark11_Iterations = 65536; |
---|
837 | static const int cfgBenchmark11_Reference = 2510; |
---|
838 | //[12] btDbvtVolume notequal |
---|
839 | bool cfgBenchmark12_Enable = cfgEnable; |
---|
840 | static const int cfgBenchmark12_Iterations = 32; |
---|
841 | static const int cfgBenchmark12_Reference = 3677; |
---|
842 | //[13] culling(OCL+fullsort) |
---|
843 | bool cfgBenchmark13_Enable = cfgEnable; |
---|
844 | static const int cfgBenchmark13_Iterations = 1024; |
---|
845 | static const int cfgBenchmark13_Reference = 2231; |
---|
846 | //[14] culling(OCL+qsort) |
---|
847 | bool cfgBenchmark14_Enable = cfgEnable; |
---|
848 | static const int cfgBenchmark14_Iterations = 8192; |
---|
849 | static const int cfgBenchmark14_Reference = 3500; |
---|
850 | //[15] culling(KDOP+qsort) |
---|
851 | bool cfgBenchmark15_Enable = cfgEnable; |
---|
852 | static const int cfgBenchmark15_Iterations = 8192; |
---|
853 | static const int cfgBenchmark15_Reference = 1151; |
---|
854 | //[16] insert/remove batch |
---|
855 | bool cfgBenchmark16_Enable = cfgEnable; |
---|
856 | static const int cfgBenchmark16_BatchCount = 256; |
---|
857 | static const int cfgBenchmark16_Passes = 16384; |
---|
858 | static const int cfgBenchmark16_Reference = 5138; |
---|
859 | //[17] select |
---|
860 | bool cfgBenchmark17_Enable = cfgEnable; |
---|
861 | static const int cfgBenchmark17_Iterations = 4; |
---|
862 | static const int cfgBenchmark17_Reference = 3390; |
---|
863 | |
---|
864 | btClock wallclock; |
---|
865 | printf("Benchmarking dbvt...\r\n"); |
---|
866 | printf("\tWorld scale: %f\r\n",cfgVolumeCenterScale); |
---|
867 | printf("\tExtents base: %f\r\n",cfgVolumeExentsBase); |
---|
868 | printf("\tExtents range: %f\r\n",cfgVolumeExentsScale); |
---|
869 | printf("\tLeaves: %u\r\n",cfgLeaves); |
---|
870 | printf("\tsizeof(btDbvtVolume): %u bytes\r\n",sizeof(btDbvtVolume)); |
---|
871 | printf("\tsizeof(btDbvtNode): %u bytes\r\n",sizeof(btDbvtNode)); |
---|
872 | if(cfgBenchmark1_Enable) |
---|
873 | {// Benchmark 1 |
---|
874 | srand(380843); |
---|
875 | btAlignedObjectArray<btDbvtVolume> volumes; |
---|
876 | btAlignedObjectArray<bool> results; |
---|
877 | volumes.resize(cfgLeaves); |
---|
878 | results.resize(cfgLeaves); |
---|
879 | for(int i=0;i<cfgLeaves;++i) |
---|
880 | { |
---|
881 | volumes[i]=btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale); |
---|
882 | } |
---|
883 | printf("[1] btDbvtVolume intersections: "); |
---|
884 | wallclock.reset(); |
---|
885 | for(int i=0;i<cfgBenchmark1_Iterations;++i) |
---|
886 | { |
---|
887 | for(int j=0;j<cfgLeaves;++j) |
---|
888 | { |
---|
889 | for(int k=0;k<cfgLeaves;++k) |
---|
890 | { |
---|
891 | results[k]=Intersect(volumes[j],volumes[k]); |
---|
892 | } |
---|
893 | } |
---|
894 | } |
---|
895 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
896 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark1_Reference)*100/time); |
---|
897 | } |
---|
898 | if(cfgBenchmark2_Enable) |
---|
899 | {// Benchmark 2 |
---|
900 | srand(380843); |
---|
901 | btAlignedObjectArray<btDbvtVolume> volumes; |
---|
902 | btAlignedObjectArray<btDbvtVolume> results; |
---|
903 | volumes.resize(cfgLeaves); |
---|
904 | results.resize(cfgLeaves); |
---|
905 | for(int i=0;i<cfgLeaves;++i) |
---|
906 | { |
---|
907 | volumes[i]=btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale); |
---|
908 | } |
---|
909 | printf("[2] btDbvtVolume merges: "); |
---|
910 | wallclock.reset(); |
---|
911 | for(int i=0;i<cfgBenchmark2_Iterations;++i) |
---|
912 | { |
---|
913 | for(int j=0;j<cfgLeaves;++j) |
---|
914 | { |
---|
915 | for(int k=0;k<cfgLeaves;++k) |
---|
916 | { |
---|
917 | Merge(volumes[j],volumes[k],results[k]); |
---|
918 | } |
---|
919 | } |
---|
920 | } |
---|
921 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
922 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark2_Reference)*100/time); |
---|
923 | } |
---|
924 | if(cfgBenchmark3_Enable) |
---|
925 | {// Benchmark 3 |
---|
926 | srand(380843); |
---|
927 | btDbvt dbvt[2]; |
---|
928 | btDbvtBenchmark::NilPolicy policy; |
---|
929 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt[0]); |
---|
930 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt[1]); |
---|
931 | dbvt[0].optimizeTopDown(); |
---|
932 | dbvt[1].optimizeTopDown(); |
---|
933 | printf("[3] btDbvt::collideTT: "); |
---|
934 | wallclock.reset(); |
---|
935 | for(int i=0;i<cfgBenchmark3_Iterations;++i) |
---|
936 | { |
---|
937 | btDbvt::collideTT(dbvt[0].m_root,dbvt[1].m_root,policy); |
---|
938 | } |
---|
939 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
940 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark3_Reference)*100/time); |
---|
941 | } |
---|
942 | if(cfgBenchmark4_Enable) |
---|
943 | {// Benchmark 4 |
---|
944 | srand(380843); |
---|
945 | btDbvt dbvt; |
---|
946 | btDbvtBenchmark::NilPolicy policy; |
---|
947 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
948 | dbvt.optimizeTopDown(); |
---|
949 | printf("[4] btDbvt::collideTT self: "); |
---|
950 | wallclock.reset(); |
---|
951 | for(int i=0;i<cfgBenchmark4_Iterations;++i) |
---|
952 | { |
---|
953 | btDbvt::collideTT(dbvt.m_root,dbvt.m_root,policy); |
---|
954 | } |
---|
955 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
956 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark4_Reference)*100/time); |
---|
957 | } |
---|
958 | if(cfgBenchmark5_Enable) |
---|
959 | {// Benchmark 5 |
---|
960 | srand(380843); |
---|
961 | btDbvt dbvt[2]; |
---|
962 | btAlignedObjectArray<btTransform> transforms; |
---|
963 | btDbvtBenchmark::NilPolicy policy; |
---|
964 | transforms.resize(cfgBenchmark5_Iterations); |
---|
965 | for(int i=0;i<transforms.size();++i) |
---|
966 | { |
---|
967 | transforms[i]=btDbvtBenchmark::RandTransform(cfgVolumeCenterScale*cfgBenchmark5_OffsetScale); |
---|
968 | } |
---|
969 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt[0]); |
---|
970 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt[1]); |
---|
971 | dbvt[0].optimizeTopDown(); |
---|
972 | dbvt[1].optimizeTopDown(); |
---|
973 | printf("[5] btDbvt::collideTT xform: "); |
---|
974 | wallclock.reset(); |
---|
975 | for(int i=0;i<cfgBenchmark5_Iterations;++i) |
---|
976 | { |
---|
977 | btDbvt::collideTT(dbvt[0].m_root,dbvt[1].m_root,transforms[i],policy); |
---|
978 | } |
---|
979 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
980 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark5_Reference)*100/time); |
---|
981 | } |
---|
982 | if(cfgBenchmark6_Enable) |
---|
983 | {// Benchmark 6 |
---|
984 | srand(380843); |
---|
985 | btDbvt dbvt; |
---|
986 | btAlignedObjectArray<btTransform> transforms; |
---|
987 | btDbvtBenchmark::NilPolicy policy; |
---|
988 | transforms.resize(cfgBenchmark6_Iterations); |
---|
989 | for(int i=0;i<transforms.size();++i) |
---|
990 | { |
---|
991 | transforms[i]=btDbvtBenchmark::RandTransform(cfgVolumeCenterScale*cfgBenchmark6_OffsetScale); |
---|
992 | } |
---|
993 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
994 | dbvt.optimizeTopDown(); |
---|
995 | printf("[6] btDbvt::collideTT xform,self: "); |
---|
996 | wallclock.reset(); |
---|
997 | for(int i=0;i<cfgBenchmark6_Iterations;++i) |
---|
998 | { |
---|
999 | btDbvt::collideTT(dbvt.m_root,dbvt.m_root,transforms[i],policy); |
---|
1000 | } |
---|
1001 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1002 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark6_Reference)*100/time); |
---|
1003 | } |
---|
1004 | if(cfgBenchmark7_Enable) |
---|
1005 | {// Benchmark 7 |
---|
1006 | srand(380843); |
---|
1007 | btDbvt dbvt; |
---|
1008 | btAlignedObjectArray<btVector3> rayorg; |
---|
1009 | btAlignedObjectArray<btVector3> raydir; |
---|
1010 | btDbvtBenchmark::NilPolicy policy; |
---|
1011 | rayorg.resize(cfgBenchmark7_Iterations); |
---|
1012 | raydir.resize(cfgBenchmark7_Iterations); |
---|
1013 | for(int i=0;i<rayorg.size();++i) |
---|
1014 | { |
---|
1015 | rayorg[i]=btDbvtBenchmark::RandVector3(cfgVolumeCenterScale*2); |
---|
1016 | raydir[i]=btDbvtBenchmark::RandVector3(cfgVolumeCenterScale*2); |
---|
1017 | } |
---|
1018 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1019 | dbvt.optimizeTopDown(); |
---|
1020 | printf("[7] btDbvt::collideRAY: "); |
---|
1021 | wallclock.reset(); |
---|
1022 | for(int i=0;i<cfgBenchmark7_Passes;++i) |
---|
1023 | { |
---|
1024 | for(int j=0;j<cfgBenchmark7_Iterations;++j) |
---|
1025 | { |
---|
1026 | btDbvt::collideRAY(dbvt.m_root,rayorg[j],raydir[j],policy); |
---|
1027 | } |
---|
1028 | } |
---|
1029 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1030 | unsigned rays=cfgBenchmark7_Passes*cfgBenchmark7_Iterations; |
---|
1031 | printf("%u ms (%i%%),(%u r/s)\r\n",time,(time-cfgBenchmark7_Reference)*100/time,(rays*1000)/time); |
---|
1032 | } |
---|
1033 | if(cfgBenchmark8_Enable) |
---|
1034 | {// Benchmark 8 |
---|
1035 | srand(380843); |
---|
1036 | btDbvt dbvt; |
---|
1037 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1038 | dbvt.optimizeTopDown(); |
---|
1039 | printf("[8] insert/remove: "); |
---|
1040 | wallclock.reset(); |
---|
1041 | for(int i=0;i<cfgBenchmark8_Passes;++i) |
---|
1042 | { |
---|
1043 | for(int j=0;j<cfgBenchmark8_Iterations;++j) |
---|
1044 | { |
---|
1045 | dbvt.remove(dbvt.insert(btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale),0)); |
---|
1046 | } |
---|
1047 | } |
---|
1048 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1049 | const int ir=cfgBenchmark8_Passes*cfgBenchmark8_Iterations; |
---|
1050 | printf("%u ms (%i%%),(%u ir/s)\r\n",time,(time-cfgBenchmark8_Reference)*100/time,ir*1000/time); |
---|
1051 | } |
---|
1052 | if(cfgBenchmark9_Enable) |
---|
1053 | {// Benchmark 9 |
---|
1054 | srand(380843); |
---|
1055 | btDbvt dbvt; |
---|
1056 | btAlignedObjectArray<const btDbvtNode*> leaves; |
---|
1057 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1058 | dbvt.optimizeTopDown(); |
---|
1059 | dbvt.extractLeaves(dbvt.m_root,leaves); |
---|
1060 | printf("[9] updates (teleport): "); |
---|
1061 | wallclock.reset(); |
---|
1062 | for(int i=0;i<cfgBenchmark9_Passes;++i) |
---|
1063 | { |
---|
1064 | for(int j=0;j<cfgBenchmark9_Iterations;++j) |
---|
1065 | { |
---|
1066 | dbvt.update(const_cast<btDbvtNode*>(leaves[rand()%cfgLeaves]), |
---|
1067 | btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale)); |
---|
1068 | } |
---|
1069 | } |
---|
1070 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1071 | const int up=cfgBenchmark9_Passes*cfgBenchmark9_Iterations; |
---|
1072 | printf("%u ms (%i%%),(%u u/s)\r\n",time,(time-cfgBenchmark9_Reference)*100/time,up*1000/time); |
---|
1073 | } |
---|
1074 | if(cfgBenchmark10_Enable) |
---|
1075 | {// Benchmark 10 |
---|
1076 | srand(380843); |
---|
1077 | btDbvt dbvt; |
---|
1078 | btAlignedObjectArray<const btDbvtNode*> leaves; |
---|
1079 | btAlignedObjectArray<btVector3> vectors; |
---|
1080 | vectors.resize(cfgBenchmark10_Iterations); |
---|
1081 | for(int i=0;i<vectors.size();++i) |
---|
1082 | { |
---|
1083 | vectors[i]=(btDbvtBenchmark::RandVector3()*2-btVector3(1,1,1))*cfgBenchmark10_Scale; |
---|
1084 | } |
---|
1085 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1086 | dbvt.optimizeTopDown(); |
---|
1087 | dbvt.extractLeaves(dbvt.m_root,leaves); |
---|
1088 | printf("[10] updates (jitter): "); |
---|
1089 | wallclock.reset(); |
---|
1090 | |
---|
1091 | for(int i=0;i<cfgBenchmark10_Passes;++i) |
---|
1092 | { |
---|
1093 | for(int j=0;j<cfgBenchmark10_Iterations;++j) |
---|
1094 | { |
---|
1095 | const btVector3& d=vectors[j]; |
---|
1096 | btDbvtNode* l=const_cast<btDbvtNode*>(leaves[rand()%cfgLeaves]); |
---|
1097 | btDbvtVolume v=btDbvtVolume::FromMM(l->volume.Mins()+d,l->volume.Maxs()+d); |
---|
1098 | dbvt.update(l,v); |
---|
1099 | } |
---|
1100 | } |
---|
1101 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1102 | const int up=cfgBenchmark10_Passes*cfgBenchmark10_Iterations; |
---|
1103 | printf("%u ms (%i%%),(%u u/s)\r\n",time,(time-cfgBenchmark10_Reference)*100/time,up*1000/time); |
---|
1104 | } |
---|
1105 | if(cfgBenchmark11_Enable) |
---|
1106 | {// Benchmark 11 |
---|
1107 | srand(380843); |
---|
1108 | btDbvt dbvt; |
---|
1109 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1110 | dbvt.optimizeTopDown(); |
---|
1111 | printf("[11] optimize (incremental): "); |
---|
1112 | wallclock.reset(); |
---|
1113 | for(int i=0;i<cfgBenchmark11_Passes;++i) |
---|
1114 | { |
---|
1115 | dbvt.optimizeIncremental(cfgBenchmark11_Iterations); |
---|
1116 | } |
---|
1117 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1118 | const int op=cfgBenchmark11_Passes*cfgBenchmark11_Iterations; |
---|
1119 | printf("%u ms (%i%%),(%u o/s)\r\n",time,(time-cfgBenchmark11_Reference)*100/time,op/time*1000); |
---|
1120 | } |
---|
1121 | if(cfgBenchmark12_Enable) |
---|
1122 | {// Benchmark 12 |
---|
1123 | srand(380843); |
---|
1124 | btAlignedObjectArray<btDbvtVolume> volumes; |
---|
1125 | btAlignedObjectArray<bool> results; |
---|
1126 | volumes.resize(cfgLeaves); |
---|
1127 | results.resize(cfgLeaves); |
---|
1128 | for(int i=0;i<cfgLeaves;++i) |
---|
1129 | { |
---|
1130 | volumes[i]=btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale); |
---|
1131 | } |
---|
1132 | printf("[12] btDbvtVolume notequal: "); |
---|
1133 | wallclock.reset(); |
---|
1134 | for(int i=0;i<cfgBenchmark12_Iterations;++i) |
---|
1135 | { |
---|
1136 | for(int j=0;j<cfgLeaves;++j) |
---|
1137 | { |
---|
1138 | for(int k=0;k<cfgLeaves;++k) |
---|
1139 | { |
---|
1140 | results[k]=NotEqual(volumes[j],volumes[k]); |
---|
1141 | } |
---|
1142 | } |
---|
1143 | } |
---|
1144 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1145 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark12_Reference)*100/time); |
---|
1146 | } |
---|
1147 | if(cfgBenchmark13_Enable) |
---|
1148 | {// Benchmark 13 |
---|
1149 | srand(380843); |
---|
1150 | btDbvt dbvt; |
---|
1151 | btAlignedObjectArray<btVector3> vectors; |
---|
1152 | btDbvtBenchmark::NilPolicy policy; |
---|
1153 | vectors.resize(cfgBenchmark13_Iterations); |
---|
1154 | for(int i=0;i<vectors.size();++i) |
---|
1155 | { |
---|
1156 | vectors[i]=(btDbvtBenchmark::RandVector3()*2-btVector3(1,1,1)).normalized(); |
---|
1157 | } |
---|
1158 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1159 | dbvt.optimizeTopDown(); |
---|
1160 | printf("[13] culling(OCL+fullsort): "); |
---|
1161 | wallclock.reset(); |
---|
1162 | for(int i=0;i<cfgBenchmark13_Iterations;++i) |
---|
1163 | { |
---|
1164 | static const btScalar offset=0; |
---|
1165 | policy.m_depth=-SIMD_INFINITY; |
---|
1166 | dbvt.collideOCL(dbvt.m_root,&vectors[i],&offset,vectors[i],1,policy); |
---|
1167 | } |
---|
1168 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1169 | const int t=cfgBenchmark13_Iterations; |
---|
1170 | printf("%u ms (%i%%),(%u t/s)\r\n",time,(time-cfgBenchmark13_Reference)*100/time,(t*1000)/time); |
---|
1171 | } |
---|
1172 | if(cfgBenchmark14_Enable) |
---|
1173 | {// Benchmark 14 |
---|
1174 | srand(380843); |
---|
1175 | btDbvt dbvt; |
---|
1176 | btAlignedObjectArray<btVector3> vectors; |
---|
1177 | btDbvtBenchmark::P14 policy; |
---|
1178 | vectors.resize(cfgBenchmark14_Iterations); |
---|
1179 | for(int i=0;i<vectors.size();++i) |
---|
1180 | { |
---|
1181 | vectors[i]=(btDbvtBenchmark::RandVector3()*2-btVector3(1,1,1)).normalized(); |
---|
1182 | } |
---|
1183 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1184 | dbvt.optimizeTopDown(); |
---|
1185 | policy.m_nodes.reserve(cfgLeaves); |
---|
1186 | printf("[14] culling(OCL+qsort): "); |
---|
1187 | wallclock.reset(); |
---|
1188 | for(int i=0;i<cfgBenchmark14_Iterations;++i) |
---|
1189 | { |
---|
1190 | static const btScalar offset=0; |
---|
1191 | policy.m_nodes.resize(0); |
---|
1192 | dbvt.collideOCL(dbvt.m_root,&vectors[i],&offset,vectors[i],1,policy,false); |
---|
1193 | policy.m_nodes.quickSort(btDbvtBenchmark::P14::sortfnc); |
---|
1194 | } |
---|
1195 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1196 | const int t=cfgBenchmark14_Iterations; |
---|
1197 | printf("%u ms (%i%%),(%u t/s)\r\n",time,(time-cfgBenchmark14_Reference)*100/time,(t*1000)/time); |
---|
1198 | } |
---|
1199 | if(cfgBenchmark15_Enable) |
---|
1200 | {// Benchmark 15 |
---|
1201 | srand(380843); |
---|
1202 | btDbvt dbvt; |
---|
1203 | btAlignedObjectArray<btVector3> vectors; |
---|
1204 | btDbvtBenchmark::P15 policy; |
---|
1205 | vectors.resize(cfgBenchmark15_Iterations); |
---|
1206 | for(int i=0;i<vectors.size();++i) |
---|
1207 | { |
---|
1208 | vectors[i]=(btDbvtBenchmark::RandVector3()*2-btVector3(1,1,1)).normalized(); |
---|
1209 | } |
---|
1210 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1211 | dbvt.optimizeTopDown(); |
---|
1212 | policy.m_nodes.reserve(cfgLeaves); |
---|
1213 | printf("[15] culling(KDOP+qsort): "); |
---|
1214 | wallclock.reset(); |
---|
1215 | for(int i=0;i<cfgBenchmark15_Iterations;++i) |
---|
1216 | { |
---|
1217 | static const btScalar offset=0; |
---|
1218 | policy.m_nodes.resize(0); |
---|
1219 | policy.m_axis=vectors[i]; |
---|
1220 | dbvt.collideKDOP(dbvt.m_root,&vectors[i],&offset,1,policy); |
---|
1221 | policy.m_nodes.quickSort(btDbvtBenchmark::P15::sortfnc); |
---|
1222 | } |
---|
1223 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1224 | const int t=cfgBenchmark15_Iterations; |
---|
1225 | printf("%u ms (%i%%),(%u t/s)\r\n",time,(time-cfgBenchmark15_Reference)*100/time,(t*1000)/time); |
---|
1226 | } |
---|
1227 | if(cfgBenchmark16_Enable) |
---|
1228 | {// Benchmark 16 |
---|
1229 | srand(380843); |
---|
1230 | btDbvt dbvt; |
---|
1231 | btAlignedObjectArray<btDbvtNode*> batch; |
---|
1232 | btDbvtBenchmark::RandTree(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale,cfgLeaves,dbvt); |
---|
1233 | dbvt.optimizeTopDown(); |
---|
1234 | batch.reserve(cfgBenchmark16_BatchCount); |
---|
1235 | printf("[16] insert/remove batch(%u): ",cfgBenchmark16_BatchCount); |
---|
1236 | wallclock.reset(); |
---|
1237 | for(int i=0;i<cfgBenchmark16_Passes;++i) |
---|
1238 | { |
---|
1239 | for(int j=0;j<cfgBenchmark16_BatchCount;++j) |
---|
1240 | { |
---|
1241 | batch.push_back(dbvt.insert(btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale),0)); |
---|
1242 | } |
---|
1243 | for(int j=0;j<cfgBenchmark16_BatchCount;++j) |
---|
1244 | { |
---|
1245 | dbvt.remove(batch[j]); |
---|
1246 | } |
---|
1247 | batch.resize(0); |
---|
1248 | } |
---|
1249 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1250 | const int ir=cfgBenchmark16_Passes*cfgBenchmark16_BatchCount; |
---|
1251 | printf("%u ms (%i%%),(%u bir/s)\r\n",time,(time-cfgBenchmark16_Reference)*100/time,int(ir*1000.0/time)); |
---|
1252 | } |
---|
1253 | if(cfgBenchmark17_Enable) |
---|
1254 | {// Benchmark 17 |
---|
1255 | srand(380843); |
---|
1256 | btAlignedObjectArray<btDbvtVolume> volumes; |
---|
1257 | btAlignedObjectArray<int> results; |
---|
1258 | btAlignedObjectArray<int> indices; |
---|
1259 | volumes.resize(cfgLeaves); |
---|
1260 | results.resize(cfgLeaves); |
---|
1261 | indices.resize(cfgLeaves); |
---|
1262 | for(int i=0;i<cfgLeaves;++i) |
---|
1263 | { |
---|
1264 | indices[i]=i; |
---|
1265 | volumes[i]=btDbvtBenchmark::RandVolume(cfgVolumeCenterScale,cfgVolumeExentsBase,cfgVolumeExentsScale); |
---|
1266 | } |
---|
1267 | for(int i=0;i<cfgLeaves;++i) |
---|
1268 | { |
---|
1269 | btSwap(indices[i],indices[rand()%cfgLeaves]); |
---|
1270 | } |
---|
1271 | printf("[17] btDbvtVolume select: "); |
---|
1272 | wallclock.reset(); |
---|
1273 | for(int i=0;i<cfgBenchmark17_Iterations;++i) |
---|
1274 | { |
---|
1275 | for(int j=0;j<cfgLeaves;++j) |
---|
1276 | { |
---|
1277 | for(int k=0;k<cfgLeaves;++k) |
---|
1278 | { |
---|
1279 | const int idx=indices[k]; |
---|
1280 | results[idx]=Select(volumes[idx],volumes[j],volumes[k]); |
---|
1281 | } |
---|
1282 | } |
---|
1283 | } |
---|
1284 | const int time=(int)wallclock.getTimeMilliseconds(); |
---|
1285 | printf("%u ms (%i%%)\r\n",time,(time-cfgBenchmark17_Reference)*100/time); |
---|
1286 | } |
---|
1287 | printf("\r\n\r\n"); |
---|
1288 | } |
---|
1289 | #endif |
---|