#include "cellpreloader.hpp" #include #include #include #include #include "../mwbase/environment.hpp" #include "../mwbase/world.hpp" #include "cellstore.hpp" #include "manualref.hpp" #include "class.hpp" namespace MWWorld { struct ListModelsVisitor { ListModelsVisitor(std::vector& out) : mOut(out) { } virtual bool operator()(const MWWorld::ConstPtr& ptr) { std::string model = ptr.getClass().getModel(ptr); if (!model.empty()) mOut.push_back(model); return true; } std::vector& mOut; }; /// Worker thread item: preload models in a cell. class PreloadItem : public SceneUtil::WorkItem { public: /// Constructor to be called from the main thread. PreloadItem(const MWWorld::CellStore* cell, Resource::SceneManager* sceneManager, Resource::BulletShapeManager* bulletShapeManager) : mSceneManager(sceneManager) , mBulletShapeManager(bulletShapeManager) { if (cell->getState() == MWWorld::CellStore::State_Loaded) { ListModelsVisitor visitor (mMeshes); cell->forEachConst(visitor); } else { const std::vector& objectIds = cell->getPreloadedIds(); // could possibly build the model list in the worker thread if we manage to make the Store thread safe for (std::vector::const_iterator it = objectIds.begin(); it != objectIds.end(); ++it) { MWWorld::ManualRef ref(MWBase::Environment::get().getWorld()->getStore(), *it); std::string model = ref.getPtr().getClass().getModel(ref.getPtr()); if (!model.empty()) mMeshes.push_back(model); } } } /// Preload work to be called from the worker thread. virtual void doWork() { // TODO: make CellStore::loadRefs thread safe so we can call it from here osg::Timer preloadTimer; for (MeshList::const_iterator it = mMeshes.begin(); it != mMeshes.end(); ++it) { try { mPreloadedNodes.push_back(mSceneManager->getTemplate(*it)); mPreloadedShapes.push_back(mBulletShapeManager->getShape(*it)); // TODO: do a createInstance() and hold on to it since we can make use of it when the cell goes active } catch (std::exception& e) { // ignore error for now, would spam the log too much // error will be shown when visiting the cell } } std::cout << "preloaded " << mPreloadedNodes.size() << " nodes in " << preloadTimer.time_m() << std::endl; } private: typedef std::vector MeshList; MeshList mMeshes; Resource::SceneManager* mSceneManager; Resource::BulletShapeManager* mBulletShapeManager; // keep a ref to the loaded object to make sure it stays loaded as long as this cell is in the preloaded state std::vector > mPreloadedNodes; std::vector > mPreloadedShapes; }; /// Worker thread item: update the resource system's cache, effectively deleting unused entries. class UpdateCacheItem : public SceneUtil::WorkItem { public: UpdateCacheItem(Resource::ResourceSystem* resourceSystem, double referenceTime) : mReferenceTime(referenceTime) , mResourceSystem(resourceSystem) { } virtual void doWork() { osg::Timer timer; mResourceSystem->updateCache(mReferenceTime); std::cout << "cleared cache in " << timer.time_m() << std::endl; } private: double mReferenceTime; Resource::ResourceSystem* mResourceSystem; }; CellPreloader::CellPreloader(Resource::ResourceSystem* resourceSystem, Resource::BulletShapeManager* bulletShapeManager) : mResourceSystem(resourceSystem) , mBulletShapeManager(bulletShapeManager) { mWorkQueue = new SceneUtil::WorkQueue; } void CellPreloader::preload(const CellStore *cell, double timestamp) { if (!mWorkQueue) { std::cerr << "can't preload, no work queue set " << std::endl; return; } if (cell->getState() == CellStore::State_Unloaded) { std::cerr << "can't preload objects for unloaded cell" << std::endl; return; } PreloadMap::iterator found = mPreloadCells.find(cell); if (found != mPreloadCells.end()) { // already preloaded, nothing to do other than updating the timestamp found->second.mTimeStamp = timestamp; return; } osg::ref_ptr item (new PreloadItem(cell, mResourceSystem->getSceneManager(), mBulletShapeManager)); mWorkQueue->addWorkItem(item); mPreloadCells[cell] = PreloadEntry(timestamp, item); } void CellPreloader::updateCache(double timestamp) { // time (in seconds) to keep a preloaded cell in cache after it's no longer needed // additionally we could support a minimum/maximum size for the cache const double expiryTime = 60.0; for (PreloadMap::iterator it = mPreloadCells.begin(); it != mPreloadCells.end();) { if (it->second.mTimeStamp < timestamp - expiryTime) mPreloadCells.erase(it++); else ++it; } // the resource cache is cleared from the worker thread so that we're not holding up the main thread with delete operations mWorkQueue->addWorkItem(new UpdateCacheItem(mResourceSystem, timestamp)); } void CellPreloader::setWorkQueue(osg::ref_ptr workQueue) { mWorkQueue = workQueue; } }