#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHTILESCACHE_H #define OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHTILESCACHE_H #include "offmeshconnection.hpp" #include "navmeshdata.hpp" #include "recastmesh.hpp" #include "tileposition.hpp" #include #include #include #include namespace DetourNavigator { struct NavMeshDataRef { unsigned char* mValue; int mSize; }; class NavMeshTilesCache { public: struct Item { std::atomic mUseCount; osg::Vec3f mAgentHalfExtents; TilePosition mChangedTile; std::string mNavMeshKey; NavMeshData mNavMeshData; Item(const osg::Vec3f& agentHalfExtents, const TilePosition& changedTile, std::string navMeshKey) : mUseCount(0) , mAgentHalfExtents(agentHalfExtents) , mChangedTile(changedTile) , mNavMeshKey(std::move(navMeshKey)) {} }; using ItemIterator = std::list::iterator; class Value { public: Value() : mOwner(nullptr), mIterator() {} Value(NavMeshTilesCache& owner, ItemIterator iterator) : mOwner(&owner), mIterator(iterator) { } Value(const Value& other) = delete; Value(Value&& other) : mOwner(other.mOwner), mIterator(other.mIterator) { other.mOwner = nullptr; } ~Value() { if (mOwner) mOwner->releaseItem(mIterator); } Value& operator =(const Value& other) = delete; Value& operator =(Value&& other) { if (mOwner) mOwner->releaseItem(mIterator); mOwner = other.mOwner; mIterator = other.mIterator; other.mOwner = nullptr; return *this; } NavMeshDataRef get() const { return NavMeshDataRef {mIterator->mNavMeshData.mValue.get(), mIterator->mNavMeshData.mSize}; } operator bool() const { return mOwner; } private: NavMeshTilesCache* mOwner; ItemIterator mIterator; }; NavMeshTilesCache(const std::size_t maxNavMeshDataSize); Value get(const osg::Vec3f& agentHalfExtents, const TilePosition& changedTile, const RecastMesh& recastMesh, const std::vector& offMeshConnections); Value set(const osg::Vec3f& agentHalfExtents, const TilePosition& changedTile, const RecastMesh& recastMesh, const std::vector& offMeshConnections, NavMeshData&& value); private: struct TileMap { std::map Map; }; std::mutex mMutex; std::size_t mMaxNavMeshDataSize; std::size_t mUsedNavMeshDataSize; std::size_t mFreeNavMeshDataSize; std::list mBusyItems; std::list mFreeItems; std::map> mValues; void removeLeastRecentlyUsed(); void acquireItemUnsafe(ItemIterator iterator); void releaseItem(ItemIterator iterator); static std::size_t getSize(const Item& item) { return static_cast(item.mNavMeshData.mSize) + 2 * item.mNavMeshKey.size(); } }; } #endif