1 | // =================================================================== |
---|
2 | // $Id: $ |
---|
3 | // |
---|
4 | // ktbs.cpp |
---|
5 | // The classes for building up CKTB tree by statistical optimization. |
---|
6 | // This building up is based on sampling. |
---|
7 | // |
---|
8 | // REPLACEMENT_STRING |
---|
9 | // |
---|
10 | // Copyright by Vlastimil Havran, 2007 - email to "vhavran AT seznam.cz" |
---|
11 | // Initial coding by Vlasta Havran, January 2008. |
---|
12 | |
---|
13 | // standard headers |
---|
14 | #include <algorithm> |
---|
15 | #include <iostream> |
---|
16 | #include <string> |
---|
17 | |
---|
18 | // GOLEM headers |
---|
19 | #include "ktbs.h" |
---|
20 | #include "timer.h" |
---|
21 | #include "Environment.h" |
---|
22 | |
---|
23 | //#define _DEBUG |
---|
24 | |
---|
25 | #define AVOIDCHECKLIST |
---|
26 | |
---|
27 | #define _USE_OPTIMIZE_DIVIDE_AX |
---|
28 | |
---|
29 | // Note that the RADIX SORT does not work properly with -mtune=pentium3 for GCC.4.0 !!!! |
---|
30 | // Note that the RADIX SORT does not work properly with -mtune=pentium4 for GCC.3.4 !!!! |
---|
31 | // The reason is unknown, probably the bug in compiler |
---|
32 | |
---|
33 | namespace GtpVisibilityPreprocessor { |
---|
34 | |
---|
35 | // ------------------------------------------------ |
---|
36 | // for debugging cost function in the scene |
---|
37 | //#define _DEBUG_COSTFUNCTION |
---|
38 | #ifdef _DEBUG_COSTFUNCTION |
---|
39 | |
---|
40 | const int indexToFindSS = 0; |
---|
41 | |
---|
42 | static CFileIO ffcost; |
---|
43 | static int costEvalCnt; |
---|
44 | static int costCntObjects; |
---|
45 | static int costIndexDebug; |
---|
46 | static string fnameCost; |
---|
47 | static float maxCost; |
---|
48 | static float minCost; |
---|
49 | static float maxPos; |
---|
50 | static float minPos; |
---|
51 | static float minPosAxis; |
---|
52 | static float maxPosAxis; |
---|
53 | static bool _alreadyDebugged = false; |
---|
54 | static bool _streamOpened = false; |
---|
55 | |
---|
56 | static void |
---|
57 | InitCostStream(int indexDebug, int cntObjects, |
---|
58 | float minx, float maxx) |
---|
59 | { |
---|
60 | maxCost = -INFINITY; |
---|
61 | minCost = INFINITY; |
---|
62 | costEvalCnt = 0; |
---|
63 | costIndexDebug = indexDebug; |
---|
64 | costCntObjects = cntObjects; |
---|
65 | minPosAxis = minx; |
---|
66 | maxPosAxis = maxx; |
---|
67 | |
---|
68 | // and init the stream - rewrite |
---|
69 | string name = "debugcostfile" |
---|
70 | //Environment::GetSingleton()->GetBoolValue("OutputFileName", name); |
---|
71 | char lns[100]; |
---|
72 | sprintf(lns, "%05d.res", indexDebug); |
---|
73 | ChangeFilenameExtension(name, string(lns), fnameCost); |
---|
74 | |
---|
75 | ffcost.SetFilename(fnameCost.c_str()); |
---|
76 | if (ffcost.OpenInMode(CFileIO::EE_Mode_w)) { |
---|
77 | cerr << "Opening of stream failed" << endl; |
---|
78 | abort();; |
---|
79 | } |
---|
80 | cout << "Saving debug to " << fnameCost << endl; |
---|
81 | ffcost.WriteChars("\n"); |
---|
82 | sprintf(lns, "#CntObjects = %d\n", costCntObjects); |
---|
83 | ffcost.WriteChars(lns); |
---|
84 | sprintf(lns, "#IndexToDebug = %d\n", costIndexDebug); |
---|
85 | ffcost.WriteChars(lns); |
---|
86 | ffcost.WriteChars("#File "); |
---|
87 | ffcost.WriteChars(fnameCost.c_str()); |
---|
88 | ffcost.WriteChars("\n"); |
---|
89 | ffcost.WriteChars("#==============================================\n"); |
---|
90 | |
---|
91 | cout << "#---------------------------------------" << endl; |
---|
92 | _streamOpened = true; |
---|
93 | } |
---|
94 | |
---|
95 | static void |
---|
96 | CloseCostStream() |
---|
97 | { |
---|
98 | if (_streamOpened) { |
---|
99 | char lns[255]; |
---|
100 | ffcost.WriteChars("#==============================================\n"); |
---|
101 | ffcost.WriteChars("#EndOfFile\n"); |
---|
102 | sprintf(lns, "#minCost = %8.6f minPos = %8.6f\n", minCost, minPos); |
---|
103 | ffcost.WriteChars(lns); |
---|
104 | sprintf(lns, "#maxCost = %8.6f maxPos = %8.6f\n", maxCost, maxPos); |
---|
105 | ffcost.WriteChars(lns); |
---|
106 | sprintf(lns, "#ratioMin/Max = %6.5f\n", |
---|
107 | minCost/maxCost); |
---|
108 | ffcost.WriteChars(lns); |
---|
109 | sprintf(lns, "#CntEval = %d\n", costEvalCnt); |
---|
110 | ffcost.WriteChars(lns); |
---|
111 | sprintf(lns, "#CntObjects = %d\n", costCntObjects); |
---|
112 | ffcost.WriteChars(lns); |
---|
113 | sprintf(lns, "#IndexToDebug = %d\n", costIndexDebug); |
---|
114 | ffcost.WriteChars(lns); |
---|
115 | sprintf(lns, "#MinPosAxis = %8.6f maxPosAxis = %8.6f\n", minPosAxis, maxPosAxis); |
---|
116 | ffcost.WriteChars(lns); |
---|
117 | ffcost.WriteChars("#File "); |
---|
118 | ffcost.WriteChars(fnameCost.c_str()); |
---|
119 | ffcost.WriteChars("\n"); |
---|
120 | ffcost.Close(); |
---|
121 | //cout << "#=======================================" << endl; |
---|
122 | _streamOpened = false; |
---|
123 | _alreadyDebugged = true; |
---|
124 | // Temporarily |
---|
125 | exit(0); |
---|
126 | } |
---|
127 | } |
---|
128 | |
---|
129 | static void |
---|
130 | ReportCostStream(float pos, float cost) |
---|
131 | { |
---|
132 | // Here we normalize a position to the interval <0-1> |
---|
133 | float posn = (pos - minPosAxis)/(maxPosAxis - minPosAxis); |
---|
134 | |
---|
135 | if (_streamOpened) { |
---|
136 | costEvalCnt++; |
---|
137 | if (cost < minCost) { |
---|
138 | minCost = cost; |
---|
139 | minPos = pos; |
---|
140 | } |
---|
141 | if (cost > maxCost) { |
---|
142 | maxCost = cost; |
---|
143 | maxPos = pos; |
---|
144 | } |
---|
145 | char lns[255]; |
---|
146 | sprintf(lns, "%8.6f %8.6f\n", posn, cost); |
---|
147 | //cout << lns << endl; |
---|
148 | ffcost.WriteChars(lns); |
---|
149 | } |
---|
150 | } |
---|
151 | #endif // _DEBUG_COSTFUNCTION |
---|
152 | |
---|
153 | // ------------------------------------------------------- |
---|
154 | // the next two axes for a given one |
---|
155 | const CKTBAxes::Axes |
---|
156 | CKTBSBuildUp::oaxes[3][2]= |
---|
157 | {{CKTBAxes::EE_Y_axis, CKTBAxes::EE_Z_axis}, |
---|
158 | {CKTBAxes::EE_Z_axis, CKTBAxes::EE_X_axis}, |
---|
159 | {CKTBAxes::EE_X_axis, CKTBAxes::EE_Y_axis}}; |
---|
160 | |
---|
161 | //---------------------------------------------------------------- |
---|
162 | //---------- Implementation of CKTB tree with irregular ----------- |
---|
163 | //---------- placed splitting planes ----------------------------- |
---|
164 | //---------------------------------------------------------------- |
---|
165 | |
---|
166 | // default constructor |
---|
167 | CKTBSBuildUp::CKTBSBuildUp() |
---|
168 | { |
---|
169 | // verbose is set |
---|
170 | verbose = true; |
---|
171 | |
---|
172 | // Maximum depth of the tree is set and stack is allocated |
---|
173 | maxTreeDepth = MAX_HEIGHT; |
---|
174 | stackDepth = maxTreeDepth + 2; |
---|
175 | stackID = new GALIGN16 SInputData[stackDepth]; |
---|
176 | assert(stackID); |
---|
177 | stackIndex = 0; |
---|
178 | return; |
---|
179 | } |
---|
180 | |
---|
181 | // destructor |
---|
182 | CKTBSBuildUp::~CKTBSBuildUp() |
---|
183 | { |
---|
184 | delete [] stackID; |
---|
185 | stackID = 0; |
---|
186 | } |
---|
187 | |
---|
188 | void |
---|
189 | CKTBSBuildUp::ProvideID(ostream &app) |
---|
190 | { |
---|
191 | //bool tempvar; |
---|
192 | |
---|
193 | string termCrit; |
---|
194 | Environment::GetSingleton()->GetStringValue("BSP.termCrit", termCrit); |
---|
195 | app << "#BSP.termCrit\t\tTermination criteria to build up BSP tree\n"; |
---|
196 | app << termCrit << "\n"; |
---|
197 | |
---|
198 | maxCountTrials = 0; |
---|
199 | if ( (!termCrit.compare("auto")) && |
---|
200 | (!termCrit.compare("auto2")) ) { |
---|
201 | // everything except auto settings, when depth is determined |
---|
202 | // It should be reported for adhoc and possibly others |
---|
203 | Environment::GetSingleton()->GetIntValue("BSP.maxDepthAllowed", |
---|
204 | maxDepthAllowed); |
---|
205 | Environment::GetSingleton()->GetIntValue("BSP.maxListLength", |
---|
206 | maxListLength); |
---|
207 | app << "#MaxDepth (Dmax)\tMaximal allowed depth of the BSP tree\n"; |
---|
208 | app << maxDepthAllowed << "\n"; |
---|
209 | app << "#MaxListLength (Noilf)\tMaximal number of solids in one cell\n"; |
---|
210 | app << maxListLength << "\n"; |
---|
211 | // maximum number of trials to further subdivide |
---|
212 | maxCountTrials = (int)(1.0 + 0.2 * (float)maxDepthAllowed); |
---|
213 | if (maxCountTrials < 3) |
---|
214 | maxCountTrials = 3; |
---|
215 | } |
---|
216 | else { |
---|
217 | if (!termCrit.compare("auto2") ) { |
---|
218 | Environment::GetSingleton()->GetIntValue("BSP.maxListLength", |
---|
219 | maxListLength); |
---|
220 | app << "#MaxDepth (Dmax)\tMaximal allowed depth of the BSP tree\n"; |
---|
221 | } |
---|
222 | } |
---|
223 | |
---|
224 | float eCt, eCi, eCd; |
---|
225 | Environment::GetSingleton()->GetFloatValue("BSP.decisionCost", eCd); |
---|
226 | Environment::GetSingleton()->GetFloatValue("BSP.intersectionCost", eCi); |
---|
227 | Environment::GetSingleton()->GetFloatValue("BSP.traversalCost", eCt); |
---|
228 | |
---|
229 | // Ci should no be changed from 1.0, only Cd and Ct should be changed |
---|
230 | app << "#SetBSP.(Cd, Ci, Ct) \tDecision, Intersection, "; |
---|
231 | app << "and Traversal costs\n"; |
---|
232 | app << " ( " << eCd << ", " << eCi << " ," << eCt << " )\n"; |
---|
233 | |
---|
234 | return; |
---|
235 | } |
---|
236 | |
---|
237 | // Init required parameters |
---|
238 | void |
---|
239 | CKTBSBuildUp::InitReqPref(SReqPrefParams *pars) |
---|
240 | { |
---|
241 | // nothing is obligatory as default |
---|
242 | pars->reqPosition = Limits::Infinity; |
---|
243 | pars->reqAxis = CKTBAxes::EE_Leaf; |
---|
244 | pars->useReqAxis = false; |
---|
245 | |
---|
246 | // the values for automatic termination criteria, since it was |
---|
247 | // not subdivided |
---|
248 | pars->ratioLast = 1000.0; |
---|
249 | pars->ratioLastButOne = 1000.0; |
---|
250 | pars->failedSubDivCount = 0; |
---|
251 | |
---|
252 | Environment::GetSingleton()->GetIntValue("BSP.axisSelectionAlg", |
---|
253 | _algorithmForAxisSelection); |
---|
254 | if (_algorithmForAxisSelection != 0) { |
---|
255 | // The axis as prefered one - testing all 3 axes. Start |
---|
256 | // from the axis cutting the longest side of the box |
---|
257 | pars->reqAxis = (CKTBAxes::Axes)(wBbox.Diagonal().DrivingAxis()); |
---|
258 | } |
---|
259 | |
---|
260 | return; |
---|
261 | } |
---|
262 | |
---|
263 | // creates all the auxiliary structures for building up CKTB tree |
---|
264 | CKTBSBuildUp::SInputData* |
---|
265 | CKTBSBuildUp::Init(ObjectContainer *objlist, const AxisAlignedBox3 &box) |
---|
266 | { |
---|
267 | #ifdef _RANDOMIZE_POSITION |
---|
268 | //world->_environment.GetFloat("SetBSP.randomizePosition.Eps", randomEps); |
---|
269 | #endif // _RANDOMIZE_POSITION |
---|
270 | |
---|
271 | #ifdef _KTB_CONSTR_STATS |
---|
272 | _stats_interiorCount = 0; |
---|
273 | _stats_bboxCount = 0; |
---|
274 | _stats_leafNodeCount = 0; |
---|
275 | _stats_emptyLeftNodeCount = 0; |
---|
276 | // Aggregate statistics |
---|
277 | _maxDepth = 0; |
---|
278 | _sumLeafDepth = 0; |
---|
279 | _sumFullLeafDepth = 0; |
---|
280 | _sumObjectRefCount = 0; |
---|
281 | _maxObjectRefInLeaf = 0; |
---|
282 | // surface areas |
---|
283 | _sumSurfaceAreaLeaves = 0.f; |
---|
284 | _sumSurfaceAreaMULcntLeaves = 0.f; |
---|
285 | _sumSurfaceAreaInteriorNodes = 0.f; |
---|
286 | #endif |
---|
287 | |
---|
288 | // If to print out the tree during contruction |
---|
289 | Environment::GetSingleton()->GetBoolValue("BSP.printCuts", _printCuts); |
---|
290 | |
---|
291 | // if to cut off empty space in postprocessing and how it is performed |
---|
292 | Environment::GetSingleton()->GetIntValue("BSP.absMaxAllowedDepth", |
---|
293 | absMaxAllowedDepth); |
---|
294 | Environment::GetSingleton()->GetIntValue("BSP.maxEmptyCutDepth", |
---|
295 | maxEmptyCutDepth); |
---|
296 | Environment::GetSingleton()->GetIntValue("BSP.algAutoTermination", |
---|
297 | algorithmAutoTermination); |
---|
298 | Environment::GetSingleton()->GetFloatValue("BSP.biasFreeCuts", |
---|
299 | biasFreeCuts); |
---|
300 | |
---|
301 | // if to make tagging lists inside the tree |
---|
302 | Environment::GetSingleton()->GetBoolValue("BSP.minBoxes.use", |
---|
303 | makeMinBoxes); |
---|
304 | Environment::GetSingleton()->GetBoolValue("BSP.minBoxes.tight", |
---|
305 | makeTightMinBoxes); |
---|
306 | Environment::GetSingleton()->GetIntValue("BSP.minBoxes.minObjects", |
---|
307 | minObjectsToCreateMinBox); |
---|
308 | Environment::GetSingleton()->GetIntValue("BSP.minBoxes.minDepthDistance", |
---|
309 | minDepthDistanceBetweenMinBoxes); |
---|
310 | Environment::GetSingleton()->GetFloatValue("BSP.minBoxes.minSA2ratio", |
---|
311 | minSA2ratioMinBoxes); |
---|
312 | |
---|
313 | // How many items can be allocated at once |
---|
314 | int maxItemsAtOnce = 1; |
---|
315 | if (makeMinBoxes) { |
---|
316 | #ifdef _KTB8Bytes |
---|
317 | // We need to allocate for boxes the memory in a row |
---|
318 | maxItemsAtOnce = 5; // 8x5=40 = 16+24; |
---|
319 | #else |
---|
320 | maxItemsAtOnce = 4; // 12x4=48 = 24+24; |
---|
321 | #endif // _KTB8Bytes |
---|
322 | assert(minObjectsToCreateMinBox >= 1); |
---|
323 | assert(minDepthDistanceBetweenMinBoxes >= 0); |
---|
324 | assert(minDepthDistanceBetweenMinBoxes <= 50); |
---|
325 | } |
---|
326 | |
---|
327 | // the array of preferred parameters used as a stack |
---|
328 | pars = new SReqPrefParams[MAX_HEIGHT]; |
---|
329 | assert(pars); |
---|
330 | |
---|
331 | // Initiate the allocator |
---|
332 | InitAux(0, MAX_HEIGHT - 1, maxItemsAtOnce); |
---|
333 | |
---|
334 | // since six planes are enough to cull empty space from leaves |
---|
335 | if ( (maxEmptyCutDepth < 0) || |
---|
336 | (maxEmptyCutDepth > 6) ) { |
---|
337 | cerr << "BSP.maxEmptyCutDepth = " << maxEmptyCutDepth |
---|
338 | << " must be in range <0,6>\n"; |
---|
339 | abort();; |
---|
340 | } |
---|
341 | |
---|
342 | wBbox.Convert(box); // the box of the whole scene |
---|
343 | boxSize = box.Diagonal(); // the size of the box along the axes |
---|
344 | |
---|
345 | wholeBoxArea = wBbox.SA2(); // the half of the surface area of the box |
---|
346 | // the array of bounding boxes and duplications to the objects |
---|
347 | int objlistcnt = objlist->size(); |
---|
348 | |
---|
349 | SInputData* data = AllocNewData(); |
---|
350 | // set the box |
---|
351 | data->box = wBbox; |
---|
352 | data->tightbox = wBbox; |
---|
353 | data->objlist = new ObjectContainer(objlist->begin(), objlist->end()); |
---|
354 | data->count = objlistcnt; |
---|
355 | |
---|
356 | // Initialization of the first value to insert the min boxes |
---|
357 | data->lastMinBoxSA2 = box.SurfaceArea() * 10000.f; |
---|
358 | data->lastDepthForMinBoxes = -20; // for root you always put the node |
---|
359 | data->lastMinBoxNode = 0; |
---|
360 | |
---|
361 | // Init the parameters from the environment file |
---|
362 | InitReqPref(&(data->pars)); |
---|
363 | |
---|
364 | // the number of buckets; |
---|
365 | state.bucketN = 128; |
---|
366 | state.bucketMin = new int[state.bucketN]; |
---|
367 | assert(state.bucketMin); |
---|
368 | state.bucketMax = new int[state.bucketN]; |
---|
369 | assert(state.bucketMin); |
---|
370 | |
---|
371 | // Init the termination criteria |
---|
372 | string termCrit; |
---|
373 | Environment::GetSingleton()->GetStringValue("BSP.termCrit", termCrit); |
---|
374 | |
---|
375 | maxCountTrials = 0; |
---|
376 | if (!termCrit.compare("adhoc")) { |
---|
377 | // termination criteria are the max depth of the hierarchy, number |
---|
378 | // of primitives |
---|
379 | data->modeSubDiv = EE_SUBDIVADHOC; |
---|
380 | Environment::GetSingleton()->GetIntValue("BSP.maxListLength", |
---|
381 | maxListLength); |
---|
382 | Environment::GetSingleton()->GetIntValue("BSP.maxDepthAllowed", |
---|
383 | maxDepthAllowed); |
---|
384 | } |
---|
385 | else { |
---|
386 | if ( (!termCrit.compare("auto")) || |
---|
387 | (!termCrit.compare("auto2")) ) { |
---|
388 | // automatic termination criteria are used, everything is computed |
---|
389 | // automatically from number of objects etc. |
---|
390 | int estHeight = (int)(log((float)initcnt)/log((float)2.0) + 0.9); |
---|
391 | // cout << "EstHeight=" << estHeight << endl; |
---|
392 | maxDepthAllowed = (int)((float)estHeight * 1.15f + 1.6f); |
---|
393 | |
---|
394 | // maximum number of trials to further subdivide |
---|
395 | maxCountTrials = (int)(1.0 + 0.2 * (float)maxDepthAllowed); |
---|
396 | if (maxCountTrials < 3) |
---|
397 | maxCountTrials = 3; |
---|
398 | |
---|
399 | // for 'auto2' we specify the length of the object list by hand |
---|
400 | if (!termCrit.compare("auto2")) |
---|
401 | Environment::GetSingleton()->GetIntValue("BSP.maxListLength", |
---|
402 | maxListLength); |
---|
403 | else |
---|
404 | // for 'auto' we set the length of the object list that is supposed |
---|
405 | // to be the best for ray shooting |
---|
406 | maxListLength = 1; |
---|
407 | |
---|
408 | if (verbose) { |
---|
409 | cout << "TERMCRIT:maximum height of a leaf set to " |
---|
410 | << maxDepthAllowed << "\n"; |
---|
411 | cout << "TERMCRIT:maximum number of objects in leaf set to " |
---|
412 | << maxListLength << "\n" ; |
---|
413 | } |
---|
414 | // set the mode to be recognized |
---|
415 | data->modeSubDiv = EE_SUBDIVAUTOMATIC; |
---|
416 | } |
---|
417 | else { |
---|
418 | cerr << " unknown termination criteria for BSP tree\n"; |
---|
419 | cerr << "It was specified: " << termCrit << "\n"; |
---|
420 | cerr << "Allowed types = auto, auto2, adhoc" << endl; |
---|
421 | exit(3);; |
---|
422 | } |
---|
423 | } |
---|
424 | |
---|
425 | // for some evaluation it is necessary to determine the following costs |
---|
426 | Environment::GetSingleton()->GetFloatValue("BSP.traversalCost", Ct); |
---|
427 | Environment::GetSingleton()->GetFloatValue("BSP.intersectionCost", Ci); |
---|
428 | if (verbose) |
---|
429 | cout << "Ct=" << Ct << " Ci=" << Ci << "\n"; |
---|
430 | |
---|
431 | if (cutEmptySpace) { |
---|
432 | // correct the maximum abs depth, when late cutting is allowed |
---|
433 | if (absMaxAllowedDepth < maxDepthAllowed) |
---|
434 | absMaxAllowedDepth = maxDepthAllowed; |
---|
435 | if (absMaxAllowedDepth > (maxDepthAllowed + maxEmptyCutDepth) ) |
---|
436 | absMaxAllowedDepth = maxDepthAllowed + maxEmptyCutDepth; |
---|
437 | if (absMaxAllowedDepth >= MAX_HEIGHT) { |
---|
438 | absMaxAllowedDepth = MAX_HEIGHT; |
---|
439 | maxDepthAllowed = MAX_HEIGHT - maxEmptyCutDepth; |
---|
440 | } |
---|
441 | |
---|
442 | if (verbose) { |
---|
443 | cout << "Cutting off empty spaces ON\n"; |
---|
444 | cout << "BSP.absMaxAllowedDepth = " << absMaxAllowedDepth << "\n"; |
---|
445 | cout << "BSP.maxEmptyCutDepth = " << maxEmptyCutDepth << endl; |
---|
446 | } |
---|
447 | } |
---|
448 | |
---|
449 | if (verbose) |
---|
450 | cout << "MaxListLength = " << maxListLength << endl; |
---|
451 | |
---|
452 | return data; |
---|
453 | } |
---|
454 | |
---|
455 | // interface function for building up CKTB tree |
---|
456 | SKTBNodeT* |
---|
457 | CKTBSBuildUp::BuildUp(ObjectContainer &objlist, |
---|
458 | const AxisAlignedBox3 &box, |
---|
459 | bool verboseF) |
---|
460 | { |
---|
461 | // check the number of objects |
---|
462 | if (objlist.size() == 0) |
---|
463 | return 0; // nothing |
---|
464 | |
---|
465 | // --------------------------------------------------- |
---|
466 | // Rewriting the triangles to the just wrappers to save |
---|
467 | // the memory during building!!!! |
---|
468 | bool useWrappers = false; |
---|
469 | if (useWrappers) { |
---|
470 | cout << "WARNING: using only wrappers, not objects to save the memory!" |
---|
471 | << endl; |
---|
472 | cout << "Size of(AxisAlignedBox3Intersectable) = " << sizeof(AxisAlignedBox3Intersectable) << endl; |
---|
473 | cout << "Size of(TriangleIntersectable) = " << sizeof(TriangleIntersectable) << endl; |
---|
474 | for (ObjectContainer::iterator it = objlist.begin(); |
---|
475 | it != objlist.end(); it++) { |
---|
476 | // take the triangle |
---|
477 | Intersectable *i = *it; |
---|
478 | // store the properties to new variables |
---|
479 | AxisAlignedBox3 b = i->GetBox(); |
---|
480 | int mId = i->mId; |
---|
481 | // delete the triangle |
---|
482 | delete i; |
---|
483 | // create the wrapper with the same box as triangle |
---|
484 | AxisAlignedBox3Intersectable *a = new AxisAlignedBox3Intersectable(b); |
---|
485 | a->mId = mId; |
---|
486 | // and put it back to the list of objects |
---|
487 | *it = a; |
---|
488 | } // for |
---|
489 | cout << "Rewriting to wrappers is finished!" << endl; |
---|
490 | } // if |
---|
491 | // --------------------------------------------------- |
---|
492 | |
---|
493 | initcnt = objlist.size(); |
---|
494 | |
---|
495 | // copy verbose to be used consistenly further on |
---|
496 | this->verbose = verboseF; |
---|
497 | |
---|
498 | // the boundary entries |
---|
499 | SInputData *d = Init(&objlist, box); |
---|
500 | |
---|
501 | |
---|
502 | if (verbose) |
---|
503 | cout << "Building up KTB tree for " << objlist.size() |
---|
504 | << " objects by sampling." << endl << flush; |
---|
505 | |
---|
506 | // ----------------------------------------------- |
---|
507 | // Start to build the tree |
---|
508 | if ( (cutEmptySpace) && |
---|
509 | (initcnt <= maxListLength)) { |
---|
510 | // only cutting off empty space for initial leaf |
---|
511 | d->modeSubDiv = EE_SUBDIVCUTEMPTY; |
---|
512 | root = SubDiv(d); |
---|
513 | } |
---|
514 | else { |
---|
515 | // normal subdivision scheme, creating whole tree |
---|
516 | root = SubDiv(d); |
---|
517 | } |
---|
518 | |
---|
519 | #ifdef _DEBUG |
---|
520 | if (GetDepth() != 0) { |
---|
521 | cerr << "Using depth value does not work correctly, depth = " |
---|
522 | << GetDepth() << endl; |
---|
523 | } |
---|
524 | #endif |
---|
525 | assert(root); |
---|
526 | |
---|
527 | // the pointer to the root node |
---|
528 | return GetRootNode(); |
---|
529 | } |
---|
530 | |
---|
531 | int |
---|
532 | CKTBSBuildUp::UpdateEvaluation(float &eval, const float &newEval) |
---|
533 | { |
---|
534 | if (newEval < eval) { |
---|
535 | eval = newEval; |
---|
536 | return 1; // updated |
---|
537 | } |
---|
538 | return 0; // not updated |
---|
539 | } |
---|
540 | |
---|
541 | // recursive function for subdividing the CKTB tree into two |
---|
542 | // halves .. creates one node of CKTB tree |
---|
543 | // list .. list of boundaries, count .. number of objects in current node |
---|
544 | // bb .. current bounding box of node, (pars,reqGlobAxis, modeSubDiv) .. cut pref. |
---|
545 | SKTBNodeT* |
---|
546 | CKTBSBuildUp::SubDiv(SInputData *d) |
---|
547 | { |
---|
548 | #if 0 |
---|
549 | static int index = 0; |
---|
550 | cout << "SubDiv index = " << index << endl; |
---|
551 | const int indexToFind = 13916; |
---|
552 | if (index == indexToFind) { |
---|
553 | cout << " IndexToFind = " << indexToFind << endl; |
---|
554 | cout << " You should start debug" << endl; |
---|
555 | } |
---|
556 | index++; |
---|
557 | #endif |
---|
558 | |
---|
559 | // This is the node to return from this function |
---|
560 | SKTBNodeT *nodeToReturn = 0; |
---|
561 | |
---|
562 | // if to make the interior node extended by the box here |
---|
563 | makeMinBoxHere = false; |
---|
564 | |
---|
565 | // ----------------------------------------------------- |
---|
566 | if (makeMinBoxes) { |
---|
567 | if ( (d->count >= minObjectsToCreateMinBox) && |
---|
568 | (GetDepth() - d->lastDepthForMinBoxes >= minDepthDistanceBetweenMinBoxes) ) { |
---|
569 | makeMinBoxHere = true; |
---|
570 | d->lastDepthForMinBoxes = GetDepth(); |
---|
571 | d->lastMinBoxSA2 = d->box.SA2(); |
---|
572 | } |
---|
573 | } |
---|
574 | |
---|
575 | if ( (makeMinBoxHere) && (makeTightMinBoxes) ) { |
---|
576 | // In case we should make the box of the current |
---|
577 | // interior node and this should be tight, we have |
---|
578 | // to first compute its extent before splitting |
---|
579 | SBBox tightbox; |
---|
580 | GetTightBox(*d, tightbox); |
---|
581 | // Here we decrease the box to the minimum size |
---|
582 | d->box = tightbox; |
---|
583 | } |
---|
584 | |
---|
585 | // ------------------------------------------------------------------ |
---|
586 | // if we are forced to make subdivision at given point |
---|
587 | if (d->pars.reqPosition != Limits::Infinity) { |
---|
588 | // this code preceeds switch(mode) since 2-level evaluation |
---|
589 | d->position = d->pars.reqPosition; |
---|
590 | d->axis = d->pars.reqAxis; |
---|
591 | // next subdivison will not be forced |
---|
592 | d->pars.reqPosition = Limits::Infinity; |
---|
593 | d->pars.reqAxis = CKTBAxes::EE_Leaf; |
---|
594 | IncDepth(); |
---|
595 | SKTBNodeT* n = MakeOneCut(d); |
---|
596 | if (!nodeToReturn) |
---|
597 | nodeToReturn = n; |
---|
598 | DecDepth(); |
---|
599 | return nodeToReturn; |
---|
600 | } |
---|
601 | |
---|
602 | // ------------------------------------------------------------------------- |
---|
603 | // determine between subdivision modes - different termination criteria |
---|
604 | switch (d->modeSubDiv) { // subdivision mode |
---|
605 | case EE_SUBDIVCUTEMPTY: { // cutting off empty space |
---|
606 | if ((GetDepth() >= absMaxAllowedDepth) || |
---|
607 | (GetDepth() >= (startEmptyCutDepth + maxEmptyCutDepth)) || |
---|
608 | (d->count == 0) ) { |
---|
609 | SKTBNodeT* n = MakeLeaf(d); |
---|
610 | if (!nodeToReturn) |
---|
611 | nodeToReturn = n; |
---|
612 | // cout << "LEC\n"; |
---|
613 | return nodeToReturn; |
---|
614 | } |
---|
615 | // to require the axis has no meaning in cutting off |
---|
616 | d->pars.reqAxis = d->axis = CKTBAxes::EE_Leaf; |
---|
617 | break; |
---|
618 | } |
---|
619 | // both adhoc, clustering and automatic termination criteria |
---|
620 | case EE_SUBDIVADHOC: |
---|
621 | case EE_SUBDIVAUTOMATIC: { |
---|
622 | // automatic termination criteria are used in this phase as |
---|
623 | // the absolute threshold similarly to SUBDIVADHOC mode |
---|
624 | if ( (GetDepth() >= maxDepthAllowed) || |
---|
625 | (d->count <= maxListLength)) { |
---|
626 | SKTBNodeT* n = MakeLeaf(d); |
---|
627 | if (!nodeToReturn) |
---|
628 | nodeToReturn = n; |
---|
629 | // cout << "LLA\n"; |
---|
630 | return nodeToReturn; |
---|
631 | } |
---|
632 | break; |
---|
633 | } |
---|
634 | default: { |
---|
635 | cerr << " unknown subdivision mode in ibsp.cpp\n"; |
---|
636 | abort();; |
---|
637 | } |
---|
638 | } // switch(modeSubDiv) |
---|
639 | |
---|
640 | // the axis where splitting should be done |
---|
641 | CKTBAxes::Axes axis = CKTBAxes::EE_Leaf; |
---|
642 | d->twoSplits = -1; |
---|
643 | d->bestCost = WorstEvaluation() * 0.5f; |
---|
644 | d->position = MAXFLOAT; |
---|
645 | const bool secondPositionMax = true; |
---|
646 | |
---|
647 | #define CallGetSplitPlane GetSplitPlaneOpt |
---|
648 | |
---|
649 | // if the axis is required in pars structure for some reasons |
---|
650 | if (d->pars.useReqAxis) { |
---|
651 | axis = d->pars.reqAxis; |
---|
652 | // The next subdivision is not prescribed |
---|
653 | d->pars.useReqAxis = false; |
---|
654 | switch(axis) { |
---|
655 | case CKTBAxes::EE_X_axis : { |
---|
656 | state.InitXaxis(d->count, d->box); // init |
---|
657 | CallGetSplitPlane(d, 0); // evaluate |
---|
658 | break; |
---|
659 | } |
---|
660 | case CKTBAxes::EE_Y_axis : { |
---|
661 | state.InitYaxis(d->count, d->box); // init |
---|
662 | CallGetSplitPlane(d, 1); // evaluate |
---|
663 | break; |
---|
664 | } |
---|
665 | case CKTBAxes::EE_Z_axis : { |
---|
666 | state.InitZaxis(d->count, d->box); // init |
---|
667 | CallGetSplitPlane(d, 2); // evaluate |
---|
668 | break; |
---|
669 | } |
---|
670 | default: { |
---|
671 | cerr << "No other option is allowed here" << endl; |
---|
672 | abort();; |
---|
673 | } |
---|
674 | } |
---|
675 | // compute the cost, normalized by surface area |
---|
676 | state.NormalizeCostBySA2(); |
---|
677 | d->bestCost /= state.areaWholeSA2; |
---|
678 | if (UpdateEvaluation(d->bestCost, state.bestCost)) { |
---|
679 | // Compute correct position to be used for splitting |
---|
680 | d->position = state.bestPosition; |
---|
681 | d->twoSplits = state.bestTwoSplits; // one or two splitting planes planed |
---|
682 | if (d->twoSplits > 1) { |
---|
683 | if (secondPositionMax) |
---|
684 | d->position2 = state.bestPosition2; |
---|
685 | else |
---|
686 | d->position2 = state.box.Max(axis); |
---|
687 | } |
---|
688 | } |
---|
689 | } |
---|
690 | else { |
---|
691 | int algorithmForAxisSel = _algorithmForAxisSelection; |
---|
692 | if (d->modeSubDiv == EE_SUBDIVCUTEMPTY) |
---|
693 | algorithmForAxisSel = 0; |
---|
694 | |
---|
695 | // only a single axis will be tested |
---|
696 | bool useSingleAxis = true; |
---|
697 | |
---|
698 | // the required axis is by global function .. e.g. cyclic change x, y, z |
---|
699 | switch (algorithmForAxisSel) { |
---|
700 | case 0: { |
---|
701 | // all three axis will be used, starting from x, y, z |
---|
702 | useSingleAxis = false; |
---|
703 | break; |
---|
704 | } |
---|
705 | case 1: { |
---|
706 | // cyclic order of axes, x, y, z, x, y.... |
---|
707 | axis = d->pars.reqAxis; |
---|
708 | // compute the axis for the next subdivision step |
---|
709 | d->pars.reqAxis = oaxes[axis][0]; |
---|
710 | break; |
---|
711 | } |
---|
712 | case 2 : { |
---|
713 | // The next time will be determined the same way |
---|
714 | axis = (CKTBAxes::Axes)(d->box.Diagonal().DrivingAxis()); |
---|
715 | break; |
---|
716 | } |
---|
717 | case 3 : { |
---|
718 | float p = RandomValue(0.0f, 1.0f); |
---|
719 | const float thresholdAxisP = 0.8f; |
---|
720 | if (p < thresholdAxisP) { |
---|
721 | axis = (CKTBAxes::Axes)(d->box.Diagonal().DrivingAxis()); |
---|
722 | } |
---|
723 | else { |
---|
724 | // Use the axis prescribed from previous step |
---|
725 | axis = d->pars.reqAxis; |
---|
726 | } |
---|
727 | // for the next time, different axis |
---|
728 | d->pars.reqAxis = oaxes[axis][0]; |
---|
729 | break; |
---|
730 | } |
---|
731 | case 4 : { |
---|
732 | float p = RandomValue(0.0f, 1.0f); |
---|
733 | const float thresholdAxisP = 0.3f; |
---|
734 | if (p < thresholdAxisP) { |
---|
735 | axis = (CKTBAxes::Axes)(d->box.Diagonal().DrivingAxis()); |
---|
736 | } |
---|
737 | else { |
---|
738 | // Use the axis prescribed from previous step |
---|
739 | axis = d->pars.reqAxis; |
---|
740 | } |
---|
741 | // for the next time, different axis |
---|
742 | d->pars.reqAxis = oaxes[axis][0]; |
---|
743 | break; |
---|
744 | } |
---|
745 | case 5 : { |
---|
746 | float p = RandomValue(0.0f, 1.0f); |
---|
747 | const float thresholdAxisP = 0.2f; |
---|
748 | d->pars.reqAxis = (CKTBAxes::Axes)-1; |
---|
749 | if (p < thresholdAxisP) { |
---|
750 | axis = (CKTBAxes::Axes)(d->box.Diagonal().DrivingAxis()); |
---|
751 | // for the next time, different axis |
---|
752 | } |
---|
753 | else { |
---|
754 | // Use the axis for which cost model gets minimum, |
---|
755 | // compute it for all axes x, y, z |
---|
756 | useSingleAxis = false; |
---|
757 | } |
---|
758 | break; |
---|
759 | } |
---|
760 | default: { |
---|
761 | cerr << "Selection algorithm = " << algorithmForAxisSel |
---|
762 | << " for axis is not implemented" << endl; |
---|
763 | abort();; |
---|
764 | } |
---|
765 | } // switch |
---|
766 | |
---|
767 | // How the algorithm is used |
---|
768 | if (useSingleAxis) { |
---|
769 | // ------------------------------------------- |
---|
770 | // Compute subdivision only for one axis |
---|
771 | switch(axis) { |
---|
772 | case CKTBAxes::EE_X_axis : { |
---|
773 | state.InitXaxis(d->count, d->box); // init |
---|
774 | CallGetSplitPlane(d, 0); // evaluate |
---|
775 | break; |
---|
776 | } |
---|
777 | case CKTBAxes::EE_Y_axis : { |
---|
778 | state.InitYaxis(d->count, d->box); // init |
---|
779 | CallGetSplitPlane(d, 1); // evaluate |
---|
780 | break; |
---|
781 | } |
---|
782 | case CKTBAxes::EE_Z_axis : { |
---|
783 | state.InitZaxis(d->count, d->box); // init |
---|
784 | CallGetSplitPlane(d, 2); // evaluate |
---|
785 | break; |
---|
786 | } |
---|
787 | //case CKTBAxes::EE_Leaf : { |
---|
788 | //goto MAKE_LEAF; |
---|
789 | //} |
---|
790 | default: { |
---|
791 | cerr << "Selection algorithm = " << algorithmForAxisSel |
---|
792 | << " for axis = " << axis <<" is not implemented" << endl; |
---|
793 | abort();; |
---|
794 | } |
---|
795 | } |
---|
796 | state.NormalizeCostBySA2(); |
---|
797 | d->bestCost /= state.areaWholeSA2; |
---|
798 | if (UpdateEvaluation(d->bestCost, state.bestCost)) { |
---|
799 | // Compute correct position to be used for splitting |
---|
800 | d->position = state.bestPosition; |
---|
801 | d->twoSplits = state.bestTwoSplits; // one or two splitting planes planed |
---|
802 | if (d->twoSplits > 1) { |
---|
803 | if (secondPositionMax) |
---|
804 | d->position2 = state.bestPosition2; |
---|
805 | else |
---|
806 | d->position2 = state.box.Max(axis); |
---|
807 | } |
---|
808 | } |
---|
809 | } |
---|
810 | else { |
---|
811 | // no axis is required, we can pickup any we like. We test all three axes |
---|
812 | // and we pickup the minimum cost for all three axes. |
---|
813 | state.InitXaxis(d->count, d->box); // init for x axis |
---|
814 | CallGetSplitPlane(d, 0); // evaluate for x axis |
---|
815 | if (UpdateEvaluation(d->bestCost, state.bestCost)) { |
---|
816 | axis = CKTBAxes::EE_X_axis; // the x-axis should be used |
---|
817 | // Compute correct position to be used for splitting |
---|
818 | d->bestCost = state.bestCost; |
---|
819 | d->position = state.bestPosition; |
---|
820 | d->twoSplits = state.bestTwoSplits; // one or two splitting planes planed |
---|
821 | if (d->twoSplits > 1) { |
---|
822 | if (secondPositionMax) |
---|
823 | d->position2 = state.bestPosition2; |
---|
824 | else |
---|
825 | d->position2 = state.box.Max(axis); |
---|
826 | } |
---|
827 | } |
---|
828 | // ----------------------- |
---|
829 | // now test for y axis |
---|
830 | state.InitYaxis(d->count, d->box); // init for x axis |
---|
831 | CallGetSplitPlane(d, 1); // evaluate for x axis |
---|
832 | if (UpdateEvaluation(d->bestCost, state.bestCost)) { |
---|
833 | axis = CKTBAxes::EE_Y_axis; // the y-axis should be used |
---|
834 | // Compute correct position to be used for splitting |
---|
835 | d->bestCost = state.bestCost; |
---|
836 | d->position = state.bestPosition; |
---|
837 | d->twoSplits = state.bestTwoSplits; // one or two splitting planes planed |
---|
838 | if (d->twoSplits > 1) { |
---|
839 | if (secondPositionMax) |
---|
840 | d->position2 = state.bestPosition2; |
---|
841 | else |
---|
842 | d->position2 = state.box.Max(axis); |
---|
843 | } |
---|
844 | } |
---|
845 | // ----------------------- |
---|
846 | // now test for z axis |
---|
847 | state.InitZaxis(d->count, d->box); // init for x axis |
---|
848 | CallGetSplitPlane(d, 2); // evaluate for x axis |
---|
849 | if (UpdateEvaluation(d->bestCost, state.bestCost)) { |
---|
850 | axis = CKTBAxes::EE_Z_axis; // the z-axis should be used |
---|
851 | // Compute correct position to be used for splitting |
---|
852 | d->bestCost = state.bestCost; |
---|
853 | d->position = state.bestPosition; |
---|
854 | d->twoSplits = state.bestTwoSplits; // one or two splitting planes planed |
---|
855 | if (d->twoSplits > 1) { |
---|
856 | if (secondPositionMax) |
---|
857 | d->position2 = state.bestPosition2; |
---|
858 | else |
---|
859 | d->position2 = state.box.Max(axis); |
---|
860 | } |
---|
861 | } |
---|
862 | // Now we have to renormalize the cost for purpose of termination the building |
---|
863 | d->bestCost /= state.areaWholeSA2; |
---|
864 | } // arbitrary order of axes |
---|
865 | } // for which axis to compute surface area heuristics |
---|
866 | |
---|
867 | // ---------------------------------------------------------- |
---|
868 | // when automatic termination criteria should be used |
---|
869 | if (d->modeSubDiv == EE_SUBDIVAUTOMATIC) { |
---|
870 | bool stopSubdivision = false; |
---|
871 | float ratio; |
---|
872 | // which algorithm to use for automatic termination criteria |
---|
873 | switch (algorithmAutoTermination) { |
---|
874 | // -------------------------------------------------- |
---|
875 | case 0: { // This is described in my thesis. |
---|
876 | // This is the cost of unsubdivided node |
---|
877 | float leafCost = (float)(d->count); |
---|
878 | ratio = d->bestCost / leafCost; |
---|
879 | const float minRatio = 0.75; |
---|
880 | if (ratio > minRatio) |
---|
881 | stopSubdivision = true; |
---|
882 | break; |
---|
883 | } |
---|
884 | // -------------------------------------------------- |
---|
885 | case 1: { |
---|
886 | // This is the cost of unsubdivided node, with uniformly distributed |
---|
887 | // objects and the spatial median subdivision, without any object straddling |
---|
888 | // the splitting plane. In some sense ideal setting for uniform distribution. |
---|
889 | Vector3 s = d->box.Diagonal(); |
---|
890 | // Taking the widest side of the object to be subdivided |
---|
891 | int diagAxis = s.DrivingAxis(); |
---|
892 | // Half of the width |
---|
893 | // ---------------------- |
---|
894 | float w2 = s[diagAxis] * 0.5; |
---|
895 | float t = s[oaxes[diagAxis][0]]; |
---|
896 | float h = s[oaxes[diagAxis][1]]; |
---|
897 | float sah2 = w2*(t+h) + h*t; // half of the surface area of a child |
---|
898 | // This is the cost for case when the node is subdivided in the half and |
---|
899 | // the half of the object is on the left and half of the objects on the right |
---|
900 | // This is not the best cost possible !!! |
---|
901 | // = float subdividedCost = 2 * (sah2/state.areaWholeSA2 * count/2); |
---|
902 | float subdividedCost = sah2/state.areaWholeSA2 * (float)(d->count); |
---|
903 | ratio = d->bestCost / subdividedCost; |
---|
904 | // This is the max ratio allowed for splitting node subdivision |
---|
905 | |
---|
906 | // The computed ratio = 1.0 corresponds to 'ideal case', so the threshold |
---|
907 | // must be higher than 1.0 ! |
---|
908 | const float minRatio = 1.1; |
---|
909 | if (ratio > minRatio) |
---|
910 | stopSubdivision = true; |
---|
911 | break; |
---|
912 | } |
---|
913 | default : { |
---|
914 | cerr << "Uknown algorithm for automatic termination criteria = " |
---|
915 | << algorithmAutoTermination << endl; |
---|
916 | } |
---|
917 | |
---|
918 | } // switch |
---|
919 | |
---|
920 | //cout << "R=" << ratio << " "; |
---|
921 | if (stopSubdivision) { |
---|
922 | // cout << "F" << pars.failedSubDivCount << " "; |
---|
923 | // when small improvement by the subdivision is reached |
---|
924 | d->pars.failedSubDivCount++; |
---|
925 | if (d->pars.failedSubDivCount > maxCountTrials) { |
---|
926 | // make leaf and finish with this KD-tree branch |
---|
927 | // cout << "L, cnt=" << count << " ,d=" << GetDepth()() |
---|
928 | // << " ,c=" << bestQ << " ,rat=" << ratio << "\n"; |
---|
929 | // possibly cut empty space before leaf is created |
---|
930 | if (cutEmptySpace) { |
---|
931 | // setting initial depth if empty space cutting |
---|
932 | startEmptyCutDepth = GetDepth(); |
---|
933 | IncDepth(); |
---|
934 | d->modeSubDiv = EE_SUBDIVCUTEMPTY; |
---|
935 | // create the leaf first, with late empty cutting |
---|
936 | SKTBNodeT *n = SubDiv(d); |
---|
937 | if (!nodeToReturn) |
---|
938 | nodeToReturn = n; |
---|
939 | DecDepth(); |
---|
940 | return nodeToReturn; // and finish by constructing a leaf |
---|
941 | } |
---|
942 | |
---|
943 | // make leaf and finish |
---|
944 | SKTBNodeT *n = MakeLeaf(d); |
---|
945 | if (!nodeToReturn) |
---|
946 | nodeToReturn = n; |
---|
947 | return nodeToReturn; |
---|
948 | } |
---|
949 | } |
---|
950 | |
---|
951 | // Do not stop subdivision now |
---|
952 | |
---|
953 | // update the progress in the parameters |
---|
954 | d->pars.ratioLastButOne = d->pars.ratioLast; |
---|
955 | d->pars.ratioLast = ratio; |
---|
956 | |
---|
957 | // assure that axis is defined, if finding the splitting plane has failed |
---|
958 | if ( (axis == CKTBAxes::EE_Leaf) || |
---|
959 | (d->position == MAXFLOAT) ) { |
---|
960 | // compute the spatial median for arbitrary axis |
---|
961 | axis = (CKTBAxes::Axes)int(RandomValue(0.f, 2.98f)); |
---|
962 | // set position based algorithm for the next cut and children |
---|
963 | d->position = (d->box.Min(axis) + d->box.Max(axis)) * 0.5f; |
---|
964 | d->twoSplits = 1; |
---|
965 | } |
---|
966 | } // subdivision automatic |
---|
967 | |
---|
968 | // ----------------------------------------------- |
---|
969 | // We have evaluated surface area heuristics above |
---|
970 | // if no winner for subdivision exists, make a leaf |
---|
971 | if ( (axis == CKTBAxes::EE_Leaf) || |
---|
972 | (d->position == MAXFLOAT) ) |
---|
973 | { |
---|
974 | //cerr << "This shall not happen" << endl; |
---|
975 | SKTBNodeT *n = MakeLeaf(d); |
---|
976 | if (!nodeToReturn) |
---|
977 | nodeToReturn = n; |
---|
978 | // cout << "LL\n"; |
---|
979 | return nodeToReturn; |
---|
980 | } |
---|
981 | |
---|
982 | if (verbose) { |
---|
983 | #ifdef _DEBUG |
---|
984 | |
---|
985 | //#define _KTBPRINTCUTS |
---|
986 | #ifndef _KTBPRINTCUTS |
---|
987 | if (GetDepth() == 0) |
---|
988 | #else |
---|
989 | if (_printCuts) |
---|
990 | #endif |
---|
991 | { |
---|
992 | cout << "position: " << d->position << ", axis: " |
---|
993 | << (int)axis << ", depth=" << GetDepth() << ", box:" << endl; |
---|
994 | Describe(d->box, cout, 0); |
---|
995 | cout << endl; |
---|
996 | } |
---|
997 | #endif // _DEBUG |
---|
998 | } |
---|
999 | |
---|
1000 | #ifdef _DEBUG |
---|
1001 | if (d->twoSplits == -1) { |
---|
1002 | cerr << "Some problem in implementation" << endl; |
---|
1003 | abort();; |
---|
1004 | } |
---|
1005 | #endif |
---|
1006 | |
---|
1007 | // copy the parameters to use for splitting |
---|
1008 | d->axis = axis; |
---|
1009 | |
---|
1010 | if (d->twoSplits == 1) { |
---|
1011 | // create interior node with two successors |
---|
1012 | // case (1) |
---|
1013 | // DEBUG << "case 1 \n"; |
---|
1014 | // Make easy cut, using one position |
---|
1015 | IncDepth(); |
---|
1016 | SKTBNodeT* n = MakeOneCut(d); |
---|
1017 | if (!nodeToReturn) |
---|
1018 | nodeToReturn = n; |
---|
1019 | DecDepth(); |
---|
1020 | return nodeToReturn; |
---|
1021 | } |
---|
1022 | |
---|
1023 | #if 1 |
---|
1024 | cerr << "Unexpected use in this implementation" << endl; |
---|
1025 | cerr << "ABORTING " << endl; |
---|
1026 | abort();; |
---|
1027 | #endif |
---|
1028 | |
---|
1029 | // case case |
---|
1030 | // (2) or (3) structure must be created |
---|
1031 | // O O # |
---|
1032 | // / \ / \ # |
---|
1033 | // L RI LI R # |
---|
1034 | // / \ / \ # |
---|
1035 | // RL RR LL LR # |
---|
1036 | |
---|
1037 | #ifdef _DEBUG |
---|
1038 | if ( (d->box.Min()[axis] >= d->position) || |
---|
1039 | (d->position >= d->position2) || |
---|
1040 | (d->position2 >= d->box.Max()[axis]) ) { |
---|
1041 | cerr << " the case 2 or 3 is defined incorrectly\n"; |
---|
1042 | abort();; |
---|
1043 | } |
---|
1044 | #endif |
---|
1045 | |
---|
1046 | // To be reworked !!! |
---|
1047 | abort();; |
---|
1048 | if (d->twoSplits == 2) { |
---|
1049 | // ----------------------------------------- |
---|
1050 | // case (2) |
---|
1051 | // DEBUG << "case 2 \n"; |
---|
1052 | |
---|
1053 | // make L part, linked in DFS order |
---|
1054 | int cntLeft = 1; |
---|
1055 | int cntRight = 0; |
---|
1056 | IncDepth(); |
---|
1057 | SKTBNodeT *n = MakeOneCut(d); |
---|
1058 | if (!nodeToReturn) |
---|
1059 | nodeToReturn = n; |
---|
1060 | |
---|
1061 | // make RI part, linked to the left node explictly |
---|
1062 | SBBox rbb = d->box; // the bounding box |
---|
1063 | rbb.Reduce(axis, 1, d->position); |
---|
1064 | d->box = rbb; |
---|
1065 | d->pars.reqAxis = axis; |
---|
1066 | d->pars.reqPosition = d->position2; |
---|
1067 | SKTBNodeT *n2 = SubDiv(d); |
---|
1068 | DecDepth(); |
---|
1069 | //Link the node |
---|
1070 | SetInteriorNodeLinks(n, 0, n2); |
---|
1071 | return nodeToReturn; |
---|
1072 | } |
---|
1073 | |
---|
1074 | // ----------------------------------------- |
---|
1075 | // case (3) ????????????????? I am not sure if this is correct implementation !!! VH |
---|
1076 | // DEBUG << "case 3 \n"; |
---|
1077 | |
---|
1078 | // make LI part |
---|
1079 | SBBox lbb = d->box; // the bounding box |
---|
1080 | lbb.Reduce(axis, 0, d->position2); |
---|
1081 | int cntLeft = 0; |
---|
1082 | int cntRight = 1; |
---|
1083 | d->box = lbb; |
---|
1084 | d->position = d->position2; |
---|
1085 | d->axis = axis; |
---|
1086 | // the next subdivision step for R part |
---|
1087 | d->pars.reqAxis = axis; |
---|
1088 | d->pars.reqPosition = d->position; |
---|
1089 | |
---|
1090 | IncDepth(); |
---|
1091 | SKTBNodeT *nl = SubDiv(d); |
---|
1092 | if (!nodeToReturn) |
---|
1093 | nodeToReturn = nl; |
---|
1094 | |
---|
1095 | // make R part |
---|
1096 | SKTBNodeT *n = MakeOneCut(d); |
---|
1097 | DecDepth(); |
---|
1098 | SetInteriorNodeLinks(nl, 0, n); |
---|
1099 | // return left node, linked in DFS order to the previous one |
---|
1100 | return nodeToReturn; |
---|
1101 | } |
---|
1102 | |
---|
1103 | |
---|
1104 | |
---|
1105 | // makes cutting on the given position on given axis |
---|
1106 | // at value position, bb is the bounding box of current node, |
---|
1107 | // count is the number of objects in the node |
---|
1108 | // flags LeftF and RightF declares if to continue down after the recursion |
---|
1109 | SKTBNodeT * |
---|
1110 | CKTBSBuildUp::MakeOneCut(SInputData *d) |
---|
1111 | { |
---|
1112 | #ifdef _DEBUG |
---|
1113 | assert(d->position >= d->box.Min(d->axis)); |
---|
1114 | assert(d->position <= d->box.Max(d->axis)); |
---|
1115 | #endif |
---|
1116 | |
---|
1117 | SKTBNodeT *nodeToReturn = 0; |
---|
1118 | |
---|
1119 | #ifdef _KTB_CONSTR_STATS |
---|
1120 | // surface area of the interior nodes to be subdivided |
---|
1121 | _sumSurfaceAreaInteriorNodes += d->box.SA2(); |
---|
1122 | #endif // _KTB_CONSTR_STATS |
---|
1123 | |
---|
1124 | //if (splitClip) { |
---|
1125 | // // empty object list to store the pointers to the split objects |
---|
1126 | // ObjectContainer objlist; |
---|
1127 | // cerr << "Not yet handled" << endl; |
---|
1128 | // abort();; |
---|
1129 | // } |
---|
1130 | |
---|
1131 | #if 0 |
---|
1132 | // check if the lists are correctly sorted .. in debug |
---|
1133 | static int index = 0; |
---|
1134 | #if 1 |
---|
1135 | cout << "MakeOneCut index = " << index << " .. check axis = " |
---|
1136 | << d->axis << " count = " << d->count |
---|
1137 | << " depth = " << GetDepth() << endl; |
---|
1138 | const int indexToFind = 3743; |
---|
1139 | if (index == indexToFind) { |
---|
1140 | |
---|
1141 | cout << "Debug should start " << endl; |
---|
1142 | cout << "index = indexToFind" << endl; |
---|
1143 | } |
---|
1144 | #endif |
---|
1145 | index++; |
---|
1146 | #endif |
---|
1147 | |
---|
1148 | // We have to allocate a new node for right child |
---|
1149 | SInputData* rightData = AllocNewData(); |
---|
1150 | // Copy the basic data from parent |
---|
1151 | rightData->CopyBasicData(d); |
---|
1152 | // We compute the tight box after the splitting |
---|
1153 | rightData->tightbox.Initialize(); |
---|
1154 | rightData->objlist = new ObjectContainer; |
---|
1155 | assert(rightData->objlist); |
---|
1156 | |
---|
1157 | // we compute the tight box for the objects on the |
---|
1158 | // left of the splitting plane |
---|
1159 | d->tightbox.Initialize(); |
---|
1160 | |
---|
1161 | // Simply go through the list of primitives and those that belong |
---|
1162 | // to the left keep on the left (d), those that belong to the right |
---|
1163 | // assign to the right list |
---|
1164 | int cntL = 0, cntR = 0; |
---|
1165 | { |
---|
1166 | ObjectContainer::iterator endit = d->objlist->end(); |
---|
1167 | int axis = d->axis; |
---|
1168 | AxisAlignedBox3 obox; |
---|
1169 | for (ObjectContainer::iterator it = d->objlist->begin(); it != endit; ) { |
---|
1170 | Intersectable *obj = *it; |
---|
1171 | obox = obj->GetBox(); |
---|
1172 | if (obox.Max(axis) > d->position) { |
---|
1173 | rightData->objlist->push_back(obj); |
---|
1174 | cntR++; |
---|
1175 | // the number of objects on the right of the splitting plane |
---|
1176 | rightData->tightbox.Include(obox); |
---|
1177 | } |
---|
1178 | if (obox.Min(axis) >= d->position) { |
---|
1179 | // we simply shorten the list - this object belongs only to the right list |
---|
1180 | (*it) = *(endit-1); |
---|
1181 | endit--; |
---|
1182 | continue; |
---|
1183 | } |
---|
1184 | else { |
---|
1185 | // the number of objects on the right of the splitting plane |
---|
1186 | cntL++; |
---|
1187 | d->tightbox.Include(obox); |
---|
1188 | } |
---|
1189 | it++; |
---|
1190 | } // for |
---|
1191 | } |
---|
1192 | |
---|
1193 | assert(cntL + cntR >= d->count); |
---|
1194 | |
---|
1195 | // --------------------------------------------- |
---|
1196 | // initialize the left bounding box 'bb' |
---|
1197 | d->box.Reduce(d->axis, 0, d->position); |
---|
1198 | d->objlist->resize(cntL); |
---|
1199 | |
---|
1200 | // initialize the right bounding box 'rbb' |
---|
1201 | rightData->box.Reduce(d->axis, 1, d->position); |
---|
1202 | |
---|
1203 | // Set correct number of objects |
---|
1204 | d->count = cntL; |
---|
1205 | rightData->count = cntR; |
---|
1206 | |
---|
1207 | d->tightbox = Intersect(d->tightbox, d->box); |
---|
1208 | rightData->tightbox = |
---|
1209 | Intersect(rightData->tightbox, rightData->box); |
---|
1210 | |
---|
1211 | // Allocate the representation of the interior node |
---|
1212 | SKTBNodeT *node = 0; |
---|
1213 | |
---|
1214 | if (makeMinBoxHere) { |
---|
1215 | // ------------------------------------------------------- |
---|
1216 | // interior node with the box |
---|
1217 | node = AllocInteriorNodeWithBox(// interior node data |
---|
1218 | d->axis, d->position, cntL, cntR, |
---|
1219 | // box data |
---|
1220 | d->box, d->lastMinBoxNode, |
---|
1221 | d->lastDepthForMinBoxes); |
---|
1222 | |
---|
1223 | //cout << "depth = " << GetDepth() << " node = " << node |
---|
1224 | // << " lastMinBoxNode = " << d->lastMinBoxNode << endl; |
---|
1225 | |
---|
1226 | d->lastMinBoxNode = node; // we have to remember the last node |
---|
1227 | rightData->lastMinBoxNode = node; |
---|
1228 | makeMinBoxHere = false; // reset for the next time |
---|
1229 | } |
---|
1230 | else |
---|
1231 | // normal interior node (=without box) |
---|
1232 | node = AllocInteriorNode(d->axis, d->position, cntL, cntR); |
---|
1233 | |
---|
1234 | #ifdef _DEBUG |
---|
1235 | if (!node) { |
---|
1236 | cerr << "Allocation of Interior Node failed.\n"; |
---|
1237 | abort();; |
---|
1238 | } |
---|
1239 | #endif // _DEBUG |
---|
1240 | // set the node to be returned |
---|
1241 | if (!nodeToReturn) |
---|
1242 | nodeToReturn = nodeToLink; |
---|
1243 | assert(node); |
---|
1244 | assert(nodeToReturn); |
---|
1245 | |
---|
1246 | // --------------------------------------------- |
---|
1247 | // initialize the left bounding box 'bb' |
---|
1248 | d->box.Reduce(d->axis, 0, d->position); |
---|
1249 | |
---|
1250 | // initialize the right bounding box 'rbb' |
---|
1251 | rightData->box.Reduce(d->axis, 1, d->position); |
---|
1252 | |
---|
1253 | // Set correct number of objects |
---|
1254 | d->count = cntL; |
---|
1255 | rightData->count = cntR; |
---|
1256 | |
---|
1257 | // The children nodes to be created |
---|
1258 | SKTBNodeT *nodeL=0, *nodeR=0; |
---|
1259 | |
---|
1260 | // Recurse to the left child |
---|
1261 | if (d->makeSubdivisionLeft) { |
---|
1262 | // subdivide branches |
---|
1263 | ESubdivMode modeLeft = d->modeSubDiv ; |
---|
1264 | if ( (cutEmptySpace) && |
---|
1265 | (modeLeft != EE_SUBDIVCUTEMPTY) && |
---|
1266 | (d->count <= maxListLength) ) { |
---|
1267 | // setting initial depth if empty space cutting |
---|
1268 | startEmptyCutDepth = GetDepth(); |
---|
1269 | d->modeSubDiv = EE_SUBDIVCUTEMPTY; |
---|
1270 | } |
---|
1271 | // DEBUG << "Left cnt= " << (cnt[0] >> 1) << " depth= " |
---|
1272 | // << GetDepth() << "\n"; |
---|
1273 | nodeL = SubDiv(d); |
---|
1274 | } |
---|
1275 | |
---|
1276 | // Recurse to the right child |
---|
1277 | if (rightData->makeSubdivisionRight) { |
---|
1278 | ESubdivMode modeRight = d->modeSubDiv ; |
---|
1279 | if ( (cutEmptySpace) && |
---|
1280 | (modeRight != EE_SUBDIVCUTEMPTY) && |
---|
1281 | (rightData->count <= maxListLength) ) { |
---|
1282 | // setting initial depth if empty space cutting |
---|
1283 | startEmptyCutDepth = GetDepth(); |
---|
1284 | rightData->modeSubDiv = EE_SUBDIVCUTEMPTY; |
---|
1285 | } |
---|
1286 | // DEBUG << "Right cnt= " << (cnt[1] >> 1) << " depth= " |
---|
1287 | // << GetDepth() << "\n"; |
---|
1288 | nodeR = SubDiv(rightData); |
---|
1289 | |
---|
1290 | } |
---|
1291 | |
---|
1292 | delete rightData->objlist; |
---|
1293 | rightData->objlist = 0; |
---|
1294 | delete d->objlist; |
---|
1295 | d->objlist = 0; |
---|
1296 | |
---|
1297 | // Free the data to be reused further on |
---|
1298 | FreeLastData(); |
---|
1299 | |
---|
1300 | // Set the node pointers to the children for created node |
---|
1301 | SetInteriorNodeLinks(node, nodeL, nodeR); |
---|
1302 | |
---|
1303 | return nodeToReturn; // return the node |
---|
1304 | } |
---|
1305 | |
---|
1306 | // returns a box enclosing all the objects in the node |
---|
1307 | void |
---|
1308 | CKTBSBuildUp::GetTightBox(const SInputData &d, SBBox &tbox) |
---|
1309 | { |
---|
1310 | tbox = d.tightbox; |
---|
1311 | } |
---|
1312 | |
---|
1313 | // make the full leaf from current node |
---|
1314 | SKTBNodeT* |
---|
1315 | CKTBSBuildUp::MakeLeaf(SInputData *d) |
---|
1316 | { |
---|
1317 | #ifdef _KTBPRINTCUTS |
---|
1318 | cout << " Creating leaf, depth = " << GetDepth() |
---|
1319 | << " cntObjs = " << count << endl; |
---|
1320 | //exit(3); |
---|
1321 | #endif |
---|
1322 | |
---|
1323 | #ifdef _KTB_CONSTR_STATS |
---|
1324 | // update the statistics |
---|
1325 | float sa2bb; |
---|
1326 | _sumSurfaceAreaLeaves += (sa2bb = d->box.SA2()); |
---|
1327 | _sumSurfaceAreaMULcntLeaves += ((float)d->count * sa2bb); |
---|
1328 | #endif // _KTB_CONSTR_STATS |
---|
1329 | |
---|
1330 | // ------------------------------------------------------ |
---|
1331 | // The version with allocating empty leaves. This makes |
---|
1332 | // sense since in DFS order it works correctly |
---|
1333 | SKTBNodeT *node = AllocLeaf(d->count); |
---|
1334 | |
---|
1335 | if (d->count > 0) { |
---|
1336 | // copy the list of objects |
---|
1337 | ObjectContainer *objlist = d->objlist; |
---|
1338 | assert(objlist->size() == d->count); |
---|
1339 | // Do not delete the object list, it is used as allocated above |
---|
1340 | SetFullLeaf(node, objlist); |
---|
1341 | // We delete the object list since this is used at the level of leaves |
---|
1342 | d->objlist = 0; |
---|
1343 | } |
---|
1344 | |
---|
1345 | // We assume that the allocation of the single leaf cannot |
---|
1346 | // fail, since this one shall be the last in the sequence. |
---|
1347 | assert(node == nodeToLink); |
---|
1348 | |
---|
1349 | // Return the node to be used in the linking |
---|
1350 | return nodeToLink; |
---|
1351 | } |
---|
1352 | |
---|
1353 | // -------------------------------------------------------------------- |
---|
1354 | // class CKTBSBuildUp::CTestAx |
---|
1355 | |
---|
1356 | // The initialization for the first axis to be tested, this time *****Z axis******* |
---|
1357 | void |
---|
1358 | CKTBSBuildUp::SSplitState::InitXaxis(int cnt, const SBBox &boxN) |
---|
1359 | { |
---|
1360 | // initialize the variables, mainly for surface area estimates |
---|
1361 | cntAll = cnt; // the number of all objects in the bounding box |
---|
1362 | cntLeft = 0; // the count of bounding boxes on the left |
---|
1363 | cntRight = cnt; // the count of bounding boxes on the right |
---|
1364 | thickness = 0; // the count of bounding boxes straddling the splitting plane |
---|
1365 | axis = CKTBAxes::EE_X_axis; // the axis, where the splitting is proposed |
---|
1366 | box = boxN; // the box, that is subdivided |
---|
1367 | //int topAxis = 1; // axis that is considered in top .. depth |
---|
1368 | //int frontAxis = 2; // axis that is considered in front .. height |
---|
1369 | // the size of bounding box along the axis to be split |
---|
1370 | width = box.Max().x - box.Min().x; |
---|
1371 | // and along two other axes |
---|
1372 | topw = box.Max().y - box.Min().y; |
---|
1373 | frontw = box.Max().z - box.Min().z; |
---|
1374 | // surface area of the splitting plane .. one face !! |
---|
1375 | areaSplitPlane = topw * frontw; |
---|
1376 | // the sum of length of the boxes not to be subdivided |
---|
1377 | areaSumLength = topw + frontw; |
---|
1378 | // The surface are of the whole box |
---|
1379 | areaWholeSA2 = width * areaSumLength + areaSplitPlane; |
---|
1380 | // initial evaluation .. the worst cost |
---|
1381 | bestCost = WorstEvaluation(); |
---|
1382 | // the buckets are zeroed |
---|
1383 | InitBuckets(); |
---|
1384 | } |
---|
1385 | |
---|
1386 | // The initialization for the first axis to be tested, this time *****Y axis******* |
---|
1387 | // This can be run independently. |
---|
1388 | void |
---|
1389 | CKTBSBuildUp::SSplitState::InitYaxis(int cnt, const SBBox &boxN) |
---|
1390 | { |
---|
1391 | // initialize the variables, mainly for surface area estimates |
---|
1392 | cntAll = cnt; // the number of all objects in the bounding box |
---|
1393 | cntLeft = 0; // the count of bounding boxes on the left |
---|
1394 | cntRight = cnt; // the count of bounding boxes on the right |
---|
1395 | thickness = 0; // the count of bounding boxes straddling the splitting plane |
---|
1396 | axis = CKTBAxes::EE_Y_axis; // the axis, where the splitting is proposed |
---|
1397 | box = boxN; // the box, that is subdivided |
---|
1398 | //int frontAxis = oaxes[axis][0]; // axis that is considered in front .. height - x-axis |
---|
1399 | //int topAxis = oaxes[axis][1]; // axis that is considered in top .. depth - y-axis |
---|
1400 | // the size along the second axis - height |
---|
1401 | frontw = box.Max().x - box.Min().x; |
---|
1402 | // the size of bounding box along the axis to be split - width |
---|
1403 | width = box.Max().y - box.Min().y; |
---|
1404 | // The size along third axis - depth |
---|
1405 | topw = box.Max().z - box.Min().z; |
---|
1406 | // surface area of the splitting plane .. one face !! |
---|
1407 | areaSplitPlane = topw * frontw; |
---|
1408 | // the sum of length of the boxes not to be subdivided |
---|
1409 | areaSumLength = topw + frontw; |
---|
1410 | areaWholeSA2 = width * areaSumLength + areaSplitPlane; |
---|
1411 | // initial evaluation .. the worst cost |
---|
1412 | bestCost = WorstEvaluation(); |
---|
1413 | // the buckets are zeroed |
---|
1414 | InitBuckets(); |
---|
1415 | } |
---|
1416 | |
---|
1417 | // The initialization for the first axis to be tested, this time *****Z axis******* |
---|
1418 | // This can be run independently. |
---|
1419 | void |
---|
1420 | CKTBSBuildUp::SSplitState::InitZaxis(int cnt, const SBBox &boxN) |
---|
1421 | { |
---|
1422 | // initialize the variables, mainly for surface area estimates |
---|
1423 | cntAll = cnt; // the number of all objects in the bounding box |
---|
1424 | cntLeft = 0; // the count of bounding boxes on the left |
---|
1425 | cntRight = cnt; // the count of bounding boxes on the right |
---|
1426 | thickness = 0; // the count of bounding boxes straddling the splitting plane |
---|
1427 | axis = CKTBAxes::EE_Z_axis; // the axis, where the splitting is proposed |
---|
1428 | box = boxN; // the box, that is subdivided |
---|
1429 | //int frontAxis = oaxes[axis][1]; // axis that is considered in front .. height - x axis |
---|
1430 | //int topAxis = oaxes[axis][0]; // axis that is considered in top .. depth - y axis |
---|
1431 | // The size along third axis - depth |
---|
1432 | topw = box.Max().x - box.Min().x; |
---|
1433 | // the size along the second axis - height |
---|
1434 | frontw = box.Max().y - box.Min().y; |
---|
1435 | // the size of bounding box along the axis to be split - width |
---|
1436 | width = box.Max().z - box.Min().z; |
---|
1437 | // surface area of the splitting plane .. one face !! |
---|
1438 | areaSplitPlane = topw * frontw; |
---|
1439 | // the sum of length of the boxes not to be subdivided |
---|
1440 | areaSumLength = topw + frontw; |
---|
1441 | areaWholeSA2 = width * areaSumLength + areaSplitPlane; |
---|
1442 | // initial evaluation .. the worst cost |
---|
1443 | bestCost = WorstEvaluation(); |
---|
1444 | // the buckets are zeroed |
---|
1445 | InitBuckets(); |
---|
1446 | } |
---|
1447 | |
---|
1448 | // This is the computation of the cost using surface area heuristics |
---|
1449 | // using LINEAR ESTIMATE |
---|
1450 | void |
---|
1451 | CKTBSBuildUp::EvaluateCost(SSplitState &state) |
---|
1452 | { |
---|
1453 | // the surface area of the bounding box for the left child |
---|
1454 | assert(state.position > -1e-9); |
---|
1455 | float areaLeft = state.position * state.areaSumLength + state.areaSplitPlane; |
---|
1456 | |
---|
1457 | // the surface area of the right bounding box for the right child |
---|
1458 | assert(state.width - state.position > -1e-9); |
---|
1459 | float areaRight = (state.width - state.position) * state.areaSumLength + |
---|
1460 | state.areaSplitPlane; |
---|
1461 | |
---|
1462 | // computation of the cost .. smaller is better |
---|
1463 | float cost = areaLeft * state.cntLeft + areaRight * state.cntRight; |
---|
1464 | |
---|
1465 | #ifdef _DEBUG_COSTFUNCTION |
---|
1466 | // Put there normalized position and cost also normalized somehow |
---|
1467 | ReportCostStream(state.position / state.width, |
---|
1468 | cost / (state.areaWholeSA2 * state.cntAll)); |
---|
1469 | #endif |
---|
1470 | |
---|
1471 | // This normalization is not in fact necessary here |
---|
1472 | //cost //= state.areaWholeSA2; |
---|
1473 | if (cost < state.bestCost) { |
---|
1474 | state.bestCost = cost; |
---|
1475 | // The iterator pointing to the best boundary |
---|
1476 | state.bestPosition = state.position; |
---|
1477 | // The number of objects whose boundaries are duplicated |
---|
1478 | //state.bestThickness = state.thickness; |
---|
1479 | } |
---|
1480 | return; |
---|
1481 | } |
---|
1482 | |
---|
1483 | // This is the computation of the cost using surface area heuristics |
---|
1484 | void |
---|
1485 | CKTBSBuildUp::EvaluateCostFreeCut(SSplitState &state) |
---|
1486 | { |
---|
1487 | // This is assumed to work for free cut (no object intersected) |
---|
1488 | assert(state.thickness == 0); |
---|
1489 | assert((state.cntLeft == 0) || (state.cntRight == 0)); |
---|
1490 | |
---|
1491 | // the surface area of the bounding box for the left child |
---|
1492 | assert(state.position > -1e-9); |
---|
1493 | float areaLeft = state.position * state.areaSumLength + state.areaSplitPlane; |
---|
1494 | |
---|
1495 | // the surface area of the right bounding box for the right child |
---|
1496 | assert(state.width - state.position > -1e-9); |
---|
1497 | float areaRight = (state.width - state.position) * state.areaSumLength + |
---|
1498 | state.areaSplitPlane; |
---|
1499 | |
---|
1500 | // computation of the cost .. smaller is better |
---|
1501 | float cost = biasFreeCuts *(areaLeft * state.cntLeft + areaRight * state.cntRight); |
---|
1502 | |
---|
1503 | #ifdef _DEBUG_COSTFUNCTION |
---|
1504 | // Put there normalized position and cost also normalized somehow |
---|
1505 | ReportCostStream(state.position / state.width, |
---|
1506 | cost / (state.areaWholeSA2 * state.cntAll)); |
---|
1507 | #endif |
---|
1508 | |
---|
1509 | // This normalization is not in fact necessary here |
---|
1510 | //cost //= state.areaWholeSA2; |
---|
1511 | if (cost < state.bestCost) { |
---|
1512 | state.bestCost = cost; |
---|
1513 | // The iterator pointing to the best boundary |
---|
1514 | state.bestPosition = state.position; |
---|
1515 | // The number of objects whose boundaries are duplicated |
---|
1516 | // state.bestThickness = 0; |
---|
1517 | } |
---|
1518 | return; |
---|
1519 | } |
---|
1520 | |
---|
1521 | // --------------------------------------------------------------------- |
---|
1522 | // For a given X-axis search for the best splitting plane position. |
---|
1523 | // This is implemented by sampling |
---|
1524 | void |
---|
1525 | CKTBSBuildUp::GetSplitPlaneOpt(SInputData *d, int axisToTest) |
---|
1526 | { |
---|
1527 | // some necessary space is required to cut off |
---|
1528 | const float MinPosition = d->tightbox.Min(axisToTest); |
---|
1529 | const float MaxPosition = d->tightbox.Max(axisToTest); |
---|
1530 | const float bMinPosition = d->box.Min(axisToTest); |
---|
1531 | state.bestTwoSplits = 1; |
---|
1532 | |
---|
1533 | // This makes no sense to compute, if the width of the box |
---|
1534 | // is zero! |
---|
1535 | if (bMinPosition == d->box.Max(axisToTest)) { |
---|
1536 | state.bestPosition = bMinPosition; |
---|
1537 | return; |
---|
1538 | } |
---|
1539 | |
---|
1540 | int bucketN = state.bucketN; |
---|
1541 | |
---|
1542 | for (int i = 0; i < bucketN; i++) { |
---|
1543 | state.bucketMin[i] = 0; |
---|
1544 | state.bucketMax[i] = 0; |
---|
1545 | } |
---|
1546 | |
---|
1547 | float mult = (float)bucketN / (MaxPosition - MinPosition); |
---|
1548 | |
---|
1549 | ObjectContainer::iterator endit = d->objlist->end(); |
---|
1550 | int axis = axisToTest; |
---|
1551 | AxisAlignedBox3 obox; |
---|
1552 | for (ObjectContainer::iterator it = d->objlist->begin(); it != endit; it++) { |
---|
1553 | Intersectable *obj = *it; |
---|
1554 | obox = obj->GetBox(); |
---|
1555 | float v = obox.Min(axis); |
---|
1556 | if (v < MinPosition) |
---|
1557 | state.bucketMin[0]++; |
---|
1558 | else |
---|
1559 | if (v > MaxPosition) |
---|
1560 | state.bucketMin[bucketN-1]++; |
---|
1561 | else { |
---|
1562 | assert(v >= MinPosition); |
---|
1563 | assert(v <= MaxPosition); |
---|
1564 | int bucketPos = (int)((v - MinPosition) * mult); |
---|
1565 | assert(bucketPos >= 0); |
---|
1566 | assert(bucketPos <= bucketN); |
---|
1567 | if (bucketPos == bucketN) |
---|
1568 | bucketPos = bucketN-1; |
---|
1569 | state.bucketMin[bucketPos]++; |
---|
1570 | } |
---|
1571 | |
---|
1572 | float v2 = obox.Max(axis); |
---|
1573 | if (v2 < MinPosition) |
---|
1574 | state.bucketMax[0]++; |
---|
1575 | else |
---|
1576 | if (v2 > MaxPosition) |
---|
1577 | state.bucketMax[bucketN-1]++; |
---|
1578 | else { |
---|
1579 | assert(v2 >= MinPosition); |
---|
1580 | assert(v2 <= MaxPosition); |
---|
1581 | int bucketPos2 = (int)((v2 - MinPosition) * mult); |
---|
1582 | assert(bucketPos2 >= 0); |
---|
1583 | assert(bucketPos2 <= bucketN); |
---|
1584 | if (bucketPos2 == bucketN) |
---|
1585 | bucketPos2 = bucketN-1; |
---|
1586 | state.bucketMax[bucketPos2]++; |
---|
1587 | } |
---|
1588 | } // for |
---|
1589 | |
---|
1590 | // First evaluate the cost, when the splitting plane is on the left |
---|
1591 | state.position = MinPosition - bMinPosition; |
---|
1592 | assert(state.position >= 0.f); |
---|
1593 | assert(state.position <= state.width); |
---|
1594 | state.cntLeft = 0; |
---|
1595 | state.cntRight = d->count; |
---|
1596 | state.thickness = 0; |
---|
1597 | EvaluateCostFreeCut(state); |
---|
1598 | int cntLeft = 0; |
---|
1599 | int cntRight = d->count; |
---|
1600 | int thickness = 0; |
---|
1601 | // -------------------------------------- |
---|
1602 | float imult = (MaxPosition - MinPosition) / (float)bucketN; |
---|
1603 | for (int i = 0; i < bucketN-1; i++) { |
---|
1604 | cntLeft += state.bucketMin[i]; |
---|
1605 | cntRight -= state.bucketMax[i]; |
---|
1606 | thickness = cntLeft + cntRight - d->count; |
---|
1607 | assert(thickness >= 0); |
---|
1608 | state.cntLeft = cntLeft; |
---|
1609 | state.cntRight = cntRight; |
---|
1610 | state.thickness = thickness; |
---|
1611 | state.position = MinPosition - bMinPosition + imult * ((float)(i+1)); |
---|
1612 | assert(state.position >= 0); |
---|
1613 | assert(state.position <= state.width); |
---|
1614 | |
---|
1615 | EvaluateCost(state); |
---|
1616 | } // for |
---|
1617 | |
---|
1618 | // free cut |
---|
1619 | state.position = MaxPosition - bMinPosition; |
---|
1620 | assert(state.position >= 0); |
---|
1621 | assert(state.position <= state.width); |
---|
1622 | state.cntLeft = d->count; |
---|
1623 | state.cntRight = 0; |
---|
1624 | state.thickness = 0; |
---|
1625 | EvaluateCostFreeCut(state); |
---|
1626 | |
---|
1627 | // we move back to the space |
---|
1628 | state.bestPosition += bMinPosition; |
---|
1629 | |
---|
1630 | // In this case the best result is kept in 'state' variable |
---|
1631 | return; |
---|
1632 | } // CTestAx::GetSplitPlaneOpt (for X, Y, Z) |
---|
1633 | |
---|
1634 | } |
---|
1635 | |
---|