aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/OpenSim/Region/Physics
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--OpenSim/Region/Physics/OdePlugin/OdePlugin.cs188
1 files changed, 102 insertions, 86 deletions
diff --git a/OpenSim/Region/Physics/OdePlugin/OdePlugin.cs b/OpenSim/Region/Physics/OdePlugin/OdePlugin.cs
index 04ef1a3..3cc690d 100644
--- a/OpenSim/Region/Physics/OdePlugin/OdePlugin.cs
+++ b/OpenSim/Region/Physics/OdePlugin/OdePlugin.cs
@@ -84,6 +84,7 @@ namespace OpenSim.Region.Physics.OdePlugin
84 private List<OdePrim> _prims = new List<OdePrim>(); 84 private List<OdePrim> _prims = new List<OdePrim>();
85 private static d.ContactGeom[] contacts = new d.ContactGeom[30]; 85 private static d.ContactGeom[] contacts = new d.ContactGeom[30];
86 private static d.Contact contact; 86 private static d.Contact contact;
87 public static Object OdeLock = new Object();
87 public OdeScene() 88 public OdeScene()
88 { 89 {
89 contact.surface.mode |= d.ContactFlags.Approx1 | d.ContactFlags.SoftCFM | d.ContactFlags.SoftERP; 90 contact.surface.mode |= d.ContactFlags.Approx1 | d.ContactFlags.SoftCFM | d.ContactFlags.SoftERP;
@@ -92,14 +93,15 @@ namespace OpenSim.Region.Physics.OdePlugin
92 contact.surface.soft_erp = 0.005f; 93 contact.surface.soft_erp = 0.005f;
93 contact.surface.soft_cfm = 0.00003f; 94 contact.surface.soft_cfm = 0.00003f;
94 95
95 Monitor.Enter(typeof(OdeScene)); 96 lock (OdeLock)
96 world = d.WorldCreate(); 97 {
97 space = d.HashSpaceCreate(IntPtr.Zero); 98 world = d.WorldCreate();
98 contactgroup = d.JointGroupCreate(0); 99 space = d.HashSpaceCreate(IntPtr.Zero);
99 d.WorldSetGravity(world, 0.0f, 0.0f, -10.0f); 100 contactgroup = d.JointGroupCreate(0);
100 d.WorldSetAutoDisableFlag(world, false); 101 d.WorldSetGravity(world, 0.0f, 0.0f, -10.0f);
101 d.WorldSetContactSurfaceLayer(world, 0.001f); 102 d.WorldSetAutoDisableFlag(world, false);
102 Monitor.Exit(typeof(OdeScene)); 103 d.WorldSetContactSurfaceLayer(world, 0.001f);
104 }
103 105
104 this._heightmap = new double[65536]; 106 this._heightmap = new double[65536];
105 } 107 }
@@ -128,7 +130,7 @@ namespace OpenSim.Region.Physics.OdePlugin
128 PhysicsVector pos = new PhysicsVector(); 130 PhysicsVector pos = new PhysicsVector();
129 pos.X = position.X; 131 pos.X = position.X;
130 pos.Y = position.Y; 132 pos.Y = position.Y;
131 pos.Z = position.Z + 20; 133 pos.Z = position.Z;
132 OdeCharacter newAv = new OdeCharacter(this, pos); 134 OdeCharacter newAv = new OdeCharacter(this, pos);
133 this._characters.Add(newAv); 135 this._characters.Add(newAv);
134 return newAv; 136 return newAv;
@@ -136,7 +138,6 @@ namespace OpenSim.Region.Physics.OdePlugin
136 138
137 public override void RemoveAvatar(PhysicsActor actor) 139 public override void RemoveAvatar(PhysicsActor actor)
138 { 140 {
139
140 } 141 }
141 142
142 public override void RemovePrim(PhysicsActor prim) 143 public override void RemovePrim(PhysicsActor prim)
@@ -150,7 +151,6 @@ namespace OpenSim.Region.Physics.OdePlugin
150 151
151 public override PhysicsActor AddPrim(PhysicsVector position, PhysicsVector size, Quaternion rotation) 152 public override PhysicsActor AddPrim(PhysicsVector position, PhysicsVector size, Quaternion rotation)
152 { 153 {
153 //Console.WriteLine("+++++++++++++++++++++++++++++++++AddPrim pos: " + position + " size: " + size + " quat: " + rotation);
154 PhysicsVector pos = new PhysicsVector(); 154 PhysicsVector pos = new PhysicsVector();
155 pos.X = position.X; 155 pos.X = position.X;
156 pos.Y = position.Y; 156 pos.Y = position.Y;
@@ -165,7 +165,8 @@ namespace OpenSim.Region.Physics.OdePlugin
165 rot.y = rotation.y; 165 rot.y = rotation.y;
166 rot.z = rotation.z; 166 rot.z = rotation.z;
167 OdePrim newPrim; 167 OdePrim newPrim;
168 lock(typeof(OdeScene)) { 168 lock (OdeLock)
169 {
169 newPrim = new OdePrim(this, pos, siz, rot); 170 newPrim = new OdePrim(this, pos, siz, rot);
170 } 171 }
171 this._prims.Add(newPrim); 172 this._prims.Add(newPrim);
@@ -174,27 +175,27 @@ namespace OpenSim.Region.Physics.OdePlugin
174 175
175 public override void Simulate(float timeStep) 176 public override void Simulate(float timeStep)
176 { 177 {
177 Monitor.Enter(typeof(OdeScene)); 178 lock (OdeLock)
178 foreach (OdePrim p in _prims)
179 {
180// Console.WriteLine("+++ prim: " + p.Position);
181 }
182 foreach (OdeCharacter actor in _characters)
183 {
184 actor.Move(timeStep);
185 }
186 d.SpaceCollide(space, IntPtr.Zero, nearCallback);
187 for (int i = 0; i < 50; i++)
188 { 179 {
189 d.WorldQuickStep(world, timeStep * 0.02f); 180 foreach (OdePrim p in _prims)
190 } 181 {
182 }
183 foreach (OdeCharacter actor in _characters)
184 {
185 actor.Move(timeStep);
186 }
187 d.SpaceCollide(space, IntPtr.Zero, nearCallback);
188 for (int i = 0; i < 50; i++)
189 {
190 d.WorldQuickStep(world, timeStep * 0.02f);
191 }
191 192
192 d.JointGroupEmpty(contactgroup); 193 d.JointGroupEmpty(contactgroup);
193 foreach (OdeCharacter actor in _characters) 194 foreach (OdeCharacter actor in _characters)
194 { 195 {
195 actor.UpdatePosition(); 196 actor.UpdatePosition();
197 }
196 } 198 }
197 Monitor.Exit(typeof(OdeScene));
198 } 199 }
199 200
200 public override void GetResults() 201 public override void GetResults()
@@ -222,26 +223,27 @@ namespace OpenSim.Region.Physics.OdePlugin
222 } 223 }
223 IntPtr HeightmapData = d.GeomHeightfieldDataCreate(); 224 IntPtr HeightmapData = d.GeomHeightfieldDataCreate();
224 225
225 Monitor.Enter(typeof(OdeScene)); 226 lock (OdeLock)
226 d.GeomHeightfieldDataBuildDouble(HeightmapData, _heightmap, 0, 256, 256, 256, 256, 1.0f, 0.0f, 2.0f, 0); 227 {
227 d.GeomHeightfieldDataSetBounds(HeightmapData, 256, 256); 228 d.GeomHeightfieldDataBuildDouble(HeightmapData, _heightmap, 0, 256, 256, 256, 256, 1.0f, 0.0f, 2.0f, 0);
228 LandGeom = d.CreateHeightfield(space, HeightmapData, 1); 229 d.GeomHeightfieldDataSetBounds(HeightmapData, 256, 256);
229 d.Matrix3 R = new d.Matrix3(); 230 LandGeom = d.CreateHeightfield(space, HeightmapData, 1);
230 231 d.Matrix3 R = new d.Matrix3();
231 Quaternion q1 =Quaternion.FromAngleAxis(1.5707f, new Vector3(1,0,0)); 232
232 Quaternion q2 =Quaternion.FromAngleAxis(1.5707f, new Vector3(0,1,0)); 233 Quaternion q1 = Quaternion.FromAngleAxis(1.5707f, new Vector3(1, 0, 0));
233 //Axiom.Math.Quaternion q3 = Axiom.Math.Quaternion.FromAngleAxis(3.14f, new Axiom.Math.Vector3(0, 0, 1)); 234 Quaternion q2 = Quaternion.FromAngleAxis(1.5707f, new Vector3(0, 1, 0));
234 235 //Axiom.Math.Quaternion q3 = Axiom.Math.Quaternion.FromAngleAxis(3.14f, new Axiom.Math.Vector3(0, 0, 1));
235 q1 = q1 * q2; 236
236 //q1 = q1 * q3; 237 q1 = q1 * q2;
237 Vector3 v3 = new Vector3(); 238 //q1 = q1 * q3;
238 float angle = 0; 239 Vector3 v3 = new Vector3();
239 q1.ToAngleAxis(ref angle, ref v3); 240 float angle = 0;
240 241 q1.ToAngleAxis(ref angle, ref v3);
241 d.RFromAxisAndAngle(out R, v3.x, v3.y, v3.z, angle); 242
242 d.GeomSetRotation(LandGeom, ref R); 243 d.RFromAxisAndAngle(out R, v3.x, v3.y, v3.z, angle);
243 d.GeomSetPosition(LandGeom, 128, 128, 0); 244 d.GeomSetRotation(LandGeom, ref R);
244 Monitor.Exit(typeof(OdeScene)); 245 d.GeomSetPosition(LandGeom, 128, 128, 0);
246 }
245 } 247 }
246 248
247 public override void DeleteTerrain() 249 public override void DeleteTerrain()
@@ -260,20 +262,23 @@ namespace OpenSim.Region.Physics.OdePlugin
260 private IntPtr BoundingCapsule; 262 private IntPtr BoundingCapsule;
261 IntPtr capsule_geom; 263 IntPtr capsule_geom;
262 d.Mass capsule_mass; 264 d.Mass capsule_mass;
265 private OdeScene _parent_scene;
263 266
264 public OdeCharacter(OdeScene parent_scene, PhysicsVector pos) 267 public OdeCharacter(OdeScene parent_scene, PhysicsVector pos)
265 { 268 {
266 _velocity = new PhysicsVector(); 269 _velocity = new PhysicsVector();
267 _position = pos; 270 _position = pos;
268 _acceleration = new PhysicsVector(); 271 _acceleration = new PhysicsVector();
269 Monitor.Enter(typeof(OdeScene)); 272 _parent_scene = parent_scene;
270 d.MassSetCapsule(out capsule_mass, 50.0f, 3, 0.5f, 2f); 273 lock (OdeScene.OdeLock)
271 capsule_geom = d.CreateSphere(OdeScene.space, 1.0f); /// not a typo! Spheres roll, capsules tumble 274 {
272 this.BoundingCapsule = d.BodyCreate(OdeScene.world); 275 d.MassSetCapsule(out capsule_mass, 50.0f, 3, 0.5f, 2f);
273 d.BodySetMass(BoundingCapsule, ref capsule_mass); 276 capsule_geom = d.CreateSphere(OdeScene.space, 1.0f); /// not a typo! Spheres roll, capsules tumble
274 d.BodySetPosition(BoundingCapsule, pos.X, pos.Y, pos.Z); 277 this.BoundingCapsule = d.BodyCreate(OdeScene.world);
275 d.GeomSetBody(capsule_geom, BoundingCapsule); 278 d.BodySetMass(BoundingCapsule, ref capsule_mass);
276 Monitor.Exit(typeof(OdeScene)); 279 d.BodySetPosition(BoundingCapsule, pos.X, pos.Y, pos.Z);
280 d.GeomSetBody(capsule_geom, BoundingCapsule);
281 }
277 } 282 }
278 283
279 public override bool Flying 284 public override bool Flying
@@ -296,7 +301,11 @@ namespace OpenSim.Region.Physics.OdePlugin
296 } 301 }
297 set 302 set
298 { 303 {
299 _position = value; 304 lock (OdeScene.OdeLock)
305 {
306 d.BodySetPosition(BoundingCapsule, value.X, value.Y, value.Z);
307 _position = value;
308 }
300 } 309 }
301 } 310 }
302 311
@@ -389,6 +398,13 @@ namespace OpenSim.Region.Physics.OdePlugin
389 { 398 {
390 // no lock; called from Simulate() -- if you call this from elsewhere, gotta lock or do Monitor.Enter/Exit! 399 // no lock; called from Simulate() -- if you call this from elsewhere, gotta lock or do Monitor.Enter/Exit!
391 d.Vector3 vec = d.BodyGetPosition(BoundingCapsule); 400 d.Vector3 vec = d.BodyGetPosition(BoundingCapsule);
401
402 // kluge to keep things in bounds. ODE lets dead avatars drift away (they should be removed!)
403 if (vec.X < 0.0f) vec.X = 0.0f;
404 if (vec.Y < 0.0f) vec.Y = 0.0f;
405 if (vec.X > 255.95f) vec.X = 255.95f;
406 if (vec.Y > 255.95f) vec.Y = 255.95f;
407
392 this._position.X = vec.X; 408 this._position.X = vec.X;
393 this._position.Y = vec.Y; 409 this._position.Y = vec.Y;
394 this._position.Z = vec.Z; 410 this._position.Z = vec.Z;
@@ -412,15 +428,16 @@ namespace OpenSim.Region.Physics.OdePlugin
412 _acceleration = new PhysicsVector(); 428 _acceleration = new PhysicsVector();
413 _orientation = rotation; 429 _orientation = rotation;
414 prim_geom = d.CreateBox(OdeScene.space, _size.X, _size.Y, _size.Z); 430 prim_geom = d.CreateBox(OdeScene.space, _size.X, _size.Y, _size.Z);
415 Monitor.Enter(typeof(OdeScene)); 431 lock (OdeScene.OdeLock)
416 d.GeomSetPosition(prim_geom, _position.X, _position.Y, _position.Z); 432 {
417 d.Quaternion myrot = new d.Quaternion(); 433 d.GeomSetPosition(prim_geom, _position.X, _position.Y, _position.Z);
418 myrot.W = rotation.w; 434 d.Quaternion myrot = new d.Quaternion();
419 myrot.X = rotation.x; 435 myrot.W = rotation.w;
420 myrot.Y = rotation.y; 436 myrot.X = rotation.x;
421 myrot.Z = rotation.z; 437 myrot.Y = rotation.y;
422 d.GeomSetQuaternion(prim_geom, ref myrot); 438 myrot.Z = rotation.z;
423 Monitor.Exit(typeof(OdeScene)); 439 d.GeomSetQuaternion(prim_geom, ref myrot);
440 }
424 } 441 }
425 442
426 public override bool Flying 443 public override bool Flying
@@ -442,12 +459,11 @@ namespace OpenSim.Region.Physics.OdePlugin
442 } 459 }
443 set 460 set
444 { 461 {
445 //Console.WriteLine("+++++++++++++++++++++++++++++++++++++++++++ setting pos: " + value);
446 _position = value; 462 _position = value;
447 Monitor.Enter(typeof(OdeScene)); 463 lock (OdeScene.OdeLock)
448 d.GeomSetPosition(prim_geom, _position.X, _position.Y, _position.Z); 464 {
449 Monitor.Exit(typeof(OdeScene)); 465 d.GeomSetPosition(prim_geom, _position.X, _position.Y, _position.Z);
450 466 }
451 } 467 }
452 } 468 }
453 469
@@ -459,11 +475,11 @@ namespace OpenSim.Region.Physics.OdePlugin
459 } 475 }
460 set 476 set
461 { 477 {
462 //Console.WriteLine("+++++++++++++++++++++++++++++++++++++++++++ setting size: " + value);
463 _size = value; 478 _size = value;
464 Monitor.Enter(typeof(OdeScene)); 479 lock (OdeScene.OdeLock)
465 d.GeomBoxSetLengths(prim_geom, _size.X, _size.Y, _size.Z); 480 {
466 Monitor.Exit(typeof(OdeScene)); 481 d.GeomBoxSetLengths(prim_geom, _size.X, _size.Y, _size.Z);
482 }
467 } 483 }
468 } 484 }
469 485
@@ -498,16 +514,16 @@ namespace OpenSim.Region.Physics.OdePlugin
498 } 514 }
499 set 515 set
500 { 516 {
501 //Console.WriteLine("+++++++++++++++++++++++++++++++++++++++++++ setting Orientation");
502 _orientation = value; 517 _orientation = value;
503 Monitor.Enter(typeof(OdeScene)); 518 lock (OdeScene.OdeLock)
504 d.Quaternion myrot = new d.Quaternion(); 519 {
505 myrot.W = _orientation.w; 520 d.Quaternion myrot = new d.Quaternion();
506 myrot.X = _orientation.x; 521 myrot.W = _orientation.w;
507 myrot.Y = _orientation.y; 522 myrot.X = _orientation.x;
508 myrot.Z = _orientation.z; 523 myrot.Y = _orientation.y;
509 d.GeomSetQuaternion(prim_geom, ref myrot); 524 myrot.Z = _orientation.z;
510 Monitor.Exit(typeof(OdeScene)); 525 d.GeomSetQuaternion(prim_geom, ref myrot);
526 }
511 } 527 }
512 } 528 }
513 529