diff options
Diffstat (limited to 'OpenSim/Region/Physics/Meshing/Meshmerizer.cs')
-rw-r--r-- | OpenSim/Region/Physics/Meshing/Meshmerizer.cs | 460 |
1 files changed, 240 insertions, 220 deletions
diff --git a/OpenSim/Region/Physics/Meshing/Meshmerizer.cs b/OpenSim/Region/Physics/Meshing/Meshmerizer.cs index b478a9c..9dfaaa2 100644 --- a/OpenSim/Region/Physics/Meshing/Meshmerizer.cs +++ b/OpenSim/Region/Physics/Meshing/Meshmerizer.cs | |||
@@ -302,123 +302,6 @@ namespace OpenSim.Region.Physics.Meshing | |||
302 | } | 302 | } |
303 | 303 | ||
304 | /// <summary> | 304 | /// <summary> |
305 | /// Generate the co-ords and faces necessary to construct a mesh from the mesh data the accompanies a prim. | ||
306 | /// </summary> | ||
307 | /// <param name="primName"></param> | ||
308 | /// <param name="primShape"></param> | ||
309 | /// <param name="size"></param> | ||
310 | /// <param name="coords">Coords are added to this list by the method.</param> | ||
311 | /// <param name="faces">Faces are added to this list by the method.</param> | ||
312 | /// <returns>true if coords and faces were successfully generated, false if not</returns> | ||
313 | private bool GenerateCoordsAndFacesFromPrimMeshData(string primName, PrimitiveBaseShape primShape, Vector3 size, List<Coord> coords, List<Face> faces) | ||
314 | { | ||
315 | m_log.DebugFormat("[MESH]: experimental mesh proxy generation for {0}", primName); | ||
316 | |||
317 | OSD meshOsd = null; | ||
318 | |||
319 | if (primShape.SculptData.Length <= 0) | ||
320 | { | ||
321 | m_log.Error("[MESH]: asset data is zero length"); | ||
322 | return false; | ||
323 | } | ||
324 | |||
325 | long start = 0; | ||
326 | using (MemoryStream data = new MemoryStream(primShape.SculptData)) | ||
327 | { | ||
328 | try | ||
329 | { | ||
330 | OSD osd = OSDParser.DeserializeLLSDBinary(data); | ||
331 | if (osd is OSDMap) | ||
332 | meshOsd = (OSDMap)osd; | ||
333 | else | ||
334 | { | ||
335 | m_log.Warn("[Mesh}: unable to cast mesh asset to OSDMap"); | ||
336 | return false; | ||
337 | } | ||
338 | } | ||
339 | catch (Exception e) | ||
340 | { | ||
341 | m_log.Error("[MESH]: Exception deserializing mesh asset header:" + e.ToString()); | ||
342 | } | ||
343 | |||
344 | start = data.Position; | ||
345 | } | ||
346 | |||
347 | if (meshOsd is OSDMap) | ||
348 | { | ||
349 | OSDMap physicsParms = null; | ||
350 | OSDMap map = (OSDMap)meshOsd; | ||
351 | if (map.ContainsKey("physics_shape")) | ||
352 | physicsParms = (OSDMap)map["physics_shape"]; // old asset format | ||
353 | else if (map.ContainsKey("physics_mesh")) | ||
354 | physicsParms = (OSDMap)map["physics_mesh"]; // new asset format | ||
355 | |||
356 | if (physicsParms == null) | ||
357 | { | ||
358 | m_log.Warn("[MESH]: no recognized physics mesh found in mesh asset"); | ||
359 | return false; | ||
360 | } | ||
361 | |||
362 | int physOffset = physicsParms["offset"].AsInteger() + (int)start; | ||
363 | int physSize = physicsParms["size"].AsInteger(); | ||
364 | |||
365 | if (physOffset < 0 || physSize == 0) | ||
366 | return false; // no mesh data in asset | ||
367 | |||
368 | OSD decodedMeshOsd = new OSD(); | ||
369 | byte[] meshBytes = new byte[physSize]; | ||
370 | System.Buffer.BlockCopy(primShape.SculptData, physOffset, meshBytes, 0, physSize); | ||
371 | // byte[] decompressed = new byte[physSize * 5]; | ||
372 | try | ||
373 | { | ||
374 | using (MemoryStream inMs = new MemoryStream(meshBytes)) | ||
375 | { | ||
376 | using (MemoryStream outMs = new MemoryStream()) | ||
377 | { | ||
378 | using (ZOutputStream zOut = new ZOutputStream(outMs)) | ||
379 | { | ||
380 | byte[] readBuffer = new byte[2048]; | ||
381 | int readLen = 0; | ||
382 | while ((readLen = inMs.Read(readBuffer, 0, readBuffer.Length)) > 0) | ||
383 | { | ||
384 | zOut.Write(readBuffer, 0, readLen); | ||
385 | } | ||
386 | zOut.Flush(); | ||
387 | outMs.Seek(0, SeekOrigin.Begin); | ||
388 | |||
389 | byte[] decompressedBuf = outMs.GetBuffer(); | ||
390 | |||
391 | decodedMeshOsd = OSDParser.DeserializeLLSDBinary(decompressedBuf); | ||
392 | } | ||
393 | } | ||
394 | } | ||
395 | } | ||
396 | catch (Exception e) | ||
397 | { | ||
398 | m_log.Error("[MESH]: exception decoding physical mesh: " + e.ToString()); | ||
399 | return false; | ||
400 | } | ||
401 | |||
402 | OSDArray decodedMeshOsdArray = null; | ||
403 | |||
404 | // physics_shape is an array of OSDMaps, one for each submesh | ||
405 | if (decodedMeshOsd is OSDArray) | ||
406 | { | ||
407 | // Console.WriteLine("decodedMeshOsd for {0} - {1}", primName, Util.GetFormattedXml(decodedMeshOsd)); | ||
408 | |||
409 | decodedMeshOsdArray = (OSDArray)decodedMeshOsd; | ||
410 | foreach (OSD subMeshOsd in decodedMeshOsdArray) | ||
411 | { | ||
412 | if (subMeshOsd is OSDMap) | ||
413 | AddSubMesh(subMeshOsd as OSDMap, size, coords, faces); | ||
414 | } | ||
415 | } | ||
416 | } | ||
417 | |||
418 | return true; | ||
419 | } | ||
420 | |||
421 | /// <summary> | ||
422 | /// Create a physics mesh from data that comes with the prim. The actual data used depends on the prim type. | 305 | /// Create a physics mesh from data that comes with the prim. The actual data used depends on the prim type. |
423 | /// </summary> | 306 | /// </summary> |
424 | /// <param name="primName"></param> | 307 | /// <param name="primName"></param> |
@@ -433,14 +316,10 @@ namespace OpenSim.Region.Physics.Meshing | |||
433 | // primName, (OpenMetaverse.SculptType)primShape.SculptType); | 316 | // primName, (OpenMetaverse.SculptType)primShape.SculptType); |
434 | 317 | ||
435 | PrimMesh primMesh; | 318 | PrimMesh primMesh; |
436 | PrimMesher.SculptMesh sculptMesh; | ||
437 | 319 | ||
438 | List<Coord> coords = new List<Coord>(); | 320 | List<Coord> coords = new List<Coord>(); |
439 | List<Face> faces = new List<Face>(); | 321 | List<Face> faces = new List<Face>(); |
440 | 322 | ||
441 | Image idata = null; | ||
442 | string decodedSculptFileName = ""; | ||
443 | |||
444 | if (primShape.SculptEntry) | 323 | if (primShape.SculptEntry) |
445 | { | 324 | { |
446 | if (((OpenMetaverse.SculptType)primShape.SculptType) == SculptType.Mesh) | 325 | if (((OpenMetaverse.SculptType)primShape.SculptType) == SculptType.Mesh) |
@@ -448,109 +327,13 @@ namespace OpenSim.Region.Physics.Meshing | |||
448 | if (!useMeshiesPhysicsMesh) | 327 | if (!useMeshiesPhysicsMesh) |
449 | return null; | 328 | return null; |
450 | 329 | ||
451 | if (!GeneratePointsAndFacesFromPrimMeshData(primName, primShape, size, coords, faces)) | 330 | if (!GenerateCoordsAndFacesFromPrimMeshData(primName, primShape, size, coords, faces)) |
452 | return null; | 331 | return null; |
453 | } | 332 | } |
454 | else | 333 | else |
455 | { | 334 | { |
456 | if (cacheSculptMaps && primShape.SculptTexture != UUID.Zero) | 335 | if (!GenerateCoordsAndFacesFromPrimSculptData(primName, primShape, size, lod, coords, faces)) |
457 | { | 336 | return null; |
458 | decodedSculptFileName = System.IO.Path.Combine(decodedSculptMapPath, "smap_" + primShape.SculptTexture.ToString()); | ||
459 | try | ||
460 | { | ||
461 | if (File.Exists(decodedSculptFileName)) | ||
462 | { | ||
463 | idata = Image.FromFile(decodedSculptFileName); | ||
464 | } | ||
465 | } | ||
466 | catch (Exception e) | ||
467 | { | ||
468 | m_log.Error("[SCULPT]: unable to load cached sculpt map " + decodedSculptFileName + " " + e.Message); | ||
469 | |||
470 | } | ||
471 | //if (idata != null) | ||
472 | // m_log.Debug("[SCULPT]: loaded cached map asset for map ID: " + primShape.SculptTexture.ToString()); | ||
473 | } | ||
474 | |||
475 | if (idata == null) | ||
476 | { | ||
477 | if (primShape.SculptData == null || primShape.SculptData.Length == 0) | ||
478 | return null; | ||
479 | |||
480 | try | ||
481 | { | ||
482 | OpenMetaverse.Imaging.ManagedImage unusedData; | ||
483 | OpenMetaverse.Imaging.OpenJPEG.DecodeToImage(primShape.SculptData, out unusedData, out idata); | ||
484 | |||
485 | if (idata == null) | ||
486 | { | ||
487 | // In some cases it seems that the decode can return a null bitmap without throwing | ||
488 | // an exception | ||
489 | m_log.WarnFormat("[PHYSICS]: OpenJPEG decoded sculpt data for {0} to a null bitmap. Ignoring.", primName); | ||
490 | |||
491 | return null; | ||
492 | } | ||
493 | |||
494 | unusedData = null; | ||
495 | |||
496 | //idata = CSJ2K.J2kImage.FromBytes(primShape.SculptData); | ||
497 | |||
498 | if (cacheSculptMaps) | ||
499 | { | ||
500 | try { idata.Save(decodedSculptFileName, ImageFormat.MemoryBmp); } | ||
501 | catch (Exception e) { m_log.Error("[SCULPT]: unable to cache sculpt map " + decodedSculptFileName + " " + e.Message); } | ||
502 | } | ||
503 | } | ||
504 | catch (DllNotFoundException) | ||
505 | { | ||
506 | m_log.Error("[PHYSICS]: OpenJpeg is not installed correctly on this system. Physics Proxy generation failed. Often times this is because of an old version of GLIBC. You must have version 2.4 or above!"); | ||
507 | return null; | ||
508 | } | ||
509 | catch (IndexOutOfRangeException) | ||
510 | { | ||
511 | m_log.Error("[PHYSICS]: OpenJpeg was unable to decode this. Physics Proxy generation failed"); | ||
512 | return null; | ||
513 | } | ||
514 | catch (Exception ex) | ||
515 | { | ||
516 | m_log.Error("[PHYSICS]: Unable to generate a Sculpty physics proxy. Sculpty texture decode failed: " + ex.Message); | ||
517 | return null; | ||
518 | } | ||
519 | } | ||
520 | |||
521 | PrimMesher.SculptMesh.SculptType sculptType; | ||
522 | switch ((OpenMetaverse.SculptType)primShape.SculptType) | ||
523 | { | ||
524 | case OpenMetaverse.SculptType.Cylinder: | ||
525 | sculptType = PrimMesher.SculptMesh.SculptType.cylinder; | ||
526 | break; | ||
527 | case OpenMetaverse.SculptType.Plane: | ||
528 | sculptType = PrimMesher.SculptMesh.SculptType.plane; | ||
529 | break; | ||
530 | case OpenMetaverse.SculptType.Torus: | ||
531 | sculptType = PrimMesher.SculptMesh.SculptType.torus; | ||
532 | break; | ||
533 | case OpenMetaverse.SculptType.Sphere: | ||
534 | sculptType = PrimMesher.SculptMesh.SculptType.sphere; | ||
535 | break; | ||
536 | default: | ||
537 | sculptType = PrimMesher.SculptMesh.SculptType.plane; | ||
538 | break; | ||
539 | } | ||
540 | |||
541 | bool mirror = ((primShape.SculptType & 128) != 0); | ||
542 | bool invert = ((primShape.SculptType & 64) != 0); | ||
543 | |||
544 | sculptMesh = new PrimMesher.SculptMesh((Bitmap)idata, sculptType, (int)lod, false, mirror, invert); | ||
545 | |||
546 | idata.Dispose(); | ||
547 | |||
548 | sculptMesh.DumpRaw(baseDir, primName, "primMesh"); | ||
549 | |||
550 | sculptMesh.Scale(size.X, size.Y, size.Z); | ||
551 | |||
552 | coords = sculptMesh.coords; | ||
553 | faces = sculptMesh.faces; | ||
554 | } | 337 | } |
555 | } | 338 | } |
556 | else | 339 | else |
@@ -691,6 +474,243 @@ namespace OpenSim.Region.Physics.Meshing | |||
691 | return mesh; | 474 | return mesh; |
692 | } | 475 | } |
693 | 476 | ||
477 | /// <summary> | ||
478 | /// Generate the co-ords and faces necessary to construct a mesh from the mesh data the accompanies a prim. | ||
479 | /// </summary> | ||
480 | /// <param name="primName"></param> | ||
481 | /// <param name="primShape"></param> | ||
482 | /// <param name="size"></param> | ||
483 | /// <param name="coords">Coords are added to this list by the method.</param> | ||
484 | /// <param name="faces">Faces are added to this list by the method.</param> | ||
485 | /// <returns>true if coords and faces were successfully generated, false if not</returns> | ||
486 | private bool GenerateCoordsAndFacesFromPrimMeshData( | ||
487 | string primName, PrimitiveBaseShape primShape, Vector3 size, List<Coord> coords, List<Face> faces) | ||
488 | { | ||
489 | m_log.DebugFormat("[MESH]: experimental mesh proxy generation for {0}", primName); | ||
490 | |||
491 | OSD meshOsd = null; | ||
492 | |||
493 | if (primShape.SculptData.Length <= 0) | ||
494 | { | ||
495 | m_log.Error("[MESH]: asset data is zero length"); | ||
496 | return false; | ||
497 | } | ||
498 | |||
499 | long start = 0; | ||
500 | using (MemoryStream data = new MemoryStream(primShape.SculptData)) | ||
501 | { | ||
502 | try | ||
503 | { | ||
504 | OSD osd = OSDParser.DeserializeLLSDBinary(data); | ||
505 | if (osd is OSDMap) | ||
506 | meshOsd = (OSDMap)osd; | ||
507 | else | ||
508 | { | ||
509 | m_log.Warn("[Mesh}: unable to cast mesh asset to OSDMap"); | ||
510 | return false; | ||
511 | } | ||
512 | } | ||
513 | catch (Exception e) | ||
514 | { | ||
515 | m_log.Error("[MESH]: Exception deserializing mesh asset header:" + e.ToString()); | ||
516 | } | ||
517 | |||
518 | start = data.Position; | ||
519 | } | ||
520 | |||
521 | if (meshOsd is OSDMap) | ||
522 | { | ||
523 | OSDMap physicsParms = null; | ||
524 | OSDMap map = (OSDMap)meshOsd; | ||
525 | if (map.ContainsKey("physics_shape")) | ||
526 | physicsParms = (OSDMap)map["physics_shape"]; // old asset format | ||
527 | else if (map.ContainsKey("physics_mesh")) | ||
528 | physicsParms = (OSDMap)map["physics_mesh"]; // new asset format | ||
529 | |||
530 | if (physicsParms == null) | ||
531 | { | ||
532 | m_log.Warn("[MESH]: no recognized physics mesh found in mesh asset"); | ||
533 | return false; | ||
534 | } | ||
535 | |||
536 | int physOffset = physicsParms["offset"].AsInteger() + (int)start; | ||
537 | int physSize = physicsParms["size"].AsInteger(); | ||
538 | |||
539 | if (physOffset < 0 || physSize == 0) | ||
540 | return false; // no mesh data in asset | ||
541 | |||
542 | OSD decodedMeshOsd = new OSD(); | ||
543 | byte[] meshBytes = new byte[physSize]; | ||
544 | System.Buffer.BlockCopy(primShape.SculptData, physOffset, meshBytes, 0, physSize); | ||
545 | // byte[] decompressed = new byte[physSize * 5]; | ||
546 | try | ||
547 | { | ||
548 | using (MemoryStream inMs = new MemoryStream(meshBytes)) | ||
549 | { | ||
550 | using (MemoryStream outMs = new MemoryStream()) | ||
551 | { | ||
552 | using (ZOutputStream zOut = new ZOutputStream(outMs)) | ||
553 | { | ||
554 | byte[] readBuffer = new byte[2048]; | ||
555 | int readLen = 0; | ||
556 | while ((readLen = inMs.Read(readBuffer, 0, readBuffer.Length)) > 0) | ||
557 | { | ||
558 | zOut.Write(readBuffer, 0, readLen); | ||
559 | } | ||
560 | zOut.Flush(); | ||
561 | outMs.Seek(0, SeekOrigin.Begin); | ||
562 | |||
563 | byte[] decompressedBuf = outMs.GetBuffer(); | ||
564 | |||
565 | decodedMeshOsd = OSDParser.DeserializeLLSDBinary(decompressedBuf); | ||
566 | } | ||
567 | } | ||
568 | } | ||
569 | } | ||
570 | catch (Exception e) | ||
571 | { | ||
572 | m_log.Error("[MESH]: exception decoding physical mesh: " + e.ToString()); | ||
573 | return false; | ||
574 | } | ||
575 | |||
576 | OSDArray decodedMeshOsdArray = null; | ||
577 | |||
578 | // physics_shape is an array of OSDMaps, one for each submesh | ||
579 | if (decodedMeshOsd is OSDArray) | ||
580 | { | ||
581 | // Console.WriteLine("decodedMeshOsd for {0} - {1}", primName, Util.GetFormattedXml(decodedMeshOsd)); | ||
582 | |||
583 | decodedMeshOsdArray = (OSDArray)decodedMeshOsd; | ||
584 | foreach (OSD subMeshOsd in decodedMeshOsdArray) | ||
585 | { | ||
586 | if (subMeshOsd is OSDMap) | ||
587 | AddSubMesh(subMeshOsd as OSDMap, size, coords, faces); | ||
588 | } | ||
589 | } | ||
590 | } | ||
591 | |||
592 | return true; | ||
593 | } | ||
594 | |||
595 | /// <summary> | ||
596 | /// Generate the co-ords and faces necessary to construct a mesh from the sculpt data the accompanies a prim. | ||
597 | /// </summary> | ||
598 | /// <param name="primName"></param> | ||
599 | /// <param name="primShape"></param> | ||
600 | /// <param name="size"></param> | ||
601 | /// <param name="lod"></param> | ||
602 | /// <param name="coords">Coords are added to this list by the method.</param> | ||
603 | /// <param name="faces">Faces are added to this list by the method.</param> | ||
604 | /// <returns>true if coords and faces were successfully generated, false if not</returns> | ||
605 | private bool GenerateCoordsAndFacesFromPrimSculptData( | ||
606 | string primName, PrimitiveBaseShape primShape, Vector3 size, float lod, List<Coord> coords, List<Face> faces) | ||
607 | { | ||
608 | PrimMesher.SculptMesh sculptMesh; | ||
609 | Image idata = null; | ||
610 | string decodedSculptFileName = ""; | ||
611 | |||
612 | if (cacheSculptMaps && primShape.SculptTexture != UUID.Zero) | ||
613 | { | ||
614 | decodedSculptFileName = System.IO.Path.Combine(decodedSculptMapPath, "smap_" + primShape.SculptTexture.ToString()); | ||
615 | try | ||
616 | { | ||
617 | if (File.Exists(decodedSculptFileName)) | ||
618 | { | ||
619 | idata = Image.FromFile(decodedSculptFileName); | ||
620 | } | ||
621 | } | ||
622 | catch (Exception e) | ||
623 | { | ||
624 | m_log.Error("[SCULPT]: unable to load cached sculpt map " + decodedSculptFileName + " " + e.Message); | ||
625 | |||
626 | } | ||
627 | //if (idata != null) | ||
628 | // m_log.Debug("[SCULPT]: loaded cached map asset for map ID: " + primShape.SculptTexture.ToString()); | ||
629 | } | ||
630 | |||
631 | if (idata == null) | ||
632 | { | ||
633 | if (primShape.SculptData == null || primShape.SculptData.Length == 0) | ||
634 | return false; | ||
635 | |||
636 | try | ||
637 | { | ||
638 | OpenMetaverse.Imaging.ManagedImage unusedData; | ||
639 | OpenMetaverse.Imaging.OpenJPEG.DecodeToImage(primShape.SculptData, out unusedData, out idata); | ||
640 | |||
641 | if (idata == null) | ||
642 | { | ||
643 | // In some cases it seems that the decode can return a null bitmap without throwing | ||
644 | // an exception | ||
645 | m_log.WarnFormat("[PHYSICS]: OpenJPEG decoded sculpt data for {0} to a null bitmap. Ignoring.", primName); | ||
646 | |||
647 | return false; | ||
648 | } | ||
649 | |||
650 | unusedData = null; | ||
651 | |||
652 | //idata = CSJ2K.J2kImage.FromBytes(primShape.SculptData); | ||
653 | |||
654 | if (cacheSculptMaps) | ||
655 | { | ||
656 | try { idata.Save(decodedSculptFileName, ImageFormat.MemoryBmp); } | ||
657 | catch (Exception e) { m_log.Error("[SCULPT]: unable to cache sculpt map " + decodedSculptFileName + " " + e.Message); } | ||
658 | } | ||
659 | } | ||
660 | catch (DllNotFoundException) | ||
661 | { | ||
662 | m_log.Error("[PHYSICS]: OpenJpeg is not installed correctly on this system. Physics Proxy generation failed. Often times this is because of an old version of GLIBC. You must have version 2.4 or above!"); | ||
663 | return false; | ||
664 | } | ||
665 | catch (IndexOutOfRangeException) | ||
666 | { | ||
667 | m_log.Error("[PHYSICS]: OpenJpeg was unable to decode this. Physics Proxy generation failed"); | ||
668 | return false; | ||
669 | } | ||
670 | catch (Exception ex) | ||
671 | { | ||
672 | m_log.Error("[PHYSICS]: Unable to generate a Sculpty physics proxy. Sculpty texture decode failed: " + ex.Message); | ||
673 | return false; | ||
674 | } | ||
675 | } | ||
676 | |||
677 | PrimMesher.SculptMesh.SculptType sculptType; | ||
678 | switch ((OpenMetaverse.SculptType)primShape.SculptType) | ||
679 | { | ||
680 | case OpenMetaverse.SculptType.Cylinder: | ||
681 | sculptType = PrimMesher.SculptMesh.SculptType.cylinder; | ||
682 | break; | ||
683 | case OpenMetaverse.SculptType.Plane: | ||
684 | sculptType = PrimMesher.SculptMesh.SculptType.plane; | ||
685 | break; | ||
686 | case OpenMetaverse.SculptType.Torus: | ||
687 | sculptType = PrimMesher.SculptMesh.SculptType.torus; | ||
688 | break; | ||
689 | case OpenMetaverse.SculptType.Sphere: | ||
690 | sculptType = PrimMesher.SculptMesh.SculptType.sphere; | ||
691 | break; | ||
692 | default: | ||
693 | sculptType = PrimMesher.SculptMesh.SculptType.plane; | ||
694 | break; | ||
695 | } | ||
696 | |||
697 | bool mirror = ((primShape.SculptType & 128) != 0); | ||
698 | bool invert = ((primShape.SculptType & 64) != 0); | ||
699 | |||
700 | sculptMesh = new PrimMesher.SculptMesh((Bitmap)idata, sculptType, (int)lod, false, mirror, invert); | ||
701 | |||
702 | idata.Dispose(); | ||
703 | |||
704 | sculptMesh.DumpRaw(baseDir, primName, "primMesh"); | ||
705 | |||
706 | sculptMesh.Scale(size.X, size.Y, size.Z); | ||
707 | |||
708 | coords = sculptMesh.coords; | ||
709 | faces = sculptMesh.faces; | ||
710 | |||
711 | return true; | ||
712 | } | ||
713 | |||
694 | public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod) | 714 | public IMesh CreateMesh(String primName, PrimitiveBaseShape primShape, Vector3 size, float lod) |
695 | { | 715 | { |
696 | return CreateMesh(primName, primShape, size, lod, false); | 716 | return CreateMesh(primName, primShape, size, lod, false); |