1 |
|
---|
2 | // Written by Vlastimil Havran, December 2007
|
---|
3 |
|
---|
4 | // This macro allows to use the ray shooting written by
|
---|
5 | // Vlastimil Havran, 2007-2008
|
---|
6 |
|
---|
7 |
|
---|
8 | #include "VssRay.h"
|
---|
9 | #include "KdTree.h"
|
---|
10 | #include "Preprocessor.h"
|
---|
11 | #include "IntersectableWrapper.h"
|
---|
12 |
|
---|
13 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
14 | #include "ktbconf.h"
|
---|
15 | #include "timer.h"
|
---|
16 | #include "ktball.h"
|
---|
17 | #include "ktb.h"
|
---|
18 | #ifdef _USE_HAVRAN_SSE
|
---|
19 |
|
---|
20 | #include "raypack.h"
|
---|
21 | #endif // _USE_HAVRAN_SSE
|
---|
22 | #endif // USE_HAVRAN_RAYCASTER
|
---|
23 | #include "HavranRayCaster.h"
|
---|
24 |
|
---|
25 | #define DEBUG_RAYCAST 0
|
---|
26 |
|
---|
27 | namespace GtpVisibilityPreprocessor {
|
---|
28 |
|
---|
29 | #ifdef _USE_HAVRAN_SSE
|
---|
30 | // static rays
|
---|
31 | GALIGN16 RayPacket2x2
|
---|
32 | HavranDynRayCaster::raypack_t;
|
---|
33 | #endif // _USE_HAVRAN_SSE
|
---|
34 |
|
---|
35 | HavranDynRayCaster::HavranDynRayCaster(const Preprocessor &preprocessor):
|
---|
36 | HavranRayCaster(preprocessor), mDynKtbtree(0),
|
---|
37 | dynobjects(0), dynamicFlag(false)
|
---|
38 | {
|
---|
39 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
40 | #ifdef _USE_HAVRAN_SSE
|
---|
41 | if ( ((int)(&raypack_t)) & 0xf != 0) {
|
---|
42 | cerr << "The ray pack data type not aligned on 16 Bytes boundary" << endl;
|
---|
43 | cerr << "Address is " << (void*)(&raypack_t) << endl;
|
---|
44 | cerr << "Reorganize your data structure to make it aligned" << endl;
|
---|
45 | abort();
|
---|
46 | }
|
---|
47 | #endif
|
---|
48 | #endif
|
---|
49 | }
|
---|
50 |
|
---|
51 |
|
---|
52 | HavranDynRayCaster::~HavranDynRayCaster()
|
---|
53 | {
|
---|
54 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
55 | DeleteDynamicObjects();
|
---|
56 | #endif // USE_HAVRAN_RAYCASTER
|
---|
57 | }
|
---|
58 |
|
---|
59 | void
|
---|
60 | HavranDynRayCaster::DeleteDynamicObjects()
|
---|
61 | {
|
---|
62 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
63 | delete mDynKtbtree; mDynKtbtree = 0;
|
---|
64 | delete dynobjects; dynobjects = 0;
|
---|
65 | dynamicFlag = false;
|
---|
66 | #endif // USE_HAVRAN_RAYCASTER
|
---|
67 | }
|
---|
68 |
|
---|
69 | void
|
---|
70 | HavranDynRayCaster::AddDynamicObjecs(const ObjectContainer &objects, const Matrix4x4 &m)
|
---|
71 | {
|
---|
72 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
73 | if (dynobjects)
|
---|
74 | DeleteDynamicObjects();
|
---|
75 |
|
---|
76 | dynamicFlag = false;
|
---|
77 |
|
---|
78 | if (objects.size()) {
|
---|
79 | dynamicFlag = true;
|
---|
80 | dynobjects = new ObjectContainer(objects);
|
---|
81 | assert(dynobjects);
|
---|
82 | // kd-tree built over the objects
|
---|
83 | mDynKtbtree = new CKTB;
|
---|
84 | mDynKtbtree->BuildUp(*dynobjects);
|
---|
85 |
|
---|
86 | // The matrix to transform the objects
|
---|
87 | UpdateDynamicObjects(m);
|
---|
88 | }
|
---|
89 | #endif // USE_HAVRAN_RAYCASTER
|
---|
90 | }
|
---|
91 |
|
---|
92 | void
|
---|
93 | HavranDynRayCaster::UpdateDynamicObjects(const Matrix4x4 &m)
|
---|
94 | {
|
---|
95 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
96 | matTr = m;
|
---|
97 | matTr_inv = Invert(matTr);
|
---|
98 | #endif // USE_HAVRAN_RAYCASTER
|
---|
99 | }
|
---|
100 |
|
---|
101 | // Using packet of 4 rays supposing that these are coherent
|
---|
102 | // We give a box to which each ray is clipped to before the
|
---|
103 | // ray shooting is computed !
|
---|
104 | void HavranDynRayCaster::CastRaysPacket4(const Vector3 &minBox,
|
---|
105 | const Vector3 &maxBox,
|
---|
106 | const Vector3 origin4[],
|
---|
107 | const Vector3 direction4[],
|
---|
108 | int result4[],
|
---|
109 | float dist4[])
|
---|
110 | {
|
---|
111 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
112 | #ifdef _USE_HAVRAN_SSE
|
---|
113 | for (int i = 0; i < 4; i++) {
|
---|
114 | result4[i] = -1;
|
---|
115 | result4_t[4] = -1;
|
---|
116 | raypack.SetLoc(i, origin4[i]);
|
---|
117 | raypack.SetDir(i, direction4[i]);
|
---|
118 | }
|
---|
119 |
|
---|
120 | // The same operations for packets of rays, if implemented by
|
---|
121 | // a particular ASDS, otherwise it is emulated by decomposition
|
---|
122 | // of a packet to individual rays and traced individually.
|
---|
123 | mKtbtree->FindNearestI(raypack, minBox, maxBox);
|
---|
124 |
|
---|
125 | for (int i = 0; i < 4; i++) {
|
---|
126 | // if ray intersects an object, we set the pointer to
|
---|
127 | // this object
|
---|
128 | Intersectable* intersectable = raypack.GetObject(i);
|
---|
129 | if (intersectable) {
|
---|
130 | result4[i] = intersectable->mId;
|
---|
131 | dist4[i] = raypack.GetT(i);
|
---|
132 | }
|
---|
133 | }
|
---|
134 |
|
---|
135 | if (dynamicFlag) {
|
---|
136 | // The ray pack to be transformed
|
---|
137 | raypack_t = raypack;
|
---|
138 | raypack_t.ApplyTransform(matTr_inv);
|
---|
139 |
|
---|
140 | mDynKtbtree->FindNearestI(raypack_t, minBox, maxBox);
|
---|
141 |
|
---|
142 | for (int i = 0; i < 4; i++) {
|
---|
143 | // if ray intersects an object, we set the pointer to
|
---|
144 | // this object
|
---|
145 | Intersectable* intersectable = raypack_t.GetObject(i);
|
---|
146 | if (intersectable) {
|
---|
147 | float t = raypack_t.GetT(i);
|
---|
148 | if ( ( (result4[i] != -1) && (t < dist4[i]) ) ||
|
---|
149 | (result4[i] == -1) ) {
|
---|
150 | // update, dynamic object is closer
|
---|
151 | dist4[i] = t;
|
---|
152 | result4[i] = intersectable->mId;
|
---|
153 | }
|
---|
154 | }
|
---|
155 | } // for i
|
---|
156 | } // dynamic flag
|
---|
157 |
|
---|
158 | return;
|
---|
159 | #else // _USE_HAVRAN_SSE
|
---|
160 | // Compute the result ray by ray
|
---|
161 | SimpleRay sray;
|
---|
162 | for (int i = 0; i < 4; i++) {
|
---|
163 | result4[i] = -1; // no intersection
|
---|
164 | sray.mOrigin = origin4[i];
|
---|
165 | sray.mDirection = direction4[i];
|
---|
166 | mKtbtree->FindNearestI(sray, minBox, maxBox);
|
---|
167 | if (SimpleRay::IntersectionRes[0].intersectable) {
|
---|
168 | // This is object ID - is this the triangle index ???
|
---|
169 | result4[i] = SimpleRay::IntersectionRes[0].intersectable->mId;
|
---|
170 | dist4[i] = SimpleRay::IntersectionRes[0].tdist;
|
---|
171 | }
|
---|
172 |
|
---|
173 | if (dynamicFlag) {
|
---|
174 | // Now apply transform
|
---|
175 | ApplyTransform(sray);
|
---|
176 | mDynKtbtree->FindNearestI(sray, minBox, maxBox);
|
---|
177 | Intersectable *objdyn = SimpleRay::IntersectionRes[0].intersectable;
|
---|
178 | if (objdyn) {
|
---|
179 | float t = SimpleRay::IntersectionRes[0].tdist;
|
---|
180 | if ( ((result4[i] != -1) && (t < dist4[i])) ||
|
---|
181 | (result4[i] == -1) ) {
|
---|
182 | // = update, dynamic object is closer
|
---|
183 | dist4[i] = t;
|
---|
184 | result4[i] = objdyn->mId;
|
---|
185 | }
|
---|
186 | }
|
---|
187 | } // dynamic flag
|
---|
188 | } // for i;
|
---|
189 |
|
---|
190 | #endif // _USE_HAVRAN_SSE
|
---|
191 | #endif // USE_HAVRAN_RAYCASTER
|
---|
192 | }
|
---|
193 |
|
---|
194 |
|
---|
195 | int HavranDynRayCaster::CastRay(const SimpleRay &simpleRay,
|
---|
196 | VssRayContainer &vssRays,
|
---|
197 | const AxisAlignedBox3 &box,
|
---|
198 | const bool castDoubleRay,
|
---|
199 | const bool pruneInvalidRays)
|
---|
200 | {
|
---|
201 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
202 |
|
---|
203 | // rays forwards - TESTED 21/1/2008 VH (only for forward rays)
|
---|
204 |
|
---|
205 | int hits = 0;
|
---|
206 | Intersection hitA(simpleRay.mOrigin), hitB(simpleRay.mOrigin);
|
---|
207 |
|
---|
208 | // ray.mFlags &= ~Ray::CULL_BACKFACES;
|
---|
209 | bool result;
|
---|
210 | hitA.mObject = 0;
|
---|
211 | if ((result = mKtbtree->FindNearestI(simpleRay))) {
|
---|
212 | hitA.mObject = SimpleRay::IntersectionRes[0].intersectable;
|
---|
213 | tdist[0] = SimpleRay::IntersectionRes[0].tdist;
|
---|
214 | hitA.mPoint = simpleRay.Extrap(tdist[0]);
|
---|
215 | hitA.mNormal = SimpleRay::IntersectionRes[0].intersectable->GetNormal(0);
|
---|
216 | //hitA.mNormal = (dynamic_cast< TriangleIntersectable *>
|
---|
217 | // (hitA.mObject))->GetNormal(0);
|
---|
218 | }
|
---|
219 |
|
---|
220 | if (dynamicFlag) {
|
---|
221 | bool result2 = false;
|
---|
222 | // For a dynamic object
|
---|
223 | sray_t = simpleRay;
|
---|
224 | // Now apply transform
|
---|
225 | ApplyTransform(sray_t);
|
---|
226 | if (mDynKtbtree->FindNearestI(sray_t)) {
|
---|
227 | float t = SimpleRay::IntersectionRes[0].tdist;
|
---|
228 | if ( ((hitA.mObject) && (t < tdist[0])) ||
|
---|
229 | (!hitA.mObject) ) {
|
---|
230 | result = result2 = true; // dynamic object hit
|
---|
231 | // There is an intersection closer than the previous one
|
---|
232 | hitA.mObject = SimpleRay::IntersectionRes[0].intersectable;
|
---|
233 | tdist[0] = SimpleRay::IntersectionRes[0].tdist;
|
---|
234 | hitA.mPoint = simpleRay.Extrap(tdist[0]);
|
---|
235 | hitA.mNormal = hitA.mObject->GetNormal(0);
|
---|
236 | // We have to transform the normal
|
---|
237 | hitA.mNormal = TransformNormal(matTr_inv, hitA.mNormal);
|
---|
238 | }
|
---|
239 | }
|
---|
240 | if (result && (!result2)) {
|
---|
241 | // We put back the result from hitA.mObject for ray tracing
|
---|
242 | SimpleRay::IntersectionRes[0].intersectable = hitA.mObject;
|
---|
243 | SimpleRay::IntersectionRes[0].tdist = tdist[0];
|
---|
244 | }
|
---|
245 | }
|
---|
246 |
|
---|
247 | bool resultB = false;
|
---|
248 | if (castDoubleRay) {
|
---|
249 | bool result2 = false;
|
---|
250 | // casting rays backward
|
---|
251 | Vector3 *v = (Vector3*)(&simpleRay.mDirection);
|
---|
252 | *v = -(*v);
|
---|
253 | // ray.mFlags &= ~Ray::CULL_BACKFACES;
|
---|
254 | if (mKtbtree->FindNearestI(simpleRay)) {
|
---|
255 | resultB = true;
|
---|
256 | hitB.mObject = SimpleRay::IntersectionRes[0].intersectable;
|
---|
257 | assert(hitB.mObject);
|
---|
258 | float tdist = SimpleRay::IntersectionRes[0].tdist;
|
---|
259 | hitB.mPoint = simpleRay.Extrap(tdist);
|
---|
260 | hitB.mNormal = hitB.mObject->GetNormal(0);
|
---|
261 | }
|
---|
262 |
|
---|
263 | if (dynamicFlag) {
|
---|
264 | // For a dynamic object
|
---|
265 | sray_t = simpleRay;
|
---|
266 | // Now apply transform
|
---|
267 | ApplyTransform(sray_t);
|
---|
268 | bool result2;
|
---|
269 | if ( mDynKtbtree->FindNearestI(sray_t)) {
|
---|
270 | assert(SimpleRay::IntersectionRes[0].intersectable);
|
---|
271 | float t = SimpleRay::IntersectionRes[0].tdist;
|
---|
272 | if ( ((hitB.mObject) && (t < tdist[0])) ||
|
---|
273 | (!hitB.mObject) ) {
|
---|
274 | // There is an intersection closer than the previous one
|
---|
275 | resultB = result2 = true; // dynamic object hit
|
---|
276 | hitB.mObject = SimpleRay::IntersectionRes[0].intersectable;
|
---|
277 | tdist[0] = SimpleRay::IntersectionRes[0].tdist;
|
---|
278 | hitB.mPoint = simpleRay.Extrap(tdist[0]);
|
---|
279 | hitB.mNormal = hitB.mObject->GetNormal(0);
|
---|
280 | // We have to transform the normal
|
---|
281 | hitB.mNormal = TransformNormal(matTr_inv, hitB.mNormal);
|
---|
282 | }
|
---|
283 | }
|
---|
284 | if ((resultB) && (!result2)) {
|
---|
285 | // For visualization
|
---|
286 | SimpleRay::IntersectionRes[0].intersectable = hitB.mObject;
|
---|
287 | SimpleRay::IntersectionRes[0].tdist = tdist[0];
|
---|
288 | }
|
---|
289 | } // dynamicFlag
|
---|
290 |
|
---|
291 | // restore the direction to the original
|
---|
292 | *v = -simpleRay.mDirection;
|
---|
293 | } // castDoubleRay
|
---|
294 |
|
---|
295 | #ifdef _PROCESS_RAY
|
---|
296 | // This code is also in IntelRayCaster.cpp
|
---|
297 | return ProcessRay(
|
---|
298 | simpleRay,
|
---|
299 | hitA,
|
---|
300 | hitB,
|
---|
301 | vssRays,
|
---|
302 | box,
|
---|
303 | castDoubleRay,
|
---|
304 | pruneInvalidRays
|
---|
305 | );
|
---|
306 | #else // _PROCESS_RAY
|
---|
307 | if (castDoubleRay)
|
---|
308 | return resultB;
|
---|
309 | return result;
|
---|
310 | #endif // _PROCESS_RAY
|
---|
311 |
|
---|
312 | #else
|
---|
313 | return 0;
|
---|
314 | #endif // USE_HAVRAN_RAYCASTER
|
---|
315 | }
|
---|
316 |
|
---|
317 | void HavranDynRayCaster::CastRays16(SimpleRayContainer &rays,
|
---|
318 | VssRayContainer &vssRays,
|
---|
319 | const AxisAlignedBox3 &sbox,
|
---|
320 | const bool castDoubleRay,
|
---|
321 | const bool pruneInvalidRays)
|
---|
322 | {
|
---|
323 | CastRays16(rays, 0, vssRays, sbox, castDoubleRay, pruneInvalidRays);
|
---|
324 | }
|
---|
325 |
|
---|
326 |
|
---|
327 | void HavranDynRayCaster::CastRays16(SimpleRayContainer &rays,
|
---|
328 | int offset,
|
---|
329 | VssRayContainer &vssRays,
|
---|
330 | const AxisAlignedBox3 &sbox,
|
---|
331 | const bool castDoubleRays,
|
---|
332 | const bool pruneInvalidRays)
|
---|
333 | {
|
---|
334 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
335 |
|
---|
336 | #if DEBUG_RAYCAST
|
---|
337 | Debug << "C16 " << flush;
|
---|
338 | #endif
|
---|
339 |
|
---|
340 | // Use special algorithm for 16 rays at once
|
---|
341 | if (castDoubleRays) {
|
---|
342 | // Here we decompose shooting into two phases
|
---|
343 |
|
---|
344 | // First we delete the results from casting dynamic objects
|
---|
345 | if (dynamicFlag) {
|
---|
346 | for (int i = 0; i < 32; i++)
|
---|
347 | objI[i] = 0;
|
---|
348 | }
|
---|
349 |
|
---|
350 | // Here we shoot first backward rays and forward ones
|
---|
351 | SimpleRayContainer::iterator sit = rays.begin() + offset;
|
---|
352 | SimpleRayContainer::const_iterator sit_end = rays.begin() + offset + 16;
|
---|
353 | for ( ; sit != sit_end; ++ sit)
|
---|
354 | {
|
---|
355 | (*sit).mDirection = - (*sit).mDirection;
|
---|
356 | }
|
---|
357 | // backward rays to be shot - saving with offset 16
|
---|
358 |
|
---|
359 | #ifdef _USE_HAVRAN_SSE
|
---|
360 | mKtbtree->SetOffset(0);
|
---|
361 | mKtbtree->FindNearestI_16oneDir(rays, offset, 16);
|
---|
362 | #else
|
---|
363 | mKtbtree->SetOffset(16);
|
---|
364 | mKtbtree->FindNearestI_16oneDirNoSSE(rays, offset);
|
---|
365 | #endif // _USE_HAVRAN_SSE
|
---|
366 |
|
---|
367 | if (dynamicFlag) {
|
---|
368 | for (int i = 0; i < 16; i++) {
|
---|
369 | // store the origin and direction of the ray
|
---|
370 | orig[i] = rays[offset+i].mOrigin;
|
---|
371 | dirs[i] = rays[offset+i].mDirection;
|
---|
372 |
|
---|
373 | objI[i+16] = SimpleRay::IntersectionRes[i+16].intersectable;
|
---|
374 | if (objI[i+16])
|
---|
375 | normal[i+16] = objI[i+16]->GetNormal(0);
|
---|
376 | tdist[i+16] = SimpleRay::IntersectionRes[i+16].tdist;
|
---|
377 | ApplyTransform(rays[offset+i]);
|
---|
378 | } // for
|
---|
379 | // Now shoot the ray with dynamic object
|
---|
380 | #ifdef _USE_HAVRAN_SSE
|
---|
381 | mDynKtbtree->SetOffset(0);
|
---|
382 | mDynKtbtree->FindNearestI_16oneDir(rays, offset, 16);
|
---|
383 | #else
|
---|
384 | mDynKtbtree->SetOffset(16);
|
---|
385 | mDynKtbtree->FindNearestI_16oneDirNoSSE(rays, offset);
|
---|
386 | #endif // _USE_HAVRAN_SSE
|
---|
387 | for (int i = 0; i < 16; i++) {
|
---|
388 | Intersectable *intersectable =
|
---|
389 | SimpleRay::IntersectionRes[i+16].intersectable;
|
---|
390 | if (intersectable) {
|
---|
391 | float t = SimpleRay::IntersectionRes[i+16].tdist;
|
---|
392 | if ( ((objI[i+16]) && (t < tdist[i+16])) ||
|
---|
393 | (!objI[i+16]) ) {
|
---|
394 | // There is an intersection closer than the previous one
|
---|
395 | objI[i+16] = intersectable;
|
---|
396 | tdist[i+16] = SimpleRay::IntersectionRes[i+16].tdist;
|
---|
397 | normal[i+16] = intersectable->GetNormal(0);
|
---|
398 | // We have to transform the normal
|
---|
399 | normal[i+16] = TransformNormal(matTr_inv, normal[i+16]);
|
---|
400 | } // if update
|
---|
401 | } // if intersected dynamic object
|
---|
402 | else {
|
---|
403 | // we copy back the static object - only for visualization
|
---|
404 | SimpleRay::IntersectionRes[i+16].intersectable = objI[i+16];
|
---|
405 | SimpleRay::IntersectionRes[i+16].tdist = tdist[i+16];
|
---|
406 | }
|
---|
407 | } // for
|
---|
408 | } // dynamic flag
|
---|
409 |
|
---|
410 | // set the ray direction to original direction
|
---|
411 | if (dynamicFlag) {
|
---|
412 | sit = rays.begin() + offset;
|
---|
413 | int i = 0;
|
---|
414 | for ( ; sit != sit_end; ++sit, ++i)
|
---|
415 | {
|
---|
416 | rays[offset+i].mOrigin = orig[i];
|
---|
417 | rays[offset+i].mDirection = -dirs[i];
|
---|
418 | }
|
---|
419 | }
|
---|
420 | else {
|
---|
421 | // no dynamic objects
|
---|
422 | sit = rays.begin() + offset;
|
---|
423 | for ( ; sit != sit_end; ++ sit)
|
---|
424 | {
|
---|
425 | (*sit).mDirection = - (*sit).mDirection;
|
---|
426 | }
|
---|
427 | }
|
---|
428 | // forward rays to be shot
|
---|
429 | #ifdef _USE_HAVRAN_SSE
|
---|
430 | mKtbtree->SetOffset(0);
|
---|
431 | mKtbtree->FindNearestI_16oneDir(rays, offset, 0);
|
---|
432 | #else
|
---|
433 | mKtbtree->SetOffset(0);
|
---|
434 | mKtbtree->FindNearestI_16oneDirNoSSE(rays, offset);
|
---|
435 | #endif // _USE_HAVRAN_SSE
|
---|
436 |
|
---|
437 | if (dynamicFlag) {
|
---|
438 | for (int i = 0; i < 16; i++) {
|
---|
439 | objI[i] = SimpleRay::IntersectionRes[i].intersectable;
|
---|
440 | if (objI[i])
|
---|
441 | normal[i] = objI[i]->GetNormal(0);
|
---|
442 | tdist[i] = SimpleRay::IntersectionRes[i].tdist;
|
---|
443 | ApplyTransform(rays[offset+i]);
|
---|
444 | } // for
|
---|
445 | // Now shoot the ray with dynamic object
|
---|
446 | #ifdef _USE_HAVRAN_SSE
|
---|
447 | mDynKtbtree->SetOffset(0);
|
---|
448 | mDynKtbtree->FindNearestI_16oneDir(rays, offset, 0);
|
---|
449 | #else
|
---|
450 | mDynKtbtree->SetOffset(0);
|
---|
451 | mDynKtbtree->FindNearestI_16oneDirNoSSE(rays, offset);
|
---|
452 | #endif // _USE_HAVRAN_SSE
|
---|
453 | for (int i = 0; i < 16; i++) {
|
---|
454 | Intersectable *intersectable
|
---|
455 | = SimpleRay::IntersectionRes[i].intersectable;
|
---|
456 | if (intersectable) {
|
---|
457 | float t = SimpleRay::IntersectionRes[i].tdist;
|
---|
458 | if ( ((objI[i]) && (t < tdist[i])) ||
|
---|
459 | (!objI[i]) ) {
|
---|
460 | // There is an intersection closer than the previous one
|
---|
461 | objI[i] = intersectable;
|
---|
462 | tdist[i] = SimpleRay::IntersectionRes[i].tdist;
|
---|
463 | normal[i] = objI[i]->GetNormal(0);
|
---|
464 | // We have to transform the normal
|
---|
465 | normal[i] = TransformNormal(matTr_inv, normal[i]);
|
---|
466 | } // update
|
---|
467 | } // intersection
|
---|
468 | // we have to recover the ray
|
---|
469 | rays[offset+i].mOrigin = orig[i];
|
---|
470 | rays[offset+i].mDirection = -dirs[i];
|
---|
471 | } // for i
|
---|
472 | } // dynamic flag
|
---|
473 | } // cast double rays
|
---|
474 | else {
|
---|
475 | // ONLY forward rays - TESTED 21/1/2007 VH
|
---|
476 |
|
---|
477 | // First we delete the results from casting dynamic objects
|
---|
478 | if (dynamicFlag) {
|
---|
479 | for (int i = 0; i < 16; i++) {
|
---|
480 | objI[i] = 0;
|
---|
481 | orig[i] = rays[offset+i].mOrigin;
|
---|
482 | dirs[i] = rays[offset+i].mDirection;
|
---|
483 | }
|
---|
484 | }
|
---|
485 |
|
---|
486 | // Shoot all 16 rays at the same time using a special algorithm
|
---|
487 | mKtbtree->SetOffset(0);
|
---|
488 | #ifdef _USE_HAVRAN_SSE
|
---|
489 | mKtbtree->FindNearestI_16oneDir(rays, offset, 0);
|
---|
490 | #else
|
---|
491 | mKtbtree->FindNearestI_16oneDirNoSSE(rays, offset);
|
---|
492 | #endif // _USE_HAVRAN_SSE
|
---|
493 |
|
---|
494 | // Here we must add dynamic objects
|
---|
495 | if (dynamicFlag) {
|
---|
496 | for (int i = 0; i < 16; i++) {
|
---|
497 | objI[i] = SimpleRay::IntersectionRes[i].intersectable;
|
---|
498 | if (objI[i])
|
---|
499 | normal[i] = objI[i]->GetNormal(0);
|
---|
500 | tdist[i] = SimpleRay::IntersectionRes[i].tdist;
|
---|
501 | ApplyTransform(rays[offset+i]);
|
---|
502 | } // for
|
---|
503 | // Now shoot the ray with dynamic object
|
---|
504 | #ifdef _USE_HAVRAN_SSE
|
---|
505 | mDynKtbtree->SetOffset(0);
|
---|
506 | mDynKtbtree->FindNearestI_16oneDir(rays, offset, 0);
|
---|
507 | #else
|
---|
508 | mDynKtbtree->SetOffset(0);
|
---|
509 | mDynKtbtree->FindNearestI_16oneDirNoSSE(rays, offset);
|
---|
510 | #endif // _USE_HAVRAN_SSE
|
---|
511 | for (int i = 0; i < 16; i++) {
|
---|
512 | Intersectable *intersectable
|
---|
513 | = SimpleRay::IntersectionRes[i].intersectable;
|
---|
514 | if (intersectable) {
|
---|
515 | float t = SimpleRay::IntersectionRes[i].tdist;
|
---|
516 | if ( ((objI[i]) && (t < tdist[i])) ||
|
---|
517 | (!objI[i]) ) {
|
---|
518 | // There is an intersection closer than the previous one
|
---|
519 | objI[i] = intersectable;
|
---|
520 | tdist[i] = SimpleRay::IntersectionRes[i].tdist;
|
---|
521 | normal[i] = objI[i]->GetNormal(0);
|
---|
522 | // We have to transform the normal
|
---|
523 | normal[i] = TransformNormal(matTr_inv, normal[i]);
|
---|
524 | } // update
|
---|
525 | } // intersection
|
---|
526 | // we have to recover the rays back
|
---|
527 | rays[offset+i].mOrigin = orig[i];
|
---|
528 | rays[offset+i].mDirection = dirs[i];
|
---|
529 | } // for i
|
---|
530 | } // dynamic flag
|
---|
531 | }
|
---|
532 |
|
---|
533 | if (dynamicFlag) {
|
---|
534 | for (int i=0, k=offset; i < 16; i++, k++)
|
---|
535 | {
|
---|
536 | Intersection hitA(rays[k].mOrigin), hitB(rays[k].mOrigin);
|
---|
537 |
|
---|
538 | #if DEBUG_RAYCAST
|
---|
539 | Debug<<"FH\n"<<flush;
|
---|
540 | #endif
|
---|
541 |
|
---|
542 | Intersectable *intersectable = objI[i];
|
---|
543 | // For visualization needs only - forward rays
|
---|
544 | SimpleRay::IntersectionRes[i].intersectable = objI[i];
|
---|
545 | SimpleRay::IntersectionRes[i].tdist = tdist[i];
|
---|
546 | if (intersectable)
|
---|
547 | {
|
---|
548 | hitA.mObject = intersectable;
|
---|
549 | // Get the normal of that face
|
---|
550 | hitA.mNormal = normal[i];
|
---|
551 |
|
---|
552 | //-rays[index+i].mDirection; // $$ temporary
|
---|
553 | hitA.mPoint = rays[k].Extrap(tdist[i]);
|
---|
554 | }
|
---|
555 |
|
---|
556 | #if DEBUG_RAYCAST
|
---|
557 | Debug<<"BH\n"<<flush;
|
---|
558 | #endif
|
---|
559 |
|
---|
560 | if (castDoubleRays)
|
---|
561 | {
|
---|
562 | Intersectable *intersectable = objI[i+16];
|
---|
563 |
|
---|
564 | if (intersectable)
|
---|
565 | {
|
---|
566 | hitB.mObject = intersectable;
|
---|
567 | hitB.mNormal = normal[i+16];
|
---|
568 | hitB.mPoint = rays[k].Extrap(-tdist[i+16]);
|
---|
569 | }
|
---|
570 | }
|
---|
571 |
|
---|
572 | #if DEBUG_RAYCAST
|
---|
573 | Debug<<"PR\n"<<flush;
|
---|
574 | #endif
|
---|
575 |
|
---|
576 | #ifdef _PROCESS_RAY
|
---|
577 | ProcessRay(rays[k],
|
---|
578 | hitA,
|
---|
579 | hitB,
|
---|
580 | vssRays,
|
---|
581 | sbox,
|
---|
582 | castDoubleRays,
|
---|
583 | pruneInvalidRays
|
---|
584 | );
|
---|
585 | #endif
|
---|
586 | } // for
|
---|
587 | }
|
---|
588 | else {
|
---|
589 | // no dynamic objects
|
---|
590 | for (int i=0, k=offset; i < 16; i++, k++)
|
---|
591 | {
|
---|
592 | Intersection hitA(rays[k].mOrigin), hitB(rays[k].mOrigin);
|
---|
593 |
|
---|
594 | #if DEBUG_RAYCAST
|
---|
595 | Debug<<"FH\n"<<flush;
|
---|
596 | #endif
|
---|
597 |
|
---|
598 | Intersectable *intersectable
|
---|
599 | = SimpleRay::IntersectionRes[i].intersectable;
|
---|
600 |
|
---|
601 | if (intersectable)
|
---|
602 | {
|
---|
603 | hitA.mObject = intersectable;
|
---|
604 | // Get the normal of that face
|
---|
605 | hitA.mNormal = intersectable->GetNormal(0);
|
---|
606 |
|
---|
607 | //-rays[index+i].mDirection; // $$ temporary
|
---|
608 | float tdist = SimpleRay::IntersectionRes[i].tdist;
|
---|
609 | hitA.mPoint = rays[k].Extrap(tdist);
|
---|
610 | }
|
---|
611 |
|
---|
612 | #if DEBUG_RAYCAST
|
---|
613 | Debug<<"BH\n"<<flush;
|
---|
614 | #endif
|
---|
615 |
|
---|
616 | if (castDoubleRays)
|
---|
617 | {
|
---|
618 | Intersectable *intersectable =
|
---|
619 | SimpleRay::IntersectionRes[i+16].intersectable;
|
---|
620 |
|
---|
621 | if (intersectable)
|
---|
622 | {
|
---|
623 | hitB.mObject = intersectable;
|
---|
624 | hitB.mNormal = intersectable->GetNormal(0);;
|
---|
625 | float tdist = SimpleRay::IntersectionRes[16+i].tdist;
|
---|
626 | hitB.mPoint = rays[k].Extrap(-tdist);
|
---|
627 | }
|
---|
628 | }
|
---|
629 |
|
---|
630 | #if DEBUG_RAYCAST
|
---|
631 | Debug<<"PR\n"<<flush;
|
---|
632 | #endif
|
---|
633 |
|
---|
634 | #ifdef _PROCESS_RAY
|
---|
635 | ProcessRay(rays[k],
|
---|
636 | hitA,
|
---|
637 | hitB,
|
---|
638 | vssRays,
|
---|
639 | sbox,
|
---|
640 | castDoubleRays,
|
---|
641 | pruneInvalidRays
|
---|
642 | );
|
---|
643 | #endif
|
---|
644 | } // for
|
---|
645 | } // dynamic objects
|
---|
646 |
|
---|
647 |
|
---|
648 | #if DEBUG_RAYCAST
|
---|
649 | Debug<<"C16F\n"<<flush;
|
---|
650 | #endif
|
---|
651 |
|
---|
652 | #endif // USE_HAVRAN_RAYCASTER
|
---|
653 | }
|
---|
654 |
|
---|
655 |
|
---|
656 |
|
---|
657 | void
|
---|
658 | HavranDynRayCaster::CastSimpleForwardRays(
|
---|
659 | SimpleRayContainer &rays,
|
---|
660 | const AxisAlignedBox3 &sbox
|
---|
661 | )
|
---|
662 | {
|
---|
663 | int hit_triangles[16];
|
---|
664 | float dist[16];
|
---|
665 | Vector3 normals[16];
|
---|
666 | Vector3 min = sbox.Min();
|
---|
667 | Vector3 max = sbox.Max();
|
---|
668 |
|
---|
669 | int packets = rays.size() / 16;
|
---|
670 |
|
---|
671 | int i, j, k = 0;
|
---|
672 | Vector3 dir;
|
---|
673 |
|
---|
674 | // By groups of rays
|
---|
675 | for (i=0; i < packets; i++) {
|
---|
676 | int offset = i * 16;
|
---|
677 | mKtbtree->FindNearestI_16oneDir(rays, offset, 0);
|
---|
678 |
|
---|
679 | if (dynamicFlag) {
|
---|
680 | for (int i = 0; i < 16; i++) {
|
---|
681 | orig[i] = rays[i].mOrigin;
|
---|
682 | dirs[i] = rays[i].mDirection;
|
---|
683 |
|
---|
684 | tdist[i] = SimpleRay::IntersectionRes[i].tdist;
|
---|
685 | objI[i] = SimpleRay::IntersectionRes[i].intersectable;
|
---|
686 | // for dynamic objects
|
---|
687 | ApplyTransform(rays[i]);
|
---|
688 | } // for
|
---|
689 |
|
---|
690 | // shoot the rays against dynamic objects
|
---|
691 | mDynKtbtree->FindNearestI_16oneDir(rays, offset, 0);
|
---|
692 | // and combine the results
|
---|
693 | for (int i = 0; i < 16; i++) {
|
---|
694 | float t = SimpleRay::IntersectionRes[i].tdist;
|
---|
695 | if (SimpleRay::IntersectionRes[i].intersectable) {
|
---|
696 | if ( ((objI[i]) && (t < tdist[i])) ||
|
---|
697 | (!objI[i]) ) {
|
---|
698 | // There is an intersection closer than the previous one
|
---|
699 | objI[i] = SimpleRay::IntersectionRes[i].intersectable;
|
---|
700 | tdist[i] = SimpleRay::IntersectionRes[i].tdist;
|
---|
701 | } // update
|
---|
702 | } // intersection
|
---|
703 | else {
|
---|
704 | // Only for visualization
|
---|
705 | SimpleRay::IntersectionRes[i].intersectable = objI[i];
|
---|
706 | SimpleRay::IntersectionRes[i].tdist = tdist[i];
|
---|
707 | }
|
---|
708 | // recover ray origin and direction
|
---|
709 | rays[i].mOrigin = orig[i];
|
---|
710 | rays[i].mDirection = dirs[i];
|
---|
711 | } // for
|
---|
712 |
|
---|
713 | } // dynamic objects
|
---|
714 |
|
---|
715 | // ??? What to do with the results ? These are
|
---|
716 | // not used at the moment also in IntelRayCaster.cpp
|
---|
717 | } // for
|
---|
718 |
|
---|
719 |
|
---|
720 | i = 0;
|
---|
721 | int k2 = k;
|
---|
722 | for (; k < rays.size(); k++, i++) {
|
---|
723 | double normal[3];
|
---|
724 | // Intersect with the static scene
|
---|
725 | bool result = mKtbtree->FindNearestI(rays[k]);
|
---|
726 | tdist[i] = SimpleRay::IntersectionRes[0].tdist;
|
---|
727 | objI[i] = SimpleRay::IntersectionRes[0].intersectable;
|
---|
728 |
|
---|
729 | if (dynamicFlag) {
|
---|
730 | orig[i] = rays[k].mOrigin;
|
---|
731 | dirs[i] = rays[k].mDirection;
|
---|
732 |
|
---|
733 | // Intersect with the dynamic object
|
---|
734 | ApplyTransform(rays[k]);
|
---|
735 | bool result2 = false;
|
---|
736 | if (mDynKtbtree->FindNearestI(rays[k])) {
|
---|
737 | float t = SimpleRay::IntersectionRes[0].tdist;
|
---|
738 | if ( ((objI[i]) && (t < tdist[i])) ||
|
---|
739 | (!objI[i]) ) {
|
---|
740 | // There is an intersection closer than the previous one
|
---|
741 | result2 = true;
|
---|
742 | objI[i] = SimpleRay::IntersectionRes[0].intersectable;
|
---|
743 | tdist[i] = t;
|
---|
744 | }
|
---|
745 | }
|
---|
746 |
|
---|
747 | // recover ray origin and direction
|
---|
748 | rays[k].mOrigin = orig[i];
|
---|
749 | rays[k].mDirection = dirs[i];
|
---|
750 | } // dynamic objects
|
---|
751 |
|
---|
752 | // ??? What to do with the results ? These are
|
---|
753 | // not used at the moment also in IntelRayCaster.cpp
|
---|
754 | }
|
---|
755 | // The result will be in the array
|
---|
756 | i = 0;
|
---|
757 | for (; k2 < rays.size(); k2++, i++) {
|
---|
758 | SimpleRay::IntersectionRes[i].intersectable = objI[i];
|
---|
759 | SimpleRay::IntersectionRes[i].tdist = tdist[i];
|
---|
760 | }
|
---|
761 |
|
---|
762 | return;
|
---|
763 | }
|
---|
764 |
|
---|
765 | void HavranDynRayCaster::CastRays(
|
---|
766 | SimpleRayContainer &rays,
|
---|
767 | VssRayContainer &vssRays,
|
---|
768 | const AxisAlignedBox3 &sbox,
|
---|
769 | const bool castDoubleRay,
|
---|
770 | const bool pruneInvalidRays )
|
---|
771 | {
|
---|
772 | int buckets = rays.size()/16;
|
---|
773 | int offset = 0;
|
---|
774 |
|
---|
775 | #if 0
|
---|
776 | int time = GetTime();
|
---|
777 | CastSimpleForwardRays(rays, sbox);
|
---|
778 | cout<<1e-3*2*rays.size()/TimeDiff(time, GetTime())<<" Mrays/sec"<<endl;
|
---|
779 | #endif
|
---|
780 |
|
---|
781 | // Cast only by 16 rays at once
|
---|
782 | for (int i=0; i < buckets; i++, offset+=16) {
|
---|
783 | CastRays16(rays, offset, vssRays, sbox,
|
---|
784 | castDoubleRay, pruneInvalidRays);
|
---|
785 |
|
---|
786 | if ((int)rays.size() > 100000 && i % (100000/16) == 0)
|
---|
787 | cout<<"\r"<<offset<<"/"<<(int)rays.size()<<"\r";
|
---|
788 | }
|
---|
789 |
|
---|
790 | // Cast the rest of the rays
|
---|
791 | for (; offset < (int)rays.size(); offset++)
|
---|
792 | CastRay(rays[offset], vssRays, sbox, castDoubleRay, pruneInvalidRays);
|
---|
793 |
|
---|
794 | return;
|
---|
795 | }
|
---|
796 |
|
---|
797 | #ifdef _USE_HAVRAN_SSE
|
---|
798 | // BUG1 41579 196948 1064111
|
---|
799 | // BUG2 254 1672 10869
|
---|
800 |
|
---|
801 | // Just for testing concept
|
---|
802 | void
|
---|
803 | HavranDynRayCaster::CastRaysPacket2x2(RayPacket2x2 &raysPack,
|
---|
804 | bool castDoubleRay,
|
---|
805 | const bool pruneInvalidRays)
|
---|
806 | {
|
---|
807 | #ifdef USE_HAVRAN_RAYCASTER
|
---|
808 | #ifdef _USE_HAVRAN_SSE
|
---|
809 |
|
---|
810 | if (castDoubleRay) {
|
---|
811 | // cast forward rays
|
---|
812 | mKtbtree->FindNearestI(raysPack);
|
---|
813 |
|
---|
814 | if (dynamicFlag) {
|
---|
815 | // The ray pack to be transformed
|
---|
816 | raypack_t = raysPack;
|
---|
817 | raypack.ApplyTransform(matTr_inv);
|
---|
818 |
|
---|
819 | mDynKtbtree->FindNearestI(raypack_t);
|
---|
820 |
|
---|
821 | for (int i = 0; i < 4; i++) {
|
---|
822 | // if ray intersects an object, we set the pointer to
|
---|
823 | // this object
|
---|
824 | Intersectable* intersectable = raypack_t.GetObject(i);
|
---|
825 | if (intersectable) {
|
---|
826 | Intersectable *o = raysPack.GetObject(i);
|
---|
827 | if ( ((o) && (raypack_t.GetT(i) < tdist[i])) ||
|
---|
828 | (!o) ) {
|
---|
829 | // update, dynamic object is closer
|
---|
830 | raysPack.SetT(i, raypack_t.GetT(i));
|
---|
831 | raysPack.SetObject(i, raypack_t.GetObject(i));
|
---|
832 | } // if update is required
|
---|
833 | } // if intersectable
|
---|
834 | } // for i
|
---|
835 | } // dynamic object
|
---|
836 |
|
---|
837 | // Double (=opposite) direction
|
---|
838 | for (int i = 0; i < 4; i++)
|
---|
839 | raysPack.SetDir(i, -raysPack.GetDir(i));
|
---|
840 | // cast backward rays
|
---|
841 | mKtbtree->FindNearestI(raysPack);
|
---|
842 |
|
---|
843 | if (dynamicFlag) {
|
---|
844 |
|
---|
845 | // The ray pack to be transformed
|
---|
846 | raypack_t = raysPack;
|
---|
847 | raypack_t.ApplyTransform(matTr_inv);
|
---|
848 |
|
---|
849 | mDynKtbtree->FindNearestI(raypack_t);
|
---|
850 |
|
---|
851 | for (int i = 0; i < 4; i++) {
|
---|
852 | // if ray intersects an object, we set the pointer to
|
---|
853 | // this object
|
---|
854 | Intersectable* intersectable = raypack_t.GetObject(i);
|
---|
855 | if (intersectable) {
|
---|
856 | Intersectable *o = raysPack.GetObject(i);
|
---|
857 | if ( ((o) && (raypack_t.GetT(i) < tdist[i])) ||
|
---|
858 | (!o) ) {
|
---|
859 | // update, dynamic object is closer
|
---|
860 | raysPack.SetT(i, raypack_t.GetT(i));
|
---|
861 | raysPack.SetObject(i, raypack_t.GetObject(i));
|
---|
862 | } // if update is required
|
---|
863 | } // if intersectable
|
---|
864 | } // for i
|
---|
865 | } // dynamic object
|
---|
866 |
|
---|
867 | // reverse the rays back
|
---|
868 | for (int i = 0; i < 4; i++)
|
---|
869 | raysPack.SetDir(i, -raysPack.GetDir(i));
|
---|
870 | }
|
---|
871 | else {
|
---|
872 | // ONLY forward rays - TESTED 21/1/2007 VH
|
---|
873 |
|
---|
874 | mKtbtree->FindNearestI(raysPack);
|
---|
875 |
|
---|
876 | if (dynamicFlag) {
|
---|
877 | // The ray pack to be transformed
|
---|
878 | raypack_t = raysPack;
|
---|
879 | raypack_t.ApplyTransform(matTr_inv);
|
---|
880 |
|
---|
881 | mDynKtbtree->FindNearestI(raypack_t);
|
---|
882 |
|
---|
883 | for (int i = 0; i < 4; i++) {
|
---|
884 | // if ray intersects an object, we set the pointer to
|
---|
885 | // this object
|
---|
886 | Intersectable* intersectable = raypack_t.GetObject(i);
|
---|
887 | if (intersectable) {
|
---|
888 | Intersectable *o = raysPack.GetObject(i);
|
---|
889 | if ( ((o) && (raypack_t.GetT(i) < tdist[i])) ||
|
---|
890 | (!o) ) {
|
---|
891 | // update, dynamic object is closer
|
---|
892 | raysPack.SetT(i, raypack_t.GetT(i));
|
---|
893 | raysPack.SetObject(i, raypack_t.GetObject(i));
|
---|
894 | } // if update is required
|
---|
895 | } // if intersectable
|
---|
896 | } // for i
|
---|
897 | } // dynamic object
|
---|
898 |
|
---|
899 | #if 0
|
---|
900 | // Only verification of correctness by casting single rays
|
---|
901 | static int cntBugs = 0;
|
---|
902 | SimpleRay ray;
|
---|
903 | int cntErrors = 0, cntDistErrors = 0;
|
---|
904 | bool newBug = false;
|
---|
905 | for (int i = 0; i < 4; i++) {
|
---|
906 | ray.mOrigin = raysPack.GetLoc(i);
|
---|
907 | ray.mDirection = raysPack.GetDir(i);
|
---|
908 | mKtbtree->FindNearestI(ray);
|
---|
909 | if (raysPack.GetObject(i) != SimpleRay::IntersectionRes[0].intersectable) {
|
---|
910 | float dist = (raysPack.GetT(i) - SimpleRay::IntersectionRes[0].tdist);
|
---|
911 | if (fabs(dist) > 0.001f) {
|
---|
912 | cntErrors++; newBug = true;
|
---|
913 | cntBugs++;
|
---|
914 | cout << " BUG1 d= " << dist;
|
---|
915 | }
|
---|
916 | }
|
---|
917 | else {
|
---|
918 | float dist = 0.f;
|
---|
919 | if (raysPack.GetObject(i) && SimpleRay::IntersectionRes[0].intersectable)
|
---|
920 | if (fabs((dist=(fabs (raysPack.GetT(i) - SimpleRay::IntersectionRes[0].tdist)))) > 1.f) {
|
---|
921 | cntDistErrors++; newBug = true; cntBugs++;
|
---|
922 | cout << " BUG2 distdf= " << dist ;
|
---|
923 | }
|
---|
924 | }
|
---|
925 | } // for
|
---|
926 | if (newBug) cout << " CB= " << cntBugs << "\n";
|
---|
927 | #endif
|
---|
928 | }
|
---|
929 |
|
---|
930 | return;
|
---|
931 | #endif // _USE_HAVRAN_SSE
|
---|
932 | #endif // USE_HAVRAN_RAYCASTER
|
---|
933 | }
|
---|
934 |
|
---|
935 |
|
---|
936 | #endif // _USE_HAVRAN_SSE
|
---|
937 |
|
---|
938 |
|
---|
939 | } // the namespace
|
---|