diff options
Diffstat (limited to 'OpenSim/Region/Physics/UbitOdePlugin/ODERayCastRequestManager.cs')
-rw-r--r-- | OpenSim/Region/Physics/UbitOdePlugin/ODERayCastRequestManager.cs | 624 |
1 files changed, 624 insertions, 0 deletions
diff --git a/OpenSim/Region/Physics/UbitOdePlugin/ODERayCastRequestManager.cs b/OpenSim/Region/Physics/UbitOdePlugin/ODERayCastRequestManager.cs new file mode 100644 index 0000000..21fe9c0 --- /dev/null +++ b/OpenSim/Region/Physics/UbitOdePlugin/ODERayCastRequestManager.cs | |||
@@ -0,0 +1,624 @@ | |||
1 | /* | ||
2 | * Copyright (c) Contributors, http://opensimulator.org/ | ||
3 | * See CONTRIBUTORS.TXT for a full list of copyright holders. | ||
4 | * | ||
5 | * Redistribution and use in source and binary forms, with or without | ||
6 | * modification, are permitted provided that the following conditions are met: | ||
7 | * * Redistributions of source code must retain the above copyright | ||
8 | * notice, this list of conditions and the following disclaimer. | ||
9 | * * Redistributions in binary form must reproduce the above copyright | ||
10 | * notice, this list of conditions and the following disclaimer in the | ||
11 | * documentation and/or other materials provided with the distribution. | ||
12 | * * Neither the name of the OpenSimulator Project nor the | ||
13 | * names of its contributors may be used to endorse or promote products | ||
14 | * derived from this software without specific prior written permission. | ||
15 | * | ||
16 | * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``AS IS'' AND ANY | ||
17 | * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | ||
18 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | ||
19 | * DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE FOR ANY | ||
20 | * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | ||
21 | * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | ||
22 | * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | ||
23 | * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
24 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | ||
25 | * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
26 | */ | ||
27 | |||
28 | using System; | ||
29 | using System.Collections.Generic; | ||
30 | using System.Reflection; | ||
31 | using System.Runtime.InteropServices; | ||
32 | using System.Text; | ||
33 | using OpenSim.Framework; | ||
34 | using OpenSim.Region.Physics.Manager; | ||
35 | using OdeAPI; | ||
36 | using log4net; | ||
37 | using OpenMetaverse; | ||
38 | |||
39 | namespace OpenSim.Region.Physics.OdePlugin | ||
40 | { | ||
41 | /// <summary> | ||
42 | /// Processes raycast requests as ODE is in a state to be able to do them. | ||
43 | /// This ensures that it's thread safe and there will be no conflicts. | ||
44 | /// Requests get returned by a different thread then they were requested by. | ||
45 | /// </summary> | ||
46 | public class ODERayCastRequestManager | ||
47 | { | ||
48 | /// <summary> | ||
49 | /// Pending ray requests | ||
50 | /// </summary> | ||
51 | protected OpenSim.Framework.LocklessQueue<ODERayRequest> m_PendingRequests = new OpenSim.Framework.LocklessQueue<ODERayRequest>(); | ||
52 | |||
53 | /// <summary> | ||
54 | /// Scene that created this object. | ||
55 | /// </summary> | ||
56 | private OdeScene m_scene; | ||
57 | |||
58 | IntPtr ray; // the ray. we only need one for our lifetime | ||
59 | |||
60 | private const int ColisionContactGeomsPerTest = 5; | ||
61 | private const int DefaultMaxCount = 25; | ||
62 | private const int MaxTimePerCallMS = 30; | ||
63 | |||
64 | /// <summary> | ||
65 | /// ODE near callback delegate | ||
66 | /// </summary> | ||
67 | private d.NearCallback nearCallback; | ||
68 | private static readonly ILog m_log = LogManager.GetLogger(MethodBase.GetCurrentMethod().DeclaringType); | ||
69 | private List<ContactResult> m_contactResults = new List<ContactResult>(); | ||
70 | private RayFilterFlags CurrentRayFilter; | ||
71 | private int CurrentMaxCount; | ||
72 | |||
73 | public ODERayCastRequestManager(OdeScene pScene) | ||
74 | { | ||
75 | m_scene = pScene; | ||
76 | nearCallback = near; | ||
77 | ray = d.CreateRay(IntPtr.Zero, 1.0f); | ||
78 | d.GeomSetCategoryBits(ray,0); | ||
79 | } | ||
80 | |||
81 | /// <summary> | ||
82 | /// Queues request for a raycast to all world | ||
83 | /// </summary> | ||
84 | /// <param name="position">Origin of Ray</param> | ||
85 | /// <param name="direction">Ray direction</param> | ||
86 | /// <param name="length">Ray length</param> | ||
87 | /// <param name="retMethod">Return method to send the results</param> | ||
88 | public void QueueRequest(Vector3 position, Vector3 direction, float length, RayCallback retMethod) | ||
89 | { | ||
90 | ODERayRequest req = new ODERayRequest(); | ||
91 | req.geom = IntPtr.Zero; | ||
92 | req.callbackMethod = retMethod; | ||
93 | req.Count = DefaultMaxCount; | ||
94 | req.length = length; | ||
95 | req.Normal = direction; | ||
96 | req.Origin = position; | ||
97 | req.filter = RayFilterFlags.AllPrims; | ||
98 | |||
99 | m_PendingRequests.Enqueue(req); | ||
100 | } | ||
101 | |||
102 | /// <summary> | ||
103 | /// Queues request for a raycast to particular part | ||
104 | /// </summary> | ||
105 | /// <param name="position">Origin of Ray</param> | ||
106 | /// <param name="direction">Ray direction</param> | ||
107 | /// <param name="length">Ray length</param> | ||
108 | /// <param name="retMethod">Return method to send the results</param> | ||
109 | public void QueueRequest(IntPtr geom, Vector3 position, Vector3 direction, float length, RayCallback retMethod) | ||
110 | { | ||
111 | ODERayRequest req = new ODERayRequest(); | ||
112 | req.geom = geom; | ||
113 | req.callbackMethod = retMethod; | ||
114 | req.length = length; | ||
115 | req.Normal = direction; | ||
116 | req.Origin = position; | ||
117 | req.Count = DefaultMaxCount; | ||
118 | req.filter = RayFilterFlags.AllPrims; | ||
119 | |||
120 | m_PendingRequests.Enqueue(req); | ||
121 | } | ||
122 | |||
123 | public void QueueRequest(Vector3 position, Vector3 direction, float length, RaycastCallback retMethod) | ||
124 | { | ||
125 | ODERayRequest req = new ODERayRequest(); | ||
126 | req.geom = IntPtr.Zero; | ||
127 | req.callbackMethod = retMethod; | ||
128 | req.Count = DefaultMaxCount; | ||
129 | req.length = length; | ||
130 | req.Normal = direction; | ||
131 | req.Origin = position; | ||
132 | req.filter = RayFilterFlags.AllPrims; | ||
133 | |||
134 | m_PendingRequests.Enqueue(req); | ||
135 | } | ||
136 | |||
137 | public void QueueRequest(IntPtr geom, Vector3 position, Vector3 direction, float length, RaycastCallback retMethod) | ||
138 | { | ||
139 | ODERayRequest req = new ODERayRequest(); | ||
140 | req.geom = geom; | ||
141 | req.callbackMethod = retMethod; | ||
142 | req.length = length; | ||
143 | req.Normal = direction; | ||
144 | req.Origin = position; | ||
145 | req.Count = DefaultMaxCount; | ||
146 | req.filter = RayFilterFlags.AllPrims; | ||
147 | |||
148 | m_PendingRequests.Enqueue(req); | ||
149 | } | ||
150 | |||
151 | /// <summary> | ||
152 | /// Queues a raycast | ||
153 | /// </summary> | ||
154 | /// <param name="position">Origin of Ray</param> | ||
155 | /// <param name="direction">Ray normal</param> | ||
156 | /// <param name="length">Ray length</param> | ||
157 | /// <param name="count"></param> | ||
158 | /// <param name="retMethod">Return method to send the results</param> | ||
159 | public void QueueRequest(Vector3 position, Vector3 direction, float length, int count, RayCallback retMethod) | ||
160 | { | ||
161 | ODERayRequest req = new ODERayRequest(); | ||
162 | req.geom = IntPtr.Zero; | ||
163 | req.callbackMethod = retMethod; | ||
164 | req.length = length; | ||
165 | req.Normal = direction; | ||
166 | req.Origin = position; | ||
167 | req.Count = count; | ||
168 | req.filter = RayFilterFlags.AllPrims; | ||
169 | |||
170 | m_PendingRequests.Enqueue(req); | ||
171 | } | ||
172 | |||
173 | |||
174 | public void QueueRequest(Vector3 position, Vector3 direction, float length, int count,RayFilterFlags filter , RayCallback retMethod) | ||
175 | { | ||
176 | ODERayRequest req = new ODERayRequest(); | ||
177 | req.geom = IntPtr.Zero; | ||
178 | req.callbackMethod = retMethod; | ||
179 | req.length = length; | ||
180 | req.Normal = direction; | ||
181 | req.Origin = position; | ||
182 | req.Count = count; | ||
183 | req.filter = filter; | ||
184 | |||
185 | m_PendingRequests.Enqueue(req); | ||
186 | } | ||
187 | |||
188 | public void QueueRequest(IntPtr geom, Vector3 position, Vector3 direction, float length, int count, RayCallback retMethod) | ||
189 | { | ||
190 | ODERayRequest req = new ODERayRequest(); | ||
191 | req.geom = geom; | ||
192 | req.callbackMethod = retMethod; | ||
193 | req.length = length; | ||
194 | req.Normal = direction; | ||
195 | req.Origin = position; | ||
196 | req.Count = count; | ||
197 | req.filter = RayFilterFlags.AllPrims; | ||
198 | |||
199 | m_PendingRequests.Enqueue(req); | ||
200 | } | ||
201 | |||
202 | public void QueueRequest(Vector3 position, Vector3 direction, float length, int count, RaycastCallback retMethod) | ||
203 | { | ||
204 | ODERayRequest req = new ODERayRequest(); | ||
205 | req.geom = IntPtr.Zero; | ||
206 | req.callbackMethod = retMethod; | ||
207 | req.length = length; | ||
208 | req.Normal = direction; | ||
209 | req.Origin = position; | ||
210 | req.Count = count; | ||
211 | req.filter = RayFilterFlags.AllPrims; | ||
212 | |||
213 | m_PendingRequests.Enqueue(req); | ||
214 | } | ||
215 | |||
216 | public void QueueRequest(IntPtr geom, Vector3 position, Vector3 direction, float length, int count, RaycastCallback retMethod) | ||
217 | { | ||
218 | ODERayRequest req = new ODERayRequest(); | ||
219 | req.geom = geom; | ||
220 | req.callbackMethod = retMethod; | ||
221 | req.length = length; | ||
222 | req.Normal = direction; | ||
223 | req.Origin = position; | ||
224 | req.Count = count; | ||
225 | req.filter = RayFilterFlags.AllPrims; | ||
226 | |||
227 | m_PendingRequests.Enqueue(req); | ||
228 | } | ||
229 | |||
230 | /// <summary> | ||
231 | /// Process all queued raycast requests | ||
232 | /// </summary> | ||
233 | /// <returns>Time in MS the raycasts took to process.</returns> | ||
234 | public int ProcessQueuedRequests() | ||
235 | { | ||
236 | |||
237 | if (m_PendingRequests.Count <= 0) | ||
238 | return 0; | ||
239 | |||
240 | if (m_scene.ContactgeomsArray == IntPtr.Zero || ray == IntPtr.Zero) | ||
241 | // oops something got wrong or scene isn't ready still | ||
242 | { | ||
243 | m_PendingRequests.Clear(); | ||
244 | return 0; | ||
245 | } | ||
246 | |||
247 | int time = Util.EnvironmentTickCount(); | ||
248 | |||
249 | ODERayRequest req; | ||
250 | int closestHit; | ||
251 | int backfacecull; | ||
252 | CollisionCategories catflags; | ||
253 | |||
254 | while (m_PendingRequests.Dequeue(out req)) | ||
255 | { | ||
256 | if (req.callbackMethod != null) | ||
257 | { | ||
258 | CurrentRayFilter = req.filter; | ||
259 | CurrentMaxCount = req.Count; | ||
260 | |||
261 | closestHit = ((CurrentRayFilter & RayFilterFlags.ClosestHit) == 0 ? 0 : 1); | ||
262 | backfacecull = ((CurrentRayFilter & RayFilterFlags.BackFaceCull) == 0 ? 0 : 1); | ||
263 | |||
264 | d.GeomRaySetLength(ray, req.length); | ||
265 | d.GeomRaySet(ray, req.Origin.X, req.Origin.Y, req.Origin.Z, req.Normal.X, req.Normal.Y, req.Normal.Z); | ||
266 | d.GeomRaySetParams(ray, 0, backfacecull); | ||
267 | d.GeomRaySetClosestHit(ray, closestHit); | ||
268 | |||
269 | if (req.callbackMethod is RaycastCallback) | ||
270 | // if we only want one get only one per colision pair saving memory | ||
271 | CurrentRayFilter |= RayFilterFlags.ClosestHit; | ||
272 | |||
273 | if (req.geom == IntPtr.Zero) | ||
274 | { | ||
275 | // translate ray filter to colision flags | ||
276 | catflags = 0; | ||
277 | if ((CurrentRayFilter & RayFilterFlags.volumedtc) != 0) | ||
278 | catflags |= CollisionCategories.VolumeDtc; | ||
279 | if ((CurrentRayFilter & RayFilterFlags.phantom) != 0) | ||
280 | catflags |= CollisionCategories.Phantom; | ||
281 | if ((CurrentRayFilter & RayFilterFlags.agent) != 0) | ||
282 | catflags |= CollisionCategories.Character; | ||
283 | if ((CurrentRayFilter & RayFilterFlags.PrimsNonPhantom) != 0) | ||
284 | catflags |= CollisionCategories.Geom; | ||
285 | if ((CurrentRayFilter & RayFilterFlags.land) != 0) | ||
286 | catflags |= CollisionCategories.Land; | ||
287 | if ((CurrentRayFilter & RayFilterFlags.water) != 0) | ||
288 | catflags |= CollisionCategories.Water; | ||
289 | |||
290 | if (catflags != 0) | ||
291 | doSpaceRay(req); | ||
292 | } | ||
293 | else | ||
294 | { | ||
295 | // if we select a geom don't use filters | ||
296 | d.GeomSetCollideBits(ray, (uint)CollisionCategories.All); | ||
297 | doGeomRay(req); | ||
298 | } | ||
299 | } | ||
300 | |||
301 | if (Util.EnvironmentTickCountSubtract(time) > MaxTimePerCallMS) | ||
302 | break; | ||
303 | } | ||
304 | |||
305 | lock (m_contactResults) | ||
306 | m_contactResults.Clear(); | ||
307 | |||
308 | return Util.EnvironmentTickCountSubtract(time); | ||
309 | } | ||
310 | /// <summary> | ||
311 | /// Method that actually initiates the raycast with spaces | ||
312 | /// </summary> | ||
313 | /// <param name="req"></param> | ||
314 | /// | ||
315 | |||
316 | private const RayFilterFlags FilterActiveSpace = RayFilterFlags.agent | RayFilterFlags.physical | RayFilterFlags.LSLPhanton; | ||
317 | private const RayFilterFlags FilterStaticSpace = RayFilterFlags.water | RayFilterFlags.land | RayFilterFlags.nonphysical | RayFilterFlags.LSLPhanton; | ||
318 | |||
319 | private void doSpaceRay(ODERayRequest req) | ||
320 | { | ||
321 | // Collide tests | ||
322 | if ((CurrentRayFilter & FilterActiveSpace) != 0) | ||
323 | d.SpaceCollide2(ray, m_scene.ActiveSpace, IntPtr.Zero, nearCallback); | ||
324 | if ((CurrentRayFilter & FilterStaticSpace) != 0 && (m_contactResults.Count < CurrentMaxCount)) | ||
325 | d.SpaceCollide2(ray, m_scene.StaticSpace, IntPtr.Zero, nearCallback); | ||
326 | |||
327 | if (req.callbackMethod is RaycastCallback) | ||
328 | { | ||
329 | // Define default results | ||
330 | bool hitYN = false; | ||
331 | uint hitConsumerID = 0; | ||
332 | float distance = float.MaxValue; | ||
333 | Vector3 closestcontact = Vector3.Zero; | ||
334 | Vector3 snormal = Vector3.Zero; | ||
335 | |||
336 | // Find closest contact and object. | ||
337 | lock (m_contactResults) | ||
338 | { | ||
339 | foreach (ContactResult cResult in m_contactResults) | ||
340 | { | ||
341 | if(cResult.Depth < distance) | ||
342 | { | ||
343 | closestcontact = cResult.Pos; | ||
344 | hitConsumerID = cResult.ConsumerID; | ||
345 | distance = cResult.Depth; | ||
346 | snormal = cResult.Normal; | ||
347 | } | ||
348 | } | ||
349 | m_contactResults.Clear(); | ||
350 | } | ||
351 | |||
352 | if (distance > 0 && distance < float.MaxValue) | ||
353 | hitYN = true; | ||
354 | ((RaycastCallback)req.callbackMethod)(hitYN, closestcontact, hitConsumerID, distance, snormal); | ||
355 | } | ||
356 | else | ||
357 | { | ||
358 | List<ContactResult> cresult = new List<ContactResult>(m_contactResults.Count); | ||
359 | lock (m_PendingRequests) | ||
360 | { | ||
361 | cresult.AddRange(m_contactResults); | ||
362 | m_contactResults.Clear(); | ||
363 | } | ||
364 | ((RayCallback)req.callbackMethod)(cresult); | ||
365 | } | ||
366 | } | ||
367 | |||
368 | /// <summary> | ||
369 | /// Method that actually initiates the raycast with a geom | ||
370 | /// </summary> | ||
371 | /// <param name="req"></param> | ||
372 | private void doGeomRay(ODERayRequest req) | ||
373 | { | ||
374 | // Collide test | ||
375 | d.SpaceCollide2(ray, req.geom, IntPtr.Zero, nearCallback); // still do this to have full AABB pre test | ||
376 | |||
377 | if (req.callbackMethod is RaycastCallback) | ||
378 | { | ||
379 | // Define default results | ||
380 | bool hitYN = false; | ||
381 | uint hitConsumerID = 0; | ||
382 | float distance = float.MaxValue; | ||
383 | Vector3 closestcontact = Vector3.Zero; | ||
384 | Vector3 snormal = Vector3.Zero; | ||
385 | |||
386 | // Find closest contact and object. | ||
387 | lock (m_contactResults) | ||
388 | { | ||
389 | foreach (ContactResult cResult in m_contactResults) | ||
390 | { | ||
391 | if(cResult.Depth < distance ) | ||
392 | { | ||
393 | closestcontact = cResult.Pos; | ||
394 | hitConsumerID = cResult.ConsumerID; | ||
395 | distance = cResult.Depth; | ||
396 | snormal = cResult.Normal; | ||
397 | } | ||
398 | } | ||
399 | m_contactResults.Clear(); | ||
400 | } | ||
401 | |||
402 | if (distance > 0 && distance < float.MaxValue) | ||
403 | hitYN = true; | ||
404 | |||
405 | ((RaycastCallback)req.callbackMethod)(hitYN, closestcontact, hitConsumerID, distance, snormal); | ||
406 | } | ||
407 | else | ||
408 | { | ||
409 | List<ContactResult> cresult = new List<ContactResult>(m_contactResults.Count); | ||
410 | lock (m_PendingRequests) | ||
411 | { | ||
412 | cresult.AddRange(m_contactResults); | ||
413 | m_contactResults.Clear(); | ||
414 | } | ||
415 | ((RayCallback)req.callbackMethod)(cresult); | ||
416 | } | ||
417 | } | ||
418 | |||
419 | private bool GetCurContactGeom(int index, ref d.ContactGeom newcontactgeom) | ||
420 | { | ||
421 | IntPtr ContactgeomsArray = m_scene.ContactgeomsArray; | ||
422 | if (ContactgeomsArray == IntPtr.Zero || index >= ColisionContactGeomsPerTest) | ||
423 | return false; | ||
424 | |||
425 | IntPtr contactptr = new IntPtr(ContactgeomsArray.ToInt64() + (Int64)(index * d.ContactGeom.unmanagedSizeOf)); | ||
426 | newcontactgeom = (d.ContactGeom)Marshal.PtrToStructure(contactptr, typeof(d.ContactGeom)); | ||
427 | return true; | ||
428 | } | ||
429 | |||
430 | // This is the standard Near. g1 is the ray | ||
431 | private void near(IntPtr space, IntPtr g1, IntPtr g2) | ||
432 | { | ||
433 | if (g2 == IntPtr.Zero || g1 == g2) | ||
434 | return; | ||
435 | |||
436 | if (m_contactResults.Count >= CurrentMaxCount) | ||
437 | return; | ||
438 | |||
439 | if (d.GeomIsSpace(g2)) | ||
440 | { | ||
441 | try | ||
442 | { | ||
443 | d.SpaceCollide2(g1, g2, IntPtr.Zero, nearCallback); | ||
444 | } | ||
445 | catch (Exception e) | ||
446 | { | ||
447 | m_log.WarnFormat("[PHYSICS Ray]: Unable to Space collide test an object: {0}", e.Message); | ||
448 | } | ||
449 | return; | ||
450 | } | ||
451 | |||
452 | int count = 0; | ||
453 | try | ||
454 | { | ||
455 | count = d.CollidePtr(g1, g2, ColisionContactGeomsPerTest, m_scene.ContactgeomsArray, d.ContactGeom.unmanagedSizeOf); | ||
456 | } | ||
457 | catch (Exception e) | ||
458 | { | ||
459 | m_log.WarnFormat("[PHYSICS Ray]: Unable to collide test an object: {0}", e.Message); | ||
460 | return; | ||
461 | } | ||
462 | |||
463 | if (count == 0) | ||
464 | return; | ||
465 | |||
466 | uint ID = 0; | ||
467 | PhysicsActor p2 = null; | ||
468 | |||
469 | m_scene.actor_name_map.TryGetValue(g2, out p2); | ||
470 | |||
471 | if (p2 == null) | ||
472 | { | ||
473 | /* | ||
474 | string name; | ||
475 | |||
476 | if (!m_scene.geom_name_map.TryGetValue(g2, out name)) | ||
477 | return; | ||
478 | |||
479 | if (name == "Terrain") | ||
480 | { | ||
481 | // land colision | ||
482 | if ((CurrentRayFilter & RayFilterFlags.land) == 0) | ||
483 | return; | ||
484 | } | ||
485 | else if (name == "Water") | ||
486 | { | ||
487 | if ((CurrentRayFilter & RayFilterFlags.water) == 0) | ||
488 | return; | ||
489 | } | ||
490 | else | ||
491 | return; | ||
492 | */ | ||
493 | return; | ||
494 | } | ||
495 | else | ||
496 | { | ||
497 | switch (p2.PhysicsActorType) | ||
498 | { | ||
499 | case (int)ActorTypes.Prim: | ||
500 | |||
501 | RayFilterFlags thisFlags; | ||
502 | |||
503 | if (p2.IsPhysical) | ||
504 | thisFlags = RayFilterFlags.physical; | ||
505 | else | ||
506 | thisFlags = RayFilterFlags.nonphysical; | ||
507 | |||
508 | if (p2.Phantom) | ||
509 | thisFlags |= RayFilterFlags.phantom; | ||
510 | |||
511 | if (p2.IsVolumeDtc) | ||
512 | thisFlags |= RayFilterFlags.volumedtc; | ||
513 | |||
514 | if ((thisFlags & CurrentRayFilter) == 0) | ||
515 | return; | ||
516 | |||
517 | ID = ((OdePrim)p2).LocalID; | ||
518 | break; | ||
519 | |||
520 | case (int)ActorTypes.Agent: | ||
521 | |||
522 | if ((CurrentRayFilter & RayFilterFlags.agent) == 0) | ||
523 | return; | ||
524 | else | ||
525 | ID = ((OdeCharacter)p2).LocalID; | ||
526 | break; | ||
527 | |||
528 | case (int)ActorTypes.Ground: | ||
529 | |||
530 | if ((CurrentRayFilter & RayFilterFlags.land) == 0) | ||
531 | return; | ||
532 | break; | ||
533 | |||
534 | case (int)ActorTypes.Water: | ||
535 | |||
536 | if ((CurrentRayFilter & RayFilterFlags.water) == 0) | ||
537 | return; | ||
538 | break; | ||
539 | |||
540 | default: | ||
541 | return; | ||
542 | break; | ||
543 | } | ||
544 | } | ||
545 | |||
546 | d.ContactGeom curcontact = new d.ContactGeom(); | ||
547 | |||
548 | // closestHit for now only works for meshs, so must do it for others | ||
549 | if ((CurrentRayFilter & RayFilterFlags.ClosestHit) == 0) | ||
550 | { | ||
551 | // Loop all contacts, build results. | ||
552 | for (int i = 0; i < count; i++) | ||
553 | { | ||
554 | if (!GetCurContactGeom(i, ref curcontact)) | ||
555 | break; | ||
556 | |||
557 | ContactResult collisionresult = new ContactResult(); | ||
558 | collisionresult.ConsumerID = ID; | ||
559 | collisionresult.Pos = new Vector3(curcontact.pos.X, curcontact.pos.Y, curcontact.pos.Z); | ||
560 | collisionresult.Depth = curcontact.depth; | ||
561 | collisionresult.Normal = new Vector3(curcontact.normal.X, curcontact.normal.Y, | ||
562 | curcontact.normal.Z); | ||
563 | lock (m_contactResults) | ||
564 | { | ||
565 | m_contactResults.Add(collisionresult); | ||
566 | if (m_contactResults.Count >= CurrentMaxCount) | ||
567 | return; | ||
568 | } | ||
569 | } | ||
570 | } | ||
571 | else | ||
572 | { | ||
573 | // keep only closest contact | ||
574 | ContactResult collisionresult = new ContactResult(); | ||
575 | collisionresult.ConsumerID = ID; | ||
576 | collisionresult.Depth = float.MaxValue; | ||
577 | |||
578 | for (int i = 0; i < count; i++) | ||
579 | { | ||
580 | if (!GetCurContactGeom(i, ref curcontact)) | ||
581 | break; | ||
582 | |||
583 | if (curcontact.depth < collisionresult.Depth) | ||
584 | { | ||
585 | collisionresult.Pos = new Vector3(curcontact.pos.X, curcontact.pos.Y, curcontact.pos.Z); | ||
586 | collisionresult.Depth = curcontact.depth; | ||
587 | collisionresult.Normal = new Vector3(curcontact.normal.X, curcontact.normal.Y, | ||
588 | curcontact.normal.Z); | ||
589 | } | ||
590 | } | ||
591 | |||
592 | if (collisionresult.Depth != float.MaxValue) | ||
593 | { | ||
594 | lock (m_contactResults) | ||
595 | m_contactResults.Add(collisionresult); | ||
596 | } | ||
597 | } | ||
598 | } | ||
599 | |||
600 | /// <summary> | ||
601 | /// Dereference the creator scene so that it can be garbage collected if needed. | ||
602 | /// </summary> | ||
603 | internal void Dispose() | ||
604 | { | ||
605 | m_scene = null; | ||
606 | if (ray != IntPtr.Zero) | ||
607 | { | ||
608 | d.GeomDestroy(ray); | ||
609 | ray = IntPtr.Zero; | ||
610 | } | ||
611 | } | ||
612 | } | ||
613 | |||
614 | public struct ODERayRequest | ||
615 | { | ||
616 | public IntPtr geom; | ||
617 | public Vector3 Origin; | ||
618 | public Vector3 Normal; | ||
619 | public int Count; | ||
620 | public float length; | ||
621 | public object callbackMethod; | ||
622 | public RayFilterFlags filter; | ||
623 | } | ||
624 | } \ No newline at end of file | ||