// NeL - MMORPG Framework // Copyright (C) 2010 Winch Gate Property Limited // // This source file has been modified by the following contributors: // Copyright (C) 2020 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/shape_bank.h" #include "nel/3d/mesh_base.h" #include "nel/misc/file.h" #include "nel/misc/path.h" #include "nel/misc/rect.h" #include "nel/misc/algo.h" #include "nel/misc/progress_callback.h" using namespace std; using namespace NLMISC; #ifdef DEBUG_NEW #define new DEBUG_NEW #endif namespace NL3D { // *************************************************************************** CShapeBank::CShapeBank() { // Default cache creation addShapeCache( "default" ); _MaxUploadPerFrame = 16*1024; } // *************************************************************************** CShapeBank::~CShapeBank() { } // *************************************************************************** IShape*CShapeBank::addRef(const string &shapeNameNotLwr) { string shapeName= toLowerAscii(shapeNameNotLwr); // get the shape info (must succeed) TShapeInfoMap::iterator scfpmIt = ShapePtrToShapeInfo.find( getShapePtrFromShapeName( shapeName ) ); nlassert( scfpmIt != ShapePtrToShapeInfo.end() ); // If The shape is not inserted into a cache, just return it if( !scfpmIt->second.isAdded ) return getShapePtrFromShapeName( shapeName ); // else If the shape is inserted in a shape cache remove it scfpmIt->second.isAdded = false; CShapeCache *pShpCache = scfpmIt->second.pShpCache; nlassert( pShpCache != NULL ); // Search the shape cache for the shape we want to remove list::iterator lsIt = pShpCache->Elements.begin(); while(lsIt != pShpCache->Elements.end()) { const string *sTemp = getShapeNameFromShapePtr(*lsIt); if( *sTemp == shapeName ) { // Ok the shape cache contains the shape remove it and return pShpCache->Elements.erase( lsIt ); return getShapePtrFromShapeName( shapeName ); } ++lsIt; } nlassert( false ); return getShapePtrFromShapeName( shapeName ); } // *************************************************************************** void CShapeBank::release(IShape* pShp) { // Do we have the last smartPtr on the shape ? const string* str = getShapeNameFromShapePtr( pShp ); if (str == NULL) { nlwarning ("Trying to release a mesh that have not be added to the shape bank"); } else { TShapeMap::iterator smIt = ShapeMap.find( *str ); if( smIt != ShapeMap.end() ) { if( smIt->second.getNbRef() == 1 ) { // Yes -> add the shape to its shapeCache CShapeCache *pShpCache = getShapeCachePtrFromShapePtr( pShp ); pShpCache->Elements.push_front( pShp ); TShapeInfoMap::iterator scfpmIt = ShapePtrToShapeInfo.find( pShp ); if( scfpmIt != ShapePtrToShapeInfo.end() ) { scfpmIt->second.isAdded = true; } // check the shape cache checkShapeCache(getShapeCachePtrFromShapePtr(pShp)); } } else { nlassert( false ); } } } // *************************************************************************** void CShapeBank::processWaitingShapes () { uint32 nTotalUploaded = 0; TWaitingShapesMap::iterator wsmmIt = WaitingShapes.begin(); while( wsmmIt != WaitingShapes.end() ) { // Backup next iterator TWaitingShapesMap::iterator wsmmItNext = wsmmIt; wsmmItNext++; const string &shapeName = wsmmIt->first; CWaitingShape &rWS = wsmmIt->second; IShape *pShp = rWS.ShapePtr; // Take care this value is shared between thread so copy it in a local variable first switch (rWS.State) { case AsyncLoad_Shape: // Check if we can pass to the AsyncLoad_Texture state if (pShp != NULL) { if (pShp == (IShape*)-1) rWS.State = AsyncLoad_Error; else rWS.State = AsyncLoad_Texture; } break; case AsyncLoad_Texture: { // Setup all textures and lightmaps of the shape if (nTotalUploaded > _MaxUploadPerFrame) break; CMeshBase *pMesh = dynamic_cast(pShp); if( pMesh != NULL ) { uint8 j; uint32 i, CurrentProgress = 0; uint32 nNbMat = pMesh->getNbMaterial(); for (i = 0; i < nNbMat; ++i) { const CMaterial &rMat = pMesh->getMaterial(i); // Parse all textures from this material and setup for (j = 0; j < IDRV_MAT_MAXTEXTURES; ++j) { if (CurrentProgress >= rWS.UpTextProgress) { if (rMat.texturePresent(j)) { if ((!_pDriver->isTextureExist(*rMat.getTexture(j))) || (rWS.UpTextLine > 0) || (rWS.UpTextMipMap > 0)) { //_pDriver->setupTexture (*rMat.getTexture(j)); if (!processWSUploadTexture (rWS, nTotalUploaded, rMat.getTexture(j))) break; } else { /* Must release texture data because of the following scenario (common!) - An IG with 2 different meshs is async loaded - the 2 meshs access the same texture "pipo.tga" - in CAsyncFileManager3D::loadMesh(), if this case arise, the textures are loaded twice in RAM!!!! (is managed per mesh, but not thourhg meshs) - hence the first mesh setup and upload above the texture in VRAM but the second still have data in RAM!! (because 2nd will say that isTextureExist() .... Note: you could say that the cool stuff would be to not load the 2 :) but it appears that this case arise rarely because of the following statement: - while the second mesh is async loaded, the first upload in VRAM so there is good change that the second don't load the texture (isTextureExist() return true in the thread) */ ITexture *tex= rMat.getTexture(j); if(tex->getReleasable()) tex->release(); } } ++rWS.UpTextProgress; } ++CurrentProgress; if (nTotalUploaded > _MaxUploadPerFrame) break; } if (nTotalUploaded > _MaxUploadPerFrame) break; // Do the same with lightmaps if (rMat.getShader() == CMaterial::LightMap) { uint j = 0; ITexture *pText = rMat.getLightMap (j); while (pText != NULL) { if (CurrentProgress >= rWS.UpTextProgress) { if ((!_pDriver->isTextureExist(*pText)) || (rWS.UpTextLine > 0) || (rWS.UpTextMipMap > 0)) { //_pDriver->setupTexture (*pText); if (!processWSUploadTexture (rWS, nTotalUploaded, pText)) break; } else { // see above for explanation if(pText->getReleasable()) pText->release(); } ++rWS.UpTextProgress; } ++CurrentProgress; ++j; pText = rMat.getLightMap (j); if (nTotalUploaded > _MaxUploadPerFrame) break; } } if (nTotalUploaded > _MaxUploadPerFrame) break; } } if (nTotalUploaded > _MaxUploadPerFrame) break; rWS.State = AsyncLoad_Ready; } break; case AsyncLoad_Ready: add (wsmmIt->first, pShp); rWS.State = AsyncLoad_Delete; break; // The delete operation can take several frames to complete but this is not a problem // For error do the same as delete but let the flag to error if a shape is asked just after // the error was found case AsyncLoad_Error: case AsyncLoad_Delete: rWS.RefCnt -= 1; if (rWS.RefCnt == 0) { // We have to signal if we are the last std::set::iterator ite = rWS.Signal.begin(); while (ite != rWS.Signal.end()) { bool *bSignal = *ite; if (bSignal != NULL) { bool bFound = false; TWaitingShapesMap::iterator wsmmIt2 = WaitingShapes.begin(); while (wsmmIt2 != WaitingShapes.end()) { const string &shapeName2 = wsmmIt2->first; if (shapeName2 != shapeName) { std::set::iterator ite2 = wsmmIt2->second.Signal.begin(); while (ite2 != wsmmIt2->second.Signal.end()) { if (*ite2 == bSignal) { bFound = true; break; } ite2++; } if (ite2 != wsmmIt2->second.Signal.end()) break; } ++wsmmIt2; } if (!bFound) *bSignal = true; } // Next iterator ite++; } WaitingShapes.erase (wsmmIt); } break; default: nlstop; // This must never happen break; } wsmmIt = wsmmItNext; } } // *************************************************************************** void CShapeBank::setMaxBytesToUpload (uint32 MaxUploadPerFrame) { _MaxUploadPerFrame = MaxUploadPerFrame; } // *************************************************************************** bool CShapeBank::processWSUploadTexture (CWaitingShape &rWS, uint32 &nTotalUploaded, ITexture *pText) { CRect zeRect; uint32 nFace, nWeight = 0, nMipMap; if ((rWS.UpTextMipMap == 0) && (rWS.UpTextLine == 0)) { // Create the texture only and do not upload anything bool isRel = pText->getReleasable (); pText->setReleasable (false); bool isAllUploaded = false; _pDriver->setupTextureEx (*pText, false, isAllUploaded); pText->setReleasable (isRel); if (isAllUploaded) return true; } if (pText->mipMapOn()) nMipMap = pText->getMipMapCount(); else nMipMap = 1; // Upload all mipmaps for (; rWS.UpTextMipMap < nMipMap;) { uint32 nMM = rWS.UpTextMipMap; // What is left to upload ? nWeight = pText->getSize (nMM) - rWS.UpTextLine*pText->getWidth(nMM); // Yoyo: important: in case of DXTC1, must mul before div nWeight= (nWeight*CBitmap::bitPerPixels[pText->getPixelFormat()]) / 8; if (pText->isTextureCube()) nWeight *= 6; // NB: nWeight can be 0 in case of 1x1 in DXTC1. Estimate 1 byte if(nWeight==0) nWeight= 1; // Setup rectangle if ((nTotalUploaded + nWeight) > _MaxUploadPerFrame) { // We cannot upload the whole mipmap -> we have to cut it uint32 nSizeToUpload = _MaxUploadPerFrame - nTotalUploaded; uint32 nLineWeight = (pText->getWidth(nMM)*CBitmap::bitPerPixels[pText->getPixelFormat()]) / 8; // NB: nLineWeight can be 0 in case of 1x1 in DXTC1. Estimate 1 byte (avoid divide by zero) if(nLineWeight==0) nLineWeight= 1; if (pText->isTextureCube()) nLineWeight *= 6; // compute the number of lines we'll upload uint32 nNbLineToUpload = nSizeToUpload / nLineWeight; nNbLineToUpload = nNbLineToUpload / 4; if (nNbLineToUpload == 0) nNbLineToUpload = 1; nNbLineToUpload *= 4; // Upload 4 line by 4 line // setup the rect uint32 nNewLine = rWS.UpTextLine + nNbLineToUpload; if (nNewLine > pText->getHeight(nMM)) nNewLine = pText->getHeight(nMM); zeRect.set (0, rWS.UpTextLine, pText->getWidth(nMM), nNewLine); rWS.UpTextLine = nNewLine; if (rWS.UpTextLine == pText->getHeight(nMM)) { rWS.UpTextLine = 0; rWS.UpTextMipMap += 1; } } else { // We can upload the whole mipmap (or the whole rest of the mipmap) zeRect.set (0, rWS.UpTextLine, pText->getWidth(nMM), pText->getHeight(nMM)); rWS.UpTextLine = 0; rWS.UpTextMipMap += 1; } // Upload ! if (pText->isTextureCube()) { for (nFace = 0; nFace < 6; ++nFace) _pDriver->uploadTextureCube (*pText, zeRect, (uint8)nMM, (uint8)nFace); } else { _pDriver->uploadTexture (*pText, zeRect, (uint8)nMM); } nTotalUploaded += nWeight; if (nTotalUploaded > _MaxUploadPerFrame) return false; } if (pText->getReleasable()) pText->release(); rWS.UpTextMipMap = 0; rWS.UpTextLine = 0; return true; } // *************************************************************************** CShapeBank::TShapeState CShapeBank::getPresentState (const string &shapeNameNotLwr) { string shapeName= toLowerAscii(shapeNameNotLwr); // Is the shape is found in the shape map so return Present TShapeMap::iterator smIt = ShapeMap.find (shapeName); if( smIt != ShapeMap.end() ) return Present; // Look in the waiting shapes TWaitingShapesMap::iterator wsmmIt = WaitingShapes.find (shapeName); if (wsmmIt != WaitingShapes.end()) return wsmmIt->second.State; // AsyncLoad_* return NotPresent; } // *************************************************************************** IShape *CShapeBank::getShape (const std::string &shapeNameNotLwr) { string shapeName= toLowerAscii(shapeNameNotLwr); // Is the shape is found in the shape map so return Present TShapeMap::iterator smIt = ShapeMap.find (shapeName); if( smIt != ShapeMap.end() ) return smIt->second; return NULL; } // *************************************************************************** void CShapeBank::load (const string &shapeNameNotLwr) { string shapeName= toLowerAscii(shapeNameNotLwr); TShapeMap::iterator smIt = ShapeMap.find(shapeName); if( smIt == ShapeMap.end() ) { // If we are loading it asynchronously so we do not have to try to load it in sync mode TWaitingShapesMap::iterator wsmmIt = WaitingShapes.find (shapeName); if (wsmmIt != WaitingShapes.end()) return; CShapeStream mesh; CIFile meshfile; if (meshfile.open(CPath::lookup(shapeName, false))) { meshfile.serial( mesh ); meshfile.close(); } else { nlwarning ("CShapeBank::load() : Can't open file %s", shapeName.c_str()); } if (mesh.getShapePointer() != NULL) { // Add the shape to the map. add( shapeName, mesh.getShapePointer() ); } } } // *************************************************************************** void CShapeBank::loadAsync (const std::string &shapeNameNotLwr, IDriver *pDriver, const CVector &position, bool *bSignal, uint selectedTexture) { string shapeName= toLowerAscii(shapeNameNotLwr); TShapeMap::iterator smIt = ShapeMap.find(shapeName); if (smIt != ShapeMap.end()) return; _pDriver = pDriver; // Backup the pointer to the driver for later use TWaitingShapesMap::iterator wsmmIt = WaitingShapes.find (shapeName); // First time this shape is loaded ? bool firstTime = wsmmIt == WaitingShapes.end(); if (firstTime) wsmmIt = WaitingShapes.insert (TWaitingShapesMap::value_type(shapeName, CWaitingShape())).first; // Add a reference to it CWaitingShape &rWS = wsmmIt->second; // Insert a new signal pointer rWS.Signal.insert (bSignal); // Add a signal rWS.RefCnt += 1; // Launch an async mesh loader the first time if (firstTime) CAsyncFileManager3D::getInstance().loadMesh (shapeName, &(wsmmIt->second.ShapePtr), pDriver, position, selectedTexture); } // *************************************************************************** void CShapeBank::cancelLoadAsync (const std::string &shapeNameNotLwr) { string shapeName= toLowerAscii(shapeNameNotLwr); TWaitingShapesMap::iterator wsmmIt = WaitingShapes.find(shapeName); if (wsmmIt != WaitingShapes.end()) { wsmmIt->second.RefCnt -= 1; if (wsmmIt->second.RefCnt == 0) { // nlinfo("unloadasync %s", shapeName); CAsyncFileManager3D::getInstance().cancelLoadMesh (shapeName); // If the state is not Delete, or Error, then the mesh has not been added to the map. if(wsmmIt->second.State!=AsyncLoad_Delete && wsmmIt->second.State!=AsyncLoad_Error) { /* but it can still have been loaded: - just ended in above cancelLoadMesh() because it was the current task - ended in async far before this cancelLoadAsync() call, but processWaintingShape() still not called on it In this case we have to delete this shape, or even cancel texture upload! NB: don't forget that the load can still be a fail too.... */ IShape *shape= wsmmIt->second.ShapePtr; if(shape!=NULL && shape!=(IShape*)-1) { // this ptr should not be added to the map nlassert(ShapePtrToShapeInfo.find(shape)==ShapePtrToShapeInfo.end()); // Before deleting this shape, we must ensure it is not currently uploading texture if(wsmmIt->second.State==AsyncLoad_Texture) { /* if it was uploading a texture, then force him to end, else may have a bug in this scenario (very improbable, but possible I think): - a mesh is loaded asynchronously, and reference a texture "pipo.tga" - mesh load async is ended, and the texture is not found in the driver => texture generate()-d too - mesh state is AsyncLoad_Texture, and begin (but doesn't end) to upload the texture - another mesh is created syncrhonously using also this texture (thus found in driver, and so just referencing it, no generate) - the async mesh is then canceled, while the texture has not end to load! - the texture is still in memory (the sync mesh still point to it), but with partialy uploaded data! */ // \todo yoyo: should be very rare, and don't know if really happens. must do tests //forceEndUpLoadTexture(wsmmIt->second); } // then delete this shape delete shape; wsmmIt->second.ShapePtr= NULL; } } // erase this waiting shape WaitingShapes.erase (wsmmIt); // Delete the waiting shape } } } // *************************************************************************** bool CShapeBank::isShapeWaiting () { return !WaitingShapes.empty(); } // *************************************************************************** void CShapeBank::add (const string &shapeNameNotLwr, IShape* pShp) { nlassert(pShp); string shapeName= toLowerAscii(shapeNameNotLwr); // request a system mem geometry copy? if(pShp && _ShapeNeedingSystemGeometryCopy.find(shapeName)!=_ShapeNeedingSystemGeometryCopy.end()) { // make a copy of the geometry, in RAM pShp->buildSystemGeometry(); } // Is the shape name already used ? TShapeMap::iterator smIt = ShapeMap.find( shapeName ); if( smIt == ShapeMap.end() ) { // No ok so lets add the smart pointer CSmartPtr spShape = pShp; ShapeMap[shapeName] = spShape; // create the shape info CShapeInfo siTemp; siTemp.sShpName = shapeName; siTemp.pShpCache = getShapeCachePtrFromShapeName( shapeName ); // Is the shape has a valid shape cache ? if( siTemp.pShpCache == NULL ) { // No -> link to default (which do the UpdateShapeInfo) siTemp.pShpCache = getShapeCachePtrFromShapeCacheName( "default" ); // Add the shape to the default shape cache ShapePtrToShapeInfo[pShp]= siTemp; ShapeNameToShapeCacheName[shapeName]= "default"; } else { // Yes -> add or replace the shape info ShapePtrToShapeInfo[pShp] = siTemp; } } } // *************************************************************************** void CShapeBank::addShapeCache(const string &shapeCacheName) { TShapeCacheMap::iterator scmIt = ShapeCacheNameToShapeCache.find( shapeCacheName ); if( scmIt == ShapeCacheNameToShapeCache.end() ) { // Not found so add it ShapeCacheNameToShapeCache.insert(TShapeCacheMap::value_type(shapeCacheName,CShapeCache())); } } // *************************************************************************** void CShapeBank::removeShapeCache(const std::string &shapeCacheName) { if( shapeCacheName == "default" ) return; // Free the shape cache CShapeCache *pShpCache = getShapeCachePtrFromShapeCacheName( shapeCacheName ); if( pShpCache == NULL ) return; pShpCache->MaxSize = 0; checkShapeCache( pShpCache ); // Remove it ShapeCacheNameToShapeCache.erase( shapeCacheName ); // All links are redirected to the default cache TShapeCacheNameMap::iterator scnIt = ShapeNameToShapeCacheName.begin(); while( scnIt != ShapeNameToShapeCacheName.end() ) { if( scnIt->second == shapeCacheName ) scnIt->second = "default"; ++scnIt; } } // *************************************************************************** void CShapeBank::reset() { // Parse map ShapeCacheNameToShapeCache to delete all caches TShapeCacheMap::iterator scmIt = ShapeCacheNameToShapeCache.begin(); while( scmIt != ShapeCacheNameToShapeCache.end() ) { CShapeCache *pShpCache = getShapeCachePtrFromShapeCacheName( scmIt->first ); nlassert(pShpCache); pShpCache->MaxSize = 0; checkShapeCache( pShpCache ); ++scmIt; } ShapeNameToShapeCacheName.clear(); ShapeCacheNameToShapeCache.clear(); addShapeCache( "default" ); } // *************************************************************************** void CShapeBank::setShapeCacheSize(const string &shapeCacheName, sint32 maxSize) { TShapeCacheMap::iterator scmIt = ShapeCacheNameToShapeCache.find( shapeCacheName ); if( scmIt != ShapeCacheNameToShapeCache.end() ) { scmIt->second.MaxSize = maxSize; checkShapeCache(getShapeCachePtrFromShapeCacheName(shapeCacheName)); } } // *************************************************************************** sint CShapeBank::getShapeCacheFreeSpace(const std::string &shapeCacheName) const { TShapeCacheMap::const_iterator scmIt = ShapeCacheNameToShapeCache.find( shapeCacheName ); if( scmIt != ShapeCacheNameToShapeCache.end() ) { return scmIt->second.MaxSize - (sint)scmIt->second.Elements.size(); } return 0; } // *************************************************************************** void CShapeBank::linkShapeToShapeCache(const string &shapeNameNotLwr, const string &shapeCacheName) { string shapeName= toLowerAscii(shapeNameNotLwr); for(;;) { // Shape exist? IShape *shapePtr= getShapePtrFromShapeName(shapeName); if(shapePtr == NULL) // No, but still link the shape name to the shapeCache name. break; // Is the shape cache exist ? CShapeCache *shapeCachePtr = getShapeCachePtrFromShapeCacheName( shapeCacheName ); if( shapeCachePtr == NULL ) // abort, since cannot correctly link to a valid shapeCache return; // Try to set to the same shape Cache as before? CShapeInfo &shapeInfo= ShapePtrToShapeInfo[shapePtr]; if( shapeCachePtr == shapeInfo.pShpCache) // abort, since same cache name / cache ptr return; // If The shape is In the cache of another Shape Cache, abort. if( shapeInfo.isAdded ) // Abort, because impossible. return; // Is the shape is present ? // Yes -> Update the ShapeInfo shapeInfo.pShpCache= shapeCachePtr; break; } // change the cache name of the shape ShapeNameToShapeCacheName[shapeName] = shapeCacheName; } // *************************************************************************** CShapeBank::CShapeCache* CShapeBank::getShapeCachePtrFromShapePtr(IShape* pShp) { TShapeInfoMap::iterator scfpmIt = ShapePtrToShapeInfo.find( pShp ); if( scfpmIt != ShapePtrToShapeInfo.end() ) { return scfpmIt->second.pShpCache; } return NULL; } // *************************************************************************** IShape* CShapeBank::getShapePtrFromShapeName(const std::string &pShpName) { TShapeMap::iterator smIt = ShapeMap.find(pShpName); if( smIt != ShapeMap.end() ) { // TMP IShape *ptr = (IShape*)(smIt->second); return ptr; } return NULL; } // *************************************************************************** CShapeBank::CShapeCache* CShapeBank::getShapeCachePtrFromShapeCacheName(const string &shapeCacheName) { TShapeCacheMap::iterator scmIt = ShapeCacheNameToShapeCache.find( shapeCacheName ); if( scmIt != ShapeCacheNameToShapeCache.end()) { return &(scmIt->second); } return NULL; } // *************************************************************************** const string* CShapeBank::getShapeNameFromShapePtr(IShape* pShp) const { TShapeInfoMap::const_iterator scfpmIt = ShapePtrToShapeInfo.find( pShp ); if( scfpmIt != ShapePtrToShapeInfo.end() ) { return &(scfpmIt->second.sShpName); } return NULL; } // *************************************************************************** CShapeBank::CShapeCache* CShapeBank::getShapeCachePtrFromShapeName(const std::string &shapeName) { TShapeCacheNameMap::iterator scnIt = ShapeNameToShapeCacheName.find( shapeName ); if( scnIt != ShapeNameToShapeCacheName.end() ) { return getShapeCachePtrFromShapeCacheName(scnIt->second); } return NULL; } // *************************************************************************** void CShapeBank::checkShapeCache(CShapeCache* pShpCache) { if( pShpCache != NULL ) while( (sint)pShpCache->Elements.size() > pShpCache->MaxSize ) { // Suppress the last shape of the cache IShape *pShp = pShpCache->Elements.back(); // Physical suppression because we own the last smart pointer on the shape ShapeMap.erase(*getShapeNameFromShapePtr(pShp)); // delete information associated with the shape ShapePtrToShapeInfo.erase( pShp ); // remove from queue pShpCache->Elements.pop_back(); } } // *************************************************************************** bool CShapeBank::isShapeCache(const std::string &shapeCacheName) const { return ShapeCacheNameToShapeCache.find(shapeCacheName) != ShapeCacheNameToShapeCache.end(); } // *************************************************************************** void CShapeBank::preLoadShapes(const std::string &shapeCacheName, const std::vector &listFile, const std::string &wildCardNotLwr, NLMISC::IProgressCallback *progress, bool flushTextures /*= false*/, IDriver *drv /*= NULL*/) { // Abort if cache don't exist. if(!isShapeCache(shapeCacheName)) return; // lower case string wildCard= toLowerAscii(wildCardNotLwr); // For all files for(uint i=0;iprogress ((float)i/(float)listFile.size ()); string fileName= toLowerAscii(CFile::getFilename(listFile[i])); // if the file is ok for the wildCard, process it if( testWildCard(fileName.c_str(), wildCard.c_str()) ) { // link the shape to the shapeCache linkShapeToShapeCache(fileName, shapeCacheName); // If !present in the shapeBank if( getPresentState(fileName)==CShapeBank::NotPresent ) { // Don't load it if no more space in the cache if( getShapeCacheFreeSpace(shapeCacheName)>0 ) { // load it. load(fileName); // If success if( getPresentState(fileName)==CShapeBank::Present ) { // When a shape is first added to the bank, it is not in the cache. // add it and release it to force it to be in the cache. IShape *shp= addRef(fileName); if(shp) { //nlinfo("Loading %s", CPath::lookup(fileName.c_str(), false, false).c_str()); if (flushTextures && drv) { shp->flushTextures(*drv, 0); } release(shp); } } } } } } } // *************************************************************************** void CShapeBank::buildSystemGeometryForshape(const std::string &shapeName) { _ShapeNeedingSystemGeometryCopy.insert(toLowerAscii(shapeName)); } }