aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/OpenSim/Region/Physics/UbitMeshing/Meshmerizer.cs
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--OpenSim/Region/Physics/UbitMeshing/Meshmerizer.cs321
1 files changed, 294 insertions, 27 deletions
diff --git a/OpenSim/Region/Physics/UbitMeshing/Meshmerizer.cs b/OpenSim/Region/Physics/UbitMeshing/Meshmerizer.cs
index c9c52c0..7667e91 100644
--- a/OpenSim/Region/Physics/UbitMeshing/Meshmerizer.cs
+++ b/OpenSim/Region/Physics/UbitMeshing/Meshmerizer.cs
@@ -41,6 +41,7 @@ using Nini.Config;
41using System.Reflection; 41using System.Reflection;
42using System.IO; 42using System.IO;
43using ComponentAce.Compression.Libs.zlib; 43using ComponentAce.Compression.Libs.zlib;
44using OpenSim.Region.Physics.ConvexDecompositionDotNet;
44 45
45namespace OpenSim.Region.Physics.Meshing 46namespace OpenSim.Region.Physics.Meshing
46{ 47{
@@ -256,7 +257,7 @@ namespace OpenSim.Region.Physics.Meshing
256 /// <param name="size"></param> 257 /// <param name="size"></param>
257 /// <param name="lod"></param> 258 /// <param name="lod"></param>
258 /// <returns></returns> 259 /// <returns></returns>
259 private Mesh CreateMeshFromPrimMesher(string primName, PrimitiveBaseShape primShape, Vector3 size, float lod) 260 private Mesh CreateMeshFromPrimMesher(string primName, PrimitiveBaseShape primShape, Vector3 size, float lod, bool convex)
260 { 261 {
261// m_log.DebugFormat( 262// m_log.DebugFormat(
262// "[MESH]: Creating physics proxy for {0}, shape {1}", 263// "[MESH]: Creating physics proxy for {0}, shape {1}",
@@ -272,7 +273,7 @@ namespace OpenSim.Region.Physics.Meshing
272 if (!useMeshiesPhysicsMesh) 273 if (!useMeshiesPhysicsMesh)
273 return null; 274 return null;
274 275
275 if (!GenerateCoordsAndFacesFromPrimMeshData(primName, primShape, size, out coords, out faces)) 276 if (!GenerateCoordsAndFacesFromPrimMeshData(primName, primShape, size, out coords, out faces, convex))
276 return null; 277 return null;
277 } 278 }
278 else 279 else
@@ -296,22 +297,20 @@ namespace OpenSim.Region.Physics.Meshing
296 int numCoords = coords.Count; 297 int numCoords = coords.Count;
297 int numFaces = faces.Count; 298 int numFaces = faces.Count;
298 299
299 // Create the list of vertices
300 List<Vertex> vertices = new List<Vertex>();
301 for (int i = 0; i < numCoords; i++)
302 {
303 Coord c = coords[i];
304 vertices.Add(new Vertex(c.X, c.Y, c.Z));
305 }
306
307 Mesh mesh = new Mesh(); 300 Mesh mesh = new Mesh();
308 // Add the corresponding triangles to the mesh 301 // Add the corresponding triangles to the mesh
309 for (int i = 0; i < numFaces; i++) 302 for (int i = 0; i < numFaces; i++)
310 { 303 {
311 Face f = faces[i]; 304 Face f = faces[i];
312 mesh.Add(new Triangle(vertices[f.v1], vertices[f.v2], vertices[f.v3])); 305 mesh.Add(new Triangle(coords[f.v1].X, coords[f.v1].Y, coords[f.v1].Z,
306 coords[f.v2].X, coords[f.v2].Y, coords[f.v2].Z,
307 coords[f.v3].X, coords[f.v3].Y, coords[f.v3].Z));
313 } 308 }
314 309
310
311// mesh.DumpRaw("c:\\lixo", "lixo", "lixo");
312 mesh.DumpRaw(".", "lixo", "lixo");
313
315 return mesh; 314 return mesh;
316 } 315 }
317 316
@@ -325,10 +324,12 @@ namespace OpenSim.Region.Physics.Meshing
325 /// <param name="faces">Faces are added to this list by the method.</param> 324 /// <param name="faces">Faces are added to this list by the method.</param>
326 /// <returns>true if coords and faces were successfully generated, false if not</returns> 325 /// <returns>true if coords and faces were successfully generated, false if not</returns>
327 private bool GenerateCoordsAndFacesFromPrimMeshData( 326 private bool GenerateCoordsAndFacesFromPrimMeshData(
328 string primName, PrimitiveBaseShape primShape, Vector3 size, out List<Coord> coords, out List<Face> faces) 327 string primName, PrimitiveBaseShape primShape, Vector3 size, out List<Coord> coords, out List<Face> faces, bool convex)
329 { 328 {
330// m_log.DebugFormat("[MESH]: experimental mesh proxy generation for {0}", primName); 329// m_log.DebugFormat("[MESH]: experimental mesh proxy generation for {0}", primName);
331 330
331 bool usemesh = false;
332
332 coords = new List<Coord>(); 333 coords = new List<Coord>();
333 faces = new List<Face>(); 334 faces = new List<Face>();
334 OSD meshOsd = null; 335 OSD meshOsd = null;
@@ -365,14 +366,25 @@ namespace OpenSim.Region.Physics.Meshing
365 { 366 {
366 OSDMap physicsParms = null; 367 OSDMap physicsParms = null;
367 OSDMap map = (OSDMap)meshOsd; 368 OSDMap map = (OSDMap)meshOsd;
368 if (map.ContainsKey("physics_shape")) 369
369 physicsParms = (OSDMap)map["physics_shape"]; // old asset format 370 if (!convex)
370 else if (map.ContainsKey("physics_mesh")) 371 {
371 physicsParms = (OSDMap)map["physics_mesh"]; // new asset format 372 if (map.ContainsKey("physics_shape"))
373 physicsParms = (OSDMap)map["physics_shape"]; // old asset format
374 else if (map.ContainsKey("physics_mesh"))
375 physicsParms = (OSDMap)map["physics_mesh"]; // new asset format
376
377 if (physicsParms != null)
378 usemesh = true;
379 }
380
381 if(!usemesh && (map.ContainsKey("physics_convex")))
382 physicsParms = (OSDMap)map["physics_convex"];
383
372 384
373 if (physicsParms == null) 385 if (physicsParms == null)
374 { 386 {
375 m_log.Warn("[MESH]: no recognized physics mesh found in mesh asset"); 387 m_log.Warn("[MESH]: unknown mesh type");
376 return false; 388 return false;
377 } 389 }
378 390
@@ -416,19 +428,269 @@ namespace OpenSim.Region.Physics.Meshing
416 return false; 428 return false;
417 } 429 }
418 430
419 OSDArray decodedMeshOsdArray = null; 431 if (usemesh)
432 {
433 OSDArray decodedMeshOsdArray = null;
420 434
421 // physics_shape is an array of OSDMaps, one for each submesh 435 // physics_shape is an array of OSDMaps, one for each submesh
422 if (decodedMeshOsd is OSDArray) 436 if (decodedMeshOsd is OSDArray)
437 {
438 // Console.WriteLine("decodedMeshOsd for {0} - {1}", primName, Util.GetFormattedXml(decodedMeshOsd));
439
440 decodedMeshOsdArray = (OSDArray)decodedMeshOsd;
441 foreach (OSD subMeshOsd in decodedMeshOsdArray)
442 {
443 if (subMeshOsd is OSDMap)
444 AddSubMesh(subMeshOsd as OSDMap, size, coords, faces);
445 }
446 }
447 }
448 else
423 { 449 {
424// Console.WriteLine("decodedMeshOsd for {0} - {1}", primName, Util.GetFormattedXml(decodedMeshOsd)); 450 OSDMap cmap = (OSDMap)decodedMeshOsd;
451 if (cmap == null)
452 return false;
453
454 byte[] data;
455 const float invMaxU16 = 1.0f / 65535f;
456 int t1;
457 int t2;
458 int t3;
459 int i;
460
461 List<float3> vs = new List<float3>();
462
463 float3 f3;
464 PHullResult hullr = new PHullResult();
465
466 Coord c;
467 Face f;
468
469 Vector3 range;
470 Vector3 min;
471 int nverts;
472 int nindexs;
473
474 if (cmap.ContainsKey("Max"))
475 range = cmap["Max"].AsVector3();
476 else
477 range = new Vector3(0.5f, 0.5f, 0.5f);
478
479 if (cmap.ContainsKey("Min"))
480 min = cmap["Min"].AsVector3();
481 else
482 min = new Vector3(-0.5f, -0.5f, -0.5f);
483
484 range = range - min;
485 range *= invMaxU16;
486
487 if (!convex && cmap.ContainsKey("HullList") && cmap.ContainsKey("Positions"))
488 {
489 List<int> hsizes = new List<int>();
490 int totalpoints = 0;
491 data = cmap["HullList"].AsBinary();
492 for (i = 0; i < data.Length; i++)
493 {
494 t1 = data[i];
495 if (t1 == 0)
496 t1 = 256;
497 totalpoints += t1;
498 hsizes.Add(t1);
499 }
500
501 data = cmap["Positions"].AsBinary();
502 int ptr = 0;
503 int vertsoffset = 0;
504
505 if (totalpoints == data.Length / 6) // 2 bytes per coord, 3 coords per point
506 {
507 foreach (int hullsize in hsizes)
508 {
509 for (i = 0; i < hullsize; i++ )
510 {
511 t1 = data[ptr++];
512 t1 += data[ptr++] << 8;
513 t2 = data[ptr++];
514 t2 += data[ptr++] << 8;
515 t3 = data[ptr++];
516 t3 += data[ptr++] << 8;
517
518 f3 = new float3((t1 * range.X + min.X) * size.X,
519 (t2 * range.Y + min.Y) * size.Y,
520 (t3 * range.Z + min.Z) * size.Z);
521 vs.Add(f3);
522 }
523
524 if(hullsize <3)
525 {
526 vs.Clear();
527 continue;
528 }
529
530 if (hullsize <5)
531 {
532 foreach (float3 point in vs)
533 {
534 c.X = point.x;
535 c.Y = point.y;
536 c.Z = point.z;
537 coords.Add(c);
538 }
539 f = new Face(vertsoffset, vertsoffset + 1, vertsoffset + 2);
540 faces.Add(f);
541
542 if (hullsize == 4)
543 {
544 // not sure about orientation..
545 f = new Face(vertsoffset, vertsoffset + 2, vertsoffset + 3);
546 faces.Add(f);
547 f = new Face(vertsoffset, vertsoffset + 3, vertsoffset + 1);
548 faces.Add(f);
549 f = new Face(vertsoffset + 3, vertsoffset + 2, vertsoffset + 1);
550 faces.Add(f);
551 }
552 vertsoffset += vs.Count;
553 vs.Clear();
554 continue;
555 }
425 556
426 decodedMeshOsdArray = (OSDArray)decodedMeshOsd; 557 if (!HullUtils.ComputeHull(vs, ref hullr, 0, 0.0f))
427 foreach (OSD subMeshOsd in decodedMeshOsdArray) 558 {
559 vs.Clear();
560 continue;
561 }
562
563 nverts = hullr.Vertices.Count;
564 nindexs = hullr.Indices.Count;
565
566 if (nindexs % 3 != 0)
567 {
568 vs.Clear();
569 continue;
570 }
571
572 for (i = 0; i < nverts; i++)
573 {
574 c.X = hullr.Vertices[i].x;
575 c.Y = hullr.Vertices[i].y;
576 c.Z = hullr.Vertices[i].z;
577 coords.Add(c);
578 }
579
580
581 for (i = 0; i < nindexs; i += 3)
582 {
583 t1 = hullr.Indices[i];
584 if (t1 > nverts)
585 break;
586 t2 = hullr.Indices[i + 1];
587 if (t2 > nverts)
588 break;
589 t3 = hullr.Indices[i + 2];
590 if (t3 > nverts)
591 break;
592 f = new Face(vertsoffset + t1, vertsoffset + t2, vertsoffset + t3);
593 faces.Add(f);
594 }
595 vertsoffset += nverts;
596 vs.Clear();
597 }
598 }
599 if (coords.Count > 0 && faces.Count > 0)
600 return true;
601
602 }
603
604 vs.Clear();
605
606 if (cmap.ContainsKey("BoundingVerts"))
428 { 607 {
429 if (subMeshOsd is OSDMap) 608 data = cmap["BoundingVerts"].AsBinary();
430 AddSubMesh(subMeshOsd as OSDMap, size, coords, faces); 609
610 for (i = 0; i < data.Length; )
611 {
612 t1 = data[i++];
613 t1 += data[i++] << 8;
614 t2 = data[i++];
615 t2 += data[i++] << 8;
616 t3 = data[i++];
617 t3 += data[i++] << 8;
618
619 f3 = new float3((t1 * range.X + min.X) * size.X,
620 (t2 * range.Y + min.Y) * size.Y,
621 (t3 * range.Z + min.Z) * size.Z);
622 vs.Add(f3);
623 }
624
625 if (vs.Count < 3)
626 {
627 vs.Clear();
628 return false;
629 }
630
631 if (vs.Count < 5)
632 {
633 foreach (float3 point in vs)
634 {
635 c.X = point.x;
636 c.Y = point.y;
637 c.Z = point.z;
638 coords.Add(c);
639 }
640 f = new Face(0, 1, 2);
641 faces.Add(f);
642
643 if (vs.Count == 4)
644 {
645 // not sure about orientation..
646 f = new Face(0, 2, 3);
647 faces.Add(f);
648 f = new Face(0, 3, 1);
649 faces.Add(f);
650 f = new Face( 3, 2, 1);
651 faces.Add(f);
652 }
653 vs.Clear();
654 return true;
655 }
656
657 if (!HullUtils.ComputeHull(vs, ref hullr, 0, 0.0f))
658 return false;
659
660 nverts = hullr.Vertices.Count;
661 nindexs = hullr.Indices.Count;
662
663 if (nindexs % 3 != 0)
664 return false;
665
666 for (i = 0; i < nverts; i++)
667 {
668 c.X = hullr.Vertices[i].x;
669 c.Y = hullr.Vertices[i].y;
670 c.Z = hullr.Vertices[i].z;
671 coords.Add(c);
672 }
673 for (i = 0; i < nindexs; i += 3)
674 {
675 t1 = hullr.Indices[i];
676 if (t1 > nverts)
677 break;
678 t2 = hullr.Indices[i + 1];
679 if (t2 > nverts)
680 break;
681 t3 = hullr.Indices[i + 2];
682 if (t3 > nverts)
683 break;
684 f = new Face(t1, t2, t3);
685 faces.Add(f);
686 }
687
688 if (coords.Count > 0 && faces.Count > 0)
689 return true;
431 } 690 }
691 else
692 return false;
693
432 } 694 }
433 } 695 }
434 696
@@ -714,11 +976,16 @@ namespace OpenSim.Region.Physics.Meshing
714 976
715 public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod) 977 public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod)
716 { 978 {
717 return CreateMesh(primName, primShape, size, lod, false); 979 return CreateMesh(primName, primShape, size, lod, false,false);
718 } 980 }
719 981
720 public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod, bool isPhysical) 982 public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod, bool isPhysical)
721 { 983 {
984 return CreateMesh(primName, primShape, size, lod, false,false);
985 }
986
987 public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod, bool isPhysical, bool convex)
988 {
722#if SPAM 989#if SPAM
723 m_log.DebugFormat("[MESH]: Creating mesh for {0}", primName); 990 m_log.DebugFormat("[MESH]: Creating mesh for {0}", primName);
724#endif 991#endif
@@ -736,7 +1003,7 @@ namespace OpenSim.Region.Physics.Meshing
736 if (size.Y < 0.01f) size.Y = 0.01f; 1003 if (size.Y < 0.01f) size.Y = 0.01f;
737 if (size.Z < 0.01f) size.Z = 0.01f; 1004 if (size.Z < 0.01f) size.Z = 0.01f;
738 1005
739 mesh = CreateMeshFromPrimMesher(primName, primShape, size, lod); 1006 mesh = CreateMeshFromPrimMesher(primName, primShape, size, lod,convex);
740 1007
741 if (mesh != null) 1008 if (mesh != null)
742 { 1009 {