2018-10-01 01:33:25 +03:00
|
|
|
#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHTILESCACHE_H
|
|
|
|
#define OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHTILESCACHE_H
|
|
|
|
|
2022-06-17 00:28:44 +02:00
|
|
|
#include "agentbounds.hpp"
|
2021-07-14 00:03:10 +02:00
|
|
|
#include "preparednavmeshdata.hpp"
|
2018-10-01 01:33:25 +03:00
|
|
|
#include "recastmesh.hpp"
|
|
|
|
#include "tileposition.hpp"
|
|
|
|
|
|
|
|
#include <atomic>
|
2019-03-10 22:04:41 +03:00
|
|
|
#include <cassert>
|
2020-10-08 00:58:11 +02:00
|
|
|
#include <cstring>
|
2018-10-01 01:33:25 +03:00
|
|
|
#include <list>
|
|
|
|
#include <map>
|
|
|
|
#include <mutex>
|
2020-10-08 00:58:11 +02:00
|
|
|
#include <vector>
|
2018-10-01 01:33:25 +03:00
|
|
|
|
|
|
|
namespace DetourNavigator
|
|
|
|
{
|
2021-02-04 00:15:54 +01:00
|
|
|
struct RecastMeshData
|
|
|
|
{
|
2021-07-11 21:43:19 +02:00
|
|
|
Mesh mMesh;
|
2021-11-04 02:48:32 +01:00
|
|
|
std::vector<CellWater> mWater;
|
2021-07-14 20:57:52 +02:00
|
|
|
std::vector<Heightfield> mHeightfields;
|
|
|
|
std::vector<FlatHeightfield> mFlatHeightfields;
|
2021-02-04 00:15:54 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
inline bool operator<(const RecastMeshData& lhs, const RecastMeshData& rhs)
|
|
|
|
{
|
2021-07-14 20:57:52 +02:00
|
|
|
return std::tie(lhs.mMesh, lhs.mWater, lhs.mHeightfields, lhs.mFlatHeightfields)
|
|
|
|
< std::tie(rhs.mMesh, rhs.mWater, rhs.mHeightfields, rhs.mFlatHeightfields);
|
2021-02-04 00:15:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool operator<(const RecastMeshData& lhs, const RecastMesh& rhs)
|
|
|
|
{
|
2021-07-14 20:57:52 +02:00
|
|
|
return std::tie(lhs.mMesh, lhs.mWater, lhs.mHeightfields, lhs.mFlatHeightfields)
|
|
|
|
< std::tie(rhs.getMesh(), rhs.getWater(), rhs.getHeightfields(), rhs.getFlatHeightfields());
|
2021-02-04 00:15:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool operator<(const RecastMesh& lhs, const RecastMeshData& rhs)
|
|
|
|
{
|
2021-07-14 20:57:52 +02:00
|
|
|
return std::tie(lhs.getMesh(), lhs.getWater(), lhs.getHeightfields(), lhs.getFlatHeightfields())
|
|
|
|
< std::tie(rhs.mMesh, rhs.mWater, rhs.mHeightfields, rhs.mFlatHeightfields);
|
2021-02-04 00:15:54 +01:00
|
|
|
}
|
|
|
|
|
2022-08-28 14:52:12 +02:00
|
|
|
struct NavMeshTilesCacheStats;
|
|
|
|
|
2018-10-01 01:33:25 +03:00
|
|
|
class NavMeshTilesCache
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
struct Item
|
|
|
|
{
|
|
|
|
std::atomic<std::int64_t> mUseCount;
|
2022-06-17 00:28:44 +02:00
|
|
|
AgentBounds mAgentBounds;
|
2018-10-01 01:33:25 +03:00
|
|
|
TilePosition mChangedTile;
|
2021-07-14 00:03:10 +02:00
|
|
|
RecastMeshData mRecastMeshData;
|
|
|
|
std::unique_ptr<PreparedNavMeshData> mPreparedNavMeshData;
|
2021-02-04 00:15:54 +01:00
|
|
|
std::size_t mSize;
|
2018-10-01 01:33:25 +03:00
|
|
|
|
2022-06-17 00:28:44 +02:00
|
|
|
Item(const AgentBounds& agentBounds, const TilePosition& changedTile, RecastMeshData&& recastMeshData,
|
2021-07-14 00:03:10 +02:00
|
|
|
std::size_t size)
|
2018-10-01 01:33:25 +03:00
|
|
|
: mUseCount(0)
|
2022-06-17 00:28:44 +02:00
|
|
|
, mAgentBounds(agentBounds)
|
2018-10-01 01:33:25 +03:00
|
|
|
, mChangedTile(changedTile)
|
2021-07-14 00:03:10 +02:00
|
|
|
, mRecastMeshData(std::move(recastMeshData))
|
2021-02-04 00:15:54 +01:00
|
|
|
, mSize(size)
|
2018-10-01 01:33:25 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
using ItemIterator = std::list<Item>::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)
|
|
|
|
{
|
2018-12-10 23:51:15 +03:00
|
|
|
other.mOwner = nullptr;
|
2018-10-01 01:33:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
~Value()
|
|
|
|
{
|
2018-12-10 23:51:15 +03:00
|
|
|
if (mOwner)
|
2018-10-01 01:33:25 +03:00
|
|
|
mOwner->releaseItem(mIterator);
|
|
|
|
}
|
|
|
|
|
|
|
|
Value& operator=(const Value& other) = delete;
|
|
|
|
|
|
|
|
Value& operator=(Value&& other)
|
|
|
|
{
|
2018-12-10 23:51:15 +03:00
|
|
|
if (mOwner)
|
2018-10-01 01:33:25 +03:00
|
|
|
mOwner->releaseItem(mIterator);
|
|
|
|
|
|
|
|
mOwner = other.mOwner;
|
|
|
|
mIterator = other.mIterator;
|
|
|
|
|
2018-12-10 23:51:15 +03:00
|
|
|
other.mOwner = nullptr;
|
2018-10-01 01:33:25 +03:00
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2021-07-14 00:03:10 +02:00
|
|
|
const PreparedNavMeshData& get() const { return *mIterator->mPreparedNavMeshData; }
|
2018-10-01 01:33:25 +03:00
|
|
|
|
|
|
|
operator bool() const { return mOwner; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
NavMeshTilesCache* mOwner;
|
|
|
|
ItemIterator mIterator;
|
|
|
|
};
|
|
|
|
|
|
|
|
NavMeshTilesCache(const std::size_t maxNavMeshDataSize);
|
|
|
|
|
2022-06-17 00:28:44 +02:00
|
|
|
Value get(const AgentBounds& agentBounds, const TilePosition& changedTile, const RecastMesh& recastMesh);
|
2018-10-01 01:33:25 +03:00
|
|
|
|
2022-06-17 00:28:44 +02:00
|
|
|
Value set(const AgentBounds& agentBounds, const TilePosition& changedTile, const RecastMesh& recastMesh,
|
2021-07-14 00:03:10 +02:00
|
|
|
std::unique_ptr<PreparedNavMeshData>&& value);
|
2018-10-01 01:33:25 +03:00
|
|
|
|
2022-08-28 14:52:12 +02:00
|
|
|
NavMeshTilesCacheStats getStats() const;
|
2021-03-25 00:59:44 +01:00
|
|
|
|
2018-10-01 01:33:25 +03:00
|
|
|
private:
|
2019-03-17 20:18:53 +03:00
|
|
|
mutable std::mutex mMutex;
|
2018-10-01 01:33:25 +03:00
|
|
|
std::size_t mMaxNavMeshDataSize;
|
|
|
|
std::size_t mUsedNavMeshDataSize;
|
|
|
|
std::size_t mFreeNavMeshDataSize;
|
2021-02-04 00:18:25 +01:00
|
|
|
std::size_t mHitCount;
|
|
|
|
std::size_t mGetCount;
|
2018-10-01 01:33:25 +03:00
|
|
|
std::list<Item> mBusyItems;
|
|
|
|
std::list<Item> mFreeItems;
|
2022-06-17 00:28:44 +02:00
|
|
|
std::map<std::tuple<AgentBounds, TilePosition, std::reference_wrapper<const RecastMeshData>>, ItemIterator,
|
|
|
|
std::less<>>
|
|
|
|
mValues;
|
2018-10-01 01:33:25 +03:00
|
|
|
|
|
|
|
void removeLeastRecentlyUsed();
|
|
|
|
|
|
|
|
void acquireItemUnsafe(ItemIterator iterator);
|
|
|
|
|
|
|
|
void releaseItem(ItemIterator iterator);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|