// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This source file has been modified by the following contributors: // Copyright (C) 2014 Jan BOON (Kaetemi) // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU Affero General Public License as // published by the Free Software Foundation, either version 3 of the // License, or (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Affero General Public License for more details. // // You should have received a copy of the GNU Affero General Public License // along with this program. If not, see . #include "std3d.h" #include "nel/3d/mesh_multi_lod.h" #include "nel/3d/mesh_multi_lod_instance.h" #include "nel/3d/mesh_instance.h" #include "nel/3d/mesh_mrm.h" #include "nel/3d/scene.h" #include "nel/3d/coarse_mesh_manager.h" #include "nel/3d/skeleton_model.h" #include "nel/misc/fast_floor.h" #include "nel/3d/mesh_blender.h" #include "nel/3d/visual_collision_mesh.h" #include "nel/misc/debug.h" #include "nel/misc/hierarchical_timer.h" using namespace NLMISC; using namespace std; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { // *************************************************************************** void CMeshMultiLod::build(CMeshMultiLodBuild &mbuild) { // Clear the mesh clear (); // Build the base mesh CMeshBase::buildMeshBase (mbuild.BaseMesh); // Static flag _StaticLod=mbuild.StaticLod; // Resize the array _MeshVector.resize (mbuild.LodMeshes.size()); // For each slots for (uint slot=0; slot(mbuild.LodMeshes[slot].MeshGeom)==NULL) { // If it is a coarse mesh, it must be a CMeshGeom. _MeshVector[slot].MeshGeom = NULL; delete mbuild.LodMeshes[slot].MeshGeom; } else // Ok, no prb _MeshVector[slot].MeshGeom = mbuild.LodMeshes[slot].MeshGeom; } else // Ok, no prb _MeshVector[slot].MeshGeom = mbuild.LodMeshes[slot].MeshGeom; } // Sort the slot by the distance... for (int i=(uint)mbuild.LodMeshes.size()-1; i>0; i--) for (int j=0; j_MeshVector[j+1].DistMax) { // Exchange slots CMeshSlot tmp=_MeshVector[j]; _MeshVector[j]=_MeshVector[j+1]; _MeshVector[j+1]=tmp; tmp.MeshGeom=NULL; } } // Calc start and end polygon count for (uint k=0; kgetNumTriangles (startDist); // Get end distance float endDist=_MeshVector[k].DistMax; // Get end poly count if (k==mbuild.LodMeshes.size()-1) { _MeshVector[k].EndPolygonCount=_MeshVector[k].MeshGeom->getNumTriangles (endDist); if (startPolyCount==_MeshVector[k].EndPolygonCount) _MeshVector[k].EndPolygonCount=startPolyCount/2; } else _MeshVector[k].EndPolygonCount=_MeshVector[k+1].MeshGeom->getNumTriangles (endDist); // Calc A if (endDist==startDist) _MeshVector[k].A=0; else _MeshVector[k].A=(_MeshVector[k].EndPolygonCount-startPolyCount)/(endDist-startDist); // Calc A _MeshVector[k].B=_MeshVector[k].EndPolygonCount-_MeshVector[k].A*endDist; } // End: compile some stuff compileRunTime(); } // *************************************************************************** CTransformShape *CMeshMultiLod::createInstance(CScene &scene) { // Create a CMeshInstance, an instance of a multi lod mesh. CMeshMultiLodInstance *mi=(CMeshMultiLodInstance*)scene.createModel(NL3D::MeshMultiLodInstanceId); mi->Shape= this; mi->_LastLodMatrixDate=0; // instanciate the material part of the Mesh, ie the CMeshBase. CMeshBase::instanciateMeshBase(mi, &scene); // Create the necessary space for Coarse Instanciation instanciateCoarseMeshSpace(mi); // For all lods, do some instance init for MeshGeom for(uint i=0; i<_MeshVector.size(); i++) { if(_MeshVector[i].MeshGeom) _MeshVector[i].MeshGeom->initInstance(mi); } // init the Filter type mi->initRenderFilterType(); return mi; } // *************************************************************************** bool CMeshMultiLod::clip(const std::vector &pyramid, const CMatrix &worldMatrix) { // Look for the biggest mesh uint meshCount=(uint)_MeshVector.size(); for (uint i=0; iclip (pyramid, worldMatrix); } } return true; } // *************************************************************************** void CMeshMultiLod::render(IDriver *drv, CTransformShape *trans, bool passOpaque) { // Render good meshes CMeshMultiLodInstance *instance=safe_cast(trans); // Static or dynamic coarse mesh ? CCoarseMeshManager *manager; // Get the coarse mesh manager manager=instance->getOwnerScene()->getCoarseMeshManager(); // *** Render Lods // Second lod ? if ( (instance->Lod1!=0xffffffff) && (passOpaque==false) ) { // build rdrFlags to rdr both transparent and opaque materials, // use globalAlphaBlend, and disable ZWrite for Lod1 uint32 rdrFlags= IMeshGeom::RenderOpaqueMaterial | IMeshGeom::RenderTransparentMaterial | IMeshGeom::RenderGlobalAlpha | IMeshGeom::RenderGADisableZWrite; // NB: very important to render Lod1 first, because Lod0 is still rendered with ZWrite enabled. renderMeshGeom (instance->Lod1, drv, instance, instance->PolygonCountLod1, rdrFlags, 1.f-instance->BlendFactor, manager); } // Have an opaque pass ? if ( (instance->Flags&CMeshMultiLodInstance::Lod0Blend) == 0) { // Is this slot a CoarseMesh? if ( _MeshVector[instance->Lod0].Flags&CMeshSlot::CoarseMesh ) { // render as a CoarseMesh the lod 0, only in opaque pass if(passOpaque) renderCoarseMesh (instance->Lod0, drv, instance, manager); } else { // build rdrFlags the normal way (as CMesh::render() for example) uint32 mask= (0-(uint32)passOpaque); uint32 rdrFlags; // select rdrFlags, without ifs. rdrFlags= mask & (IMeshGeom::RenderOpaqueMaterial | IMeshGeom::RenderPassOpaque); rdrFlags|= ~mask & (IMeshGeom::RenderTransparentMaterial); // Only render the normal way the first lod renderMeshGeom (instance->Lod0, drv, instance, instance->PolygonCountLod0, rdrFlags, 1, manager); } } else { // Should not be in opaque nlassert (passOpaque==false); // build rdrFlags to rdr both transparent and opaque materials, // use globalAlphaBlend, BUT Don't disable ZWrite for Lod0 uint32 rdrFlags= IMeshGeom::RenderOpaqueMaterial | IMeshGeom::RenderTransparentMaterial | IMeshGeom::RenderGlobalAlpha; // Render first lod in blend mode. Don't disable ZWrite for Lod0 renderMeshGeom (instance->Lod0, drv, instance, instance->PolygonCountLod0, rdrFlags, instance->BlendFactor, manager); } } // *************************************************************************** void CMeshMultiLod::serial(NLMISC::IStream &f) { /* *********************************************** * WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance * It can be loaded/called through CAsyncFileManager for instance * ***********************************************/ // Serial a version number (void)f.serialVersion (0); // serial Materials infos contained in CMeshBase. CMeshBase::serialMeshBase(f); // Static lod flag f.serial (_StaticLod); // Serial the values f.serialCont (_MeshVector); // if reading, compile some stuff if (f.isReading()) compileRunTime(); } // *************************************************************************** float CMeshMultiLod::getNumTrianglesWithCoarsestDist(float distance, float coarsestMeshDist) const { // Look in the table for good distances.. uint meshCount=(uint)_MeshVector.size(); // At least on mesh if (meshCount>0) { if (coarsestMeshDist != -1) { if (coarsestMeshDist != 0) { // rescale distance to new coarse mesh distance.. distance *= _MeshVector[meshCount - 1].DistMax / coarsestMeshDist; } } uint i=0; // Look for good i while ( _MeshVector[i].DistMax < distance) { if (i==meshCount-1) // Abort if last one break; i++; } // Ref on slot const CMeshSlot &slot=_MeshVector[i]; // Is mesh present ? if (slot.MeshGeom) { // Get the polygon count with the distance float polyCount=slot.A * distance + slot.B; /*// Get the perfect polygon count in this slot for the asked distance float goodPolyCount=slot.MeshGeom->getNumTriangles (distance); // Get the next slot perfect polygon count float realEndPolyCount; // Last slot ? if ( (igetNumTriangles (slot.DistMax); else // Take end number polygon count in the this slot realEndPolyCount=slot.EndPolygonCount; // Return blended polygon count to have a continous decreasing function return (goodPolyCount-slot.BeginPolygonCount) * (realEndPolyCount-slot.BeginPolygonCount) / (slot.EndPolygonCount-slot.BeginPolygonCount) + slot.BeginPolygonCount;*/ return polyCount; } } return 0; } // *************************************************************************** void CMeshMultiLod::getAABBox(NLMISC::CAABBox &bbox) const { // Get count uint count=(uint)_MeshVector.size(); for (uint slot=0; slotgetBoundingBox().getAABBox(); // ok break; } } } // *************************************************************************** void CMeshMultiLod::clear () { _MeshVector.clear (); } // *************************************************************************** void CMeshMultiLod::CMeshSlot::serial(NLMISC::IStream &f) { // Check version (void)f.serialVersion (0); f.serialPolyPtr (MeshGeom); f.serial (A); f.serial (B); f.serial (DistMax); f.serial (EndPolygonCount); f.serial (BlendLength); f.serial (Flags); if (f.isReading()) { } } // *************************************************************************** CMeshMultiLod::CMeshSlot::CMeshSlot () { MeshGeom=NULL; CoarseNumTris= 0; } // *************************************************************************** CMeshMultiLod::CMeshSlot::~CMeshSlot () { if (MeshGeom) delete MeshGeom; } // *************************************************************************** void CMeshMultiLod::renderMeshGeom (uint slot, IDriver *drv, CMeshMultiLodInstance *trans, float numPoylgons, uint32 rdrFlags, float alpha, CCoarseMeshManager *manager) { // Ref CMeshSlot &slotRef=_MeshVector[slot]; // MeshGeom exist? if (slotRef.MeshGeom) { // NB Here, the meshGeom may still be a coarseMesh, but rendered through CMeshGeom if(slotRef.Flags&CMeshSlot::CoarseMesh) { // Render only for opaque material if(manager && (rdrFlags & IMeshGeom::RenderOpaqueMaterial) ) { bool gaDisableZWrite= (rdrFlags & IMeshGeom::RenderGADisableZWrite)?true:false; // Render the CoarseMesh with the manager material CMaterial &material= manager->getMaterial(); // modulate material for alphaBlend transition // ---------- // get average sun color for this coarseMesh CRGBA newCol= trans->getCoarseMeshLighting(); // Use a CMeshBlender to modify material and driver. CMeshBlender blender; blender.prepareRenderForGlobalAlphaCoarseMesh(material, drv, newCol, alpha, gaDisableZWrite); // render simple the coarseMesh CMeshGeom *meshGeom= safe_cast(slotRef.MeshGeom); // Force corse mesh vertex buffer in system memory const_cast(meshGeom->getVertexBuffer ()).setPreferredMemory (CVertexBuffer::RAMPreferred, false); meshGeom->renderSimpleWithMaterial(drv, trans->getWorldMatrix(), material); // resetup standard CoarseMeshMaterial material values // ---------- // blender restore blender.restoreRenderCoarseMesh(material, drv, gaDisableZWrite); } } else { // Render the geom mesh // Disable ZWrite only if in transition and for rendering Lod1 slotRef.MeshGeom->render (drv, trans, numPoylgons, rdrFlags, alpha); } } } // *************************************************************************** void CMeshMultiLod::renderCoarseMesh (uint slot, IDriver *drv, CMeshMultiLodInstance *trans, CCoarseMeshManager *manager) { // if the manager is NULL, quit. if(manager==NULL) return; // get the scene CScene *scene= trans->getOwnerScene(); if(!scene) return; // If filtered... if( (scene->getFilterRenderFlags() & UScene::FilterCoarseMesh)==0 ) return; // Ref CMeshSlot &slotRef=_MeshVector[slot]; // the slot must be a Coarse mesh nlassert(slotRef.Flags&CMeshSlot::CoarseMesh); // Get a pointer on the geom mesh CMeshGeom *meshGeom= safe_cast(slotRef.MeshGeom); // ** If not the same as Before (or if NULL before...) if ( trans->_LastCoarseMesh!=meshGeom ) { uint numVerts= meshGeom->getVertexBuffer().getNumVertices(); uint numTris= slotRef.CoarseNumTris; // If empty meshGeom, erase cache (each frame, ugly but error mgt here...) if( numTris==0 || numVerts==0 ) trans->_LastCoarseMesh= NULL; else { // Cache trans->_LastCoarseMesh= meshGeom; trans->_LastCoarseMeshNumVertices= numVerts; // Check setuped size. nlassert( trans->_CoarseMeshVB.size() >= numVerts*manager->getVertexSize() ); // Fill only UVs here. (Pos updated in Matrix pass. Color in Lighting Pass) trans->setUVCoarseMesh( *meshGeom, manager->getVertexSize(), manager->getUVOff() ); // Dirt the matrix trans->_LastLodMatrixDate=0; // Dirt the lighting. NB: period maximum is 255. Hence the -256, to ensure lighting compute now trans->_LastLodLightingDate= -0x100; } } // ** If setuped, update and render if( trans->_LastCoarseMesh ) { // Matrix has changed ? if ( trans->ITransformable::compareMatrixDate (trans->_LastLodMatrixDate) ) { // Get date trans->_LastLodMatrixDate = trans->ITransformable::getMatrixDate(); // Set matrix trans->setPosCoarseMesh ( *meshGeom, trans->getMatrix(), manager->getVertexSize() ); } // Lighting: test if must update lighting, according to date of HrcTrav (num of CScene::render() call). sint64 currentDate= scene->getHrcTrav().CurrentDate; if( trans->_LastLodLightingDate < currentDate - scene->getCoarseMeshLightingUpdate() ) { // reset the date. trans->_LastLodLightingDate= currentDate; // get average sun color CRGBA sunContrib= trans->getCoarseMeshLighting(); // Invert BR if driver is BGRA if(drv->getVertexColorFormat()==CVertexBuffer::TBGRA) sunContrib.swapBR(); // Set color trans->setColorCoarseMesh ( sunContrib, manager->getVertexSize(), manager->getColorOff()); } // Add dynamic to the manager if( !manager->addMesh(trans->_LastCoarseMeshNumVertices, &trans->_CoarseMeshVB[0], slotRef.CoarseNumTris, &slotRef.CoarseTriangles[0] ) ) { // If failure, flush the manager manager->flushRender(drv); // then try to re-add. No-op if fails this time.. manager->addMesh(trans->_LastCoarseMeshNumVertices, &trans->_CoarseMeshVB[0], slotRef.CoarseNumTris, &slotRef.CoarseTriangles[0] ); } } } // *************************************************************************** void CMeshMultiLod::compileDistMax() { // Last element if(_MeshVector.empty()) IShape::_DistMax= -1; else IShape::_DistMax= _MeshVector.back().DistMax; } // *************************************************************************** const IMeshGeom& CMeshMultiLod::getMeshGeom (uint slot) const { // Checks nlassert (slot(_MeshVector[0].MeshGeom); if(mgeom==NULL) return; // ok, setup. mgeom->changeMRMDistanceSetup(distanceFinest, distanceMiddle, distanceCoarsest); } // *************************************************************************** IMeshGeom *CMeshMultiLod::supportMeshBlockRendering (CTransformShape *trans, float &polygonCount ) const { IMeshGeom *ret= NULL; // get the instance CMeshMultiLodInstance *instance=safe_cast(trans); // Must not be in blend transition. if ( (instance->Flags&CMeshMultiLodInstance::Lod0Blend) == 0) { uint slot= instance->Lod0; // The slot must not be a CoarseMesh if ( (_MeshVector[slot].Flags&CMeshSlot::CoarseMesh)==0 ) { // MeshGeom exist? ret= _MeshVector[slot].MeshGeom; } } // Ok if meshGeom is ok. if( ret && ret->supportMeshBlockRendering() ) { polygonCount= instance->PolygonCountLod0; return ret; } else return NULL; } // *************************************************************************** void CMeshMultiLod::profileMeshGeom (uint slot, CRenderTrav *rdrTrav, CMeshMultiLodInstance *trans, float numPoylgons, uint32 rdrFlags) { // Ref CMeshSlot &slotRef=_MeshVector[slot]; // MeshGeom exist? if (slotRef.MeshGeom) { // NB Here, the meshGeom may still be a coarseMesh, but rendered through CMeshGeom if(slotRef.Flags&CMeshSlot::CoarseMesh) { // Render only for opaque material if(rdrFlags & IMeshGeom::RenderOpaqueMaterial) { slotRef.MeshGeom->profileSceneRender(rdrTrav, trans, numPoylgons, rdrFlags); } } else { slotRef.MeshGeom->profileSceneRender(rdrTrav, trans, numPoylgons, rdrFlags); } } } // *************************************************************************** void CMeshMultiLod::profileSceneRender(CRenderTrav *rdrTrav, CTransformShape *trans, bool passOpaque) { // Render good meshes CMeshMultiLodInstance *instance=safe_cast(trans); // Second lod ? if ( (instance->Lod1!=0xffffffff) && (passOpaque==false) ) { // build rdrFlags to rdr both transparent and opaque materials, // use globalAlphaBlend, and disable ZWrite for Lod1 uint32 rdrFlags= IMeshGeom::RenderOpaqueMaterial | IMeshGeom::RenderTransparentMaterial | IMeshGeom::RenderGlobalAlpha | IMeshGeom::RenderGADisableZWrite; // NB: very important to render Lod1 first, because Lod0 is still rendered with ZWrite enabled. profileMeshGeom (instance->Lod1, rdrTrav, instance, instance->PolygonCountLod1, rdrFlags); } // Have an opaque pass ? if ( (instance->Flags&CMeshMultiLodInstance::Lod0Blend) == 0) { // Is this slot a CoarseMesh? if ( _MeshVector[instance->Lod0].Flags&CMeshSlot::CoarseMesh ) { } else { // build rdrFlags the normal way (as CMesh::render() for example) uint32 mask= (0-(uint32)passOpaque); uint32 rdrFlags; // select rdrFlags, without ifs. rdrFlags= mask & (IMeshGeom::RenderOpaqueMaterial | IMeshGeom::RenderPassOpaque); rdrFlags|= ~mask & (IMeshGeom::RenderTransparentMaterial); // Only render the normal way the first lod profileMeshGeom (instance->Lod0, rdrTrav, instance, instance->PolygonCountLod0, rdrFlags); } } else { // Should not be in opaque nlassert (passOpaque==false); // build rdrFlags to rdr both transparent and opaque materials, // use globalAlphaBlend, BUT Don't disable ZWrite for Lod0 uint32 rdrFlags= IMeshGeom::RenderOpaqueMaterial | IMeshGeom::RenderTransparentMaterial | IMeshGeom::RenderGlobalAlpha; // Render first lod in blend mode. Don't disable ZWrite for Lod0 profileMeshGeom (instance->Lod0, rdrTrav, instance, instance->PolygonCountLod0, rdrFlags); } } // *************************************************************************** void CMeshMultiLod::instanciateCoarseMeshSpace(CMeshMultiLodInstance *mi) { CCoarseMeshManager *manager= mi->getOwnerScene()->getCoarseMeshManager(); if(manager) { // For all MeshSlots that have a CoarseMesh, count max Coarse NumVertices; uint numVertices= 0; for(uint i=0;i<_MeshVector.size();i++) { CMeshSlot &slotRef= _MeshVector[i]; if( slotRef.Flags & CMeshSlot::CoarseMesh ) { // Get a pointer on the geom mesh CMeshGeom *meshGeom= safe_cast(slotRef.MeshGeom); numVertices= max(numVertices, (uint)meshGeom->getVertexBuffer().getNumVertices() ); } } // Then allocate vertex space for dest manager vertex size. mi->_CoarseMeshVB.resize( numVertices*manager->getVertexSize() ); } } // *************************************************************************** void CMeshMultiLod::compileCoarseMeshes() { // For All Slots that are CoarseMeshes. for(uint i=0;i<_MeshVector.size();i++) { CMeshSlot &slotRef= _MeshVector[i]; if( slotRef.Flags & CMeshSlot::CoarseMesh ) { // reset slotRef.CoarseNumTris= 0; // Get a pointer on the geom mesh CMeshGeom *meshGeom= safe_cast(slotRef.MeshGeom); // For All RdrPass of the 1st matrix block if( meshGeom->getNbMatrixBlock()>0 ) { // 1st count for(uint i=0;igetNbRdrPass(0);i++) { slotRef.CoarseNumTris+= meshGeom->getRdrPassPrimitiveBlock(0, i).getNumIndexes()/3; } // 2nd allocate and fill if( slotRef.CoarseNumTris ) { slotRef.CoarseTriangles.resize(slotRef.CoarseNumTris * 3); TCoarseMeshIndexType *dstPtr= &slotRef.CoarseTriangles[0]; uint totalTris = 0; for(uint i=0;igetNbRdrPass(0);i++) { const CIndexBuffer &pb= meshGeom->getRdrPassPrimitiveBlock(0, i); CIndexBufferRead ibaRead; pb.lock (ibaRead); uint numTris= pb.getNumIndexes()/3; totalTris += numTris; if (pb.getFormat() == CIndexBuffer::Indices16) { if (sizeof(TCoarseMeshIndexType) == sizeof(uint16)) { memcpy(dstPtr, (uint16 *) ibaRead.getPtr(), numTris*3*sizeof(uint16)); dstPtr+= numTris*3; } else { // 16 -> 32 uint16 *src = (uint16 *) ibaRead.getPtr(); for(uint k = 0; k < numTris; ++k) { *dstPtr++ = (TCoarseMeshIndexType) *src++; *dstPtr++ = (TCoarseMeshIndexType) *src++; *dstPtr++ = (TCoarseMeshIndexType) *src++; } } } else { if (sizeof(TCoarseMeshIndexType) == sizeof(uint32)) { memcpy(dstPtr, (uint32 *) ibaRead.getPtr(), numTris*3*sizeof(uint32)); dstPtr+= numTris*3; } else { const uint32 *src = (const uint32 *) ibaRead.getPtr(); for(uint k = 0; k < numTris; ++k) { // 32 -> 16 nlassert(src[0] <= 0xffff); nlassert(src[1] <= 0xffff); nlassert(src[2] <= 0xffff); *dstPtr++ = (TCoarseMeshIndexType) *src++; *dstPtr++ = (TCoarseMeshIndexType) *src++; *dstPtr++ = (TCoarseMeshIndexType) *src++; } } } } nlassert(totalTris == slotRef.CoarseNumTris); } } } } } // *************************************************************************** void CMeshMultiLod::compileRunTime() { /* *********************************************** * WARNING: This Class/Method must be thread-safe (ctor/dtor/serial): no static access for instance * It can be loaded/called through CAsyncFileManager for instance * ***********************************************/ // **** MultiLod basics compileDistMax(); compileCoarseMeshes(); // **** try to build a Visual Collision Mesh // clear first if(_VisualCollisionMesh) { delete _VisualCollisionMesh; _VisualCollisionMesh= NULL; } // build only if wanted if( (_CollisionMeshGeneration==AutoCameraCol && !_LightInfos.empty()) || _CollisionMeshGeneration==ForceCameraCol ) { // try to retrieve the info from a CMeshGeom only if(getNumSlotMesh()) { const CMeshGeom *meshGeom= dynamic_cast(&getMeshGeom(0)); if(meshGeom) { vector vertices; vector indices; if(meshGeom->retrieveVertices(vertices) && meshGeom->retrieveTriangles(indices)) { // ok, can build! _VisualCollisionMesh= new CVisualCollisionMesh; // if fails to build cause of too many vertices/indices for instance if(!_VisualCollisionMesh->build(vertices, indices,const_cast(meshGeom->getVertexBuffer()))) { // delete delete _VisualCollisionMesh; _VisualCollisionMesh= NULL; } } } } } } // *************************************************************************** void CMeshMultiLod::buildSystemGeometry() { // clear any _SystemGeometry.clear(); // Use the first lod, for system geometry copy if(getNumSlotMesh()) { // the first is a meshGeom? const CMeshGeom *meshGeom= dynamic_cast(&getMeshGeom(0)); if(meshGeom) { // retrieve geometry (if VB/IB not resident) if( !meshGeom->retrieveVertices(_SystemGeometry.Vertices) || !meshGeom->retrieveTriangles(_SystemGeometry.Triangles)) { _SystemGeometry.clear(); } } // else it is a mrm geom? else { const CMeshMRMGeom *meshMRMGeom= dynamic_cast(&getMeshGeom(0)); if(meshMRMGeom) { // Choose the best Lod available for system geometry if(meshMRMGeom->getNbLodLoaded()==0) return; uint lodId= meshMRMGeom->getNbLodLoaded()-1; // retrieve geometry (if VB/IB not resident) if( !meshMRMGeom->buildGeometryForLod(lodId, _SystemGeometry.Vertices, _SystemGeometry.Triangles) ) { _SystemGeometry.clear(); } } } } // TestYoyo /*static uint32 totalMem= 0; totalMem+= _SystemGeometry.Vertices.size()*sizeof(CVector); totalMem+= _SystemGeometry.Triangles.size()*sizeof(uint32); nlinfo("CMeshMultiLod: TotalMem: %d", totalMem);*/ } } // NL3D