#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_ASYNCNAVMESHUPDATER_H #define OPENMW_COMPONENTS_DETOURNAVIGATOR_ASYNCNAVMESHUPDATER_H #include "navmeshcacheitem.hpp" #include "tilecachedrecastmeshmanager.hpp" #include "tileposition.hpp" #include #include #include #include #include #include #include #include #include #include class dtNavMesh; namespace DetourNavigator { enum class ChangeType { remove = 0, mixed = 1, add = 2, }; class AsyncNavMeshUpdater { public: AsyncNavMeshUpdater(const Settings& settings, TileCachedRecastMeshManager& recastMeshManager); ~AsyncNavMeshUpdater(); void post(const osg::Vec3f& agentHalfExtents, const std::shared_ptr& mNavMeshCacheItem, const TilePosition& playerTile, const std::map& changedTiles); void wait(); private: struct Job { osg::Vec3f mAgentHalfExtents; std::shared_ptr mNavMeshCacheItem; TilePosition mChangedTile; std::tuple mPriority; friend inline bool operator <(const Job& lhs, const Job& rhs) { return lhs.mPriority > rhs.mPriority; } }; using Jobs = std::priority_queue>; std::reference_wrapper mSettings; std::reference_wrapper mRecastMeshManager; std::atomic_bool mShouldStop; std::mutex mMutex; std::condition_variable mHasJob; std::condition_variable mDone; Jobs mJobs; std::map> mPushed; std::mutex mPlayerTileMutex; TilePosition mPlayerTile; std::mutex mFirstStartMutex; boost::optional mFirstStart; std::thread mThread; void process() throw(); void processJob(const Job& job); boost::optional getNextJob(); void notifyHasJob(); void writeDebugFiles(const Job& job, const RecastMesh* recastMesh) const; std::chrono::steady_clock::time_point getFirstStart(); void setFirstStart(const std::chrono::steady_clock::time_point& value); TilePosition getPlayerTile(); void setPlayerTile(const TilePosition& value); }; } #endif