mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-19 19:53:53 +00:00
1a12c453d6
Actors may have different collision shapes. Currently there are axis-aligned bounding boxes and rotating bounding boxes. With AABB it's required to use bounding cylinder for navmesh agent to avoid providing paths where actor can't pass. But for rotating bounding boxes cylinder with diameter equal to the front face width should be used to not reduce of available paths. For example rats have rotating bounding box as collision shape because of the difference between front and side faces width. * Add agent bounds to navmesh tile db cache key. This is required to distinguish tiles for agents with different bounds. * Increase navmesh version because navmesh tile db cache key and data has changed. * Move navmesh version to the code to avoid misconfiguration by users. * Fix all places where wrong half extents were used for pathfinding.
125 lines
4.4 KiB
C++
125 lines
4.4 KiB
C++
#include "navmeshtilescache.hpp"
|
|
|
|
#include <osg/Stats>
|
|
|
|
#include <cstring>
|
|
|
|
namespace DetourNavigator
|
|
{
|
|
NavMeshTilesCache::NavMeshTilesCache(const std::size_t maxNavMeshDataSize)
|
|
: mMaxNavMeshDataSize(maxNavMeshDataSize), mUsedNavMeshDataSize(0), mFreeNavMeshDataSize(0),
|
|
mHitCount(0), mGetCount(0) {}
|
|
|
|
NavMeshTilesCache::Value NavMeshTilesCache::get(const AgentBounds& agentBounds, const TilePosition& changedTile,
|
|
const RecastMesh& recastMesh)
|
|
{
|
|
const std::lock_guard<std::mutex> lock(mMutex);
|
|
|
|
++mGetCount;
|
|
|
|
const auto tile = mValues.find(std::make_tuple(agentBounds, changedTile, recastMesh));
|
|
if (tile == mValues.end())
|
|
return Value();
|
|
|
|
acquireItemUnsafe(tile->second);
|
|
|
|
++mHitCount;
|
|
|
|
return Value(*this, tile->second);
|
|
}
|
|
|
|
NavMeshTilesCache::Value NavMeshTilesCache::set(const AgentBounds& agentBounds, const TilePosition& changedTile,
|
|
const RecastMesh& recastMesh, std::unique_ptr<PreparedNavMeshData>&& value)
|
|
{
|
|
const auto itemSize = sizeof(RecastMesh) + getSize(recastMesh)
|
|
+ (value == nullptr ? 0 : sizeof(PreparedNavMeshData) + getSize(*value));
|
|
|
|
const std::lock_guard<std::mutex> lock(mMutex);
|
|
|
|
if (itemSize > mFreeNavMeshDataSize + (mMaxNavMeshDataSize - mUsedNavMeshDataSize))
|
|
return Value();
|
|
|
|
while (!mFreeItems.empty() && mUsedNavMeshDataSize + itemSize > mMaxNavMeshDataSize)
|
|
removeLeastRecentlyUsed();
|
|
|
|
RecastMeshData key {recastMesh.getMesh(), recastMesh.getWater(),
|
|
recastMesh.getHeightfields(), recastMesh.getFlatHeightfields()};
|
|
|
|
const auto iterator = mFreeItems.emplace(mFreeItems.end(), agentBounds, changedTile, std::move(key), itemSize);
|
|
const auto emplaced = mValues.emplace(std::make_tuple(agentBounds, changedTile, std::cref(iterator->mRecastMeshData)), iterator);
|
|
|
|
if (!emplaced.second)
|
|
{
|
|
mFreeItems.erase(iterator);
|
|
acquireItemUnsafe(emplaced.first->second);
|
|
++mGetCount;
|
|
++mHitCount;
|
|
return Value(*this, emplaced.first->second);
|
|
}
|
|
|
|
iterator->mPreparedNavMeshData = std::move(value);
|
|
++iterator->mUseCount;
|
|
mUsedNavMeshDataSize += itemSize;
|
|
mBusyItems.splice(mBusyItems.end(), mFreeItems, iterator);
|
|
|
|
return Value(*this, iterator);
|
|
}
|
|
|
|
NavMeshTilesCache::Stats NavMeshTilesCache::getStats() const
|
|
{
|
|
Stats result;
|
|
{
|
|
const std::lock_guard<std::mutex> lock(mMutex);
|
|
result.mNavMeshCacheSize = mUsedNavMeshDataSize;
|
|
result.mUsedNavMeshTiles = mBusyItems.size();
|
|
result.mCachedNavMeshTiles = mFreeItems.size();
|
|
result.mHitCount = mHitCount;
|
|
result.mGetCount = mGetCount;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void reportStats(const NavMeshTilesCache::Stats& stats, unsigned int frameNumber, osg::Stats& out)
|
|
{
|
|
out.setAttribute(frameNumber, "NavMesh CacheSize", static_cast<double>(stats.mNavMeshCacheSize));
|
|
out.setAttribute(frameNumber, "NavMesh UsedTiles", static_cast<double>(stats.mUsedNavMeshTiles));
|
|
out.setAttribute(frameNumber, "NavMesh CachedTiles", static_cast<double>(stats.mCachedNavMeshTiles));
|
|
if (stats.mGetCount > 0)
|
|
out.setAttribute(frameNumber, "NavMesh CacheHitRate", static_cast<double>(stats.mHitCount) / stats.mGetCount * 100.0);
|
|
}
|
|
|
|
void NavMeshTilesCache::removeLeastRecentlyUsed()
|
|
{
|
|
const auto& item = mFreeItems.back();
|
|
|
|
const auto value = mValues.find(std::make_tuple(item.mAgentBounds, item.mChangedTile, std::cref(item.mRecastMeshData)));
|
|
if (value == mValues.end())
|
|
return;
|
|
|
|
mUsedNavMeshDataSize -= item.mSize;
|
|
mFreeNavMeshDataSize -= item.mSize;
|
|
|
|
mValues.erase(value);
|
|
mFreeItems.pop_back();
|
|
}
|
|
|
|
void NavMeshTilesCache::acquireItemUnsafe(ItemIterator iterator)
|
|
{
|
|
if (++iterator->mUseCount > 1)
|
|
return;
|
|
|
|
mBusyItems.splice(mBusyItems.end(), mFreeItems, iterator);
|
|
mFreeNavMeshDataSize -= iterator->mSize;
|
|
}
|
|
|
|
void NavMeshTilesCache::releaseItem(ItemIterator iterator)
|
|
{
|
|
if (--iterator->mUseCount > 0)
|
|
return;
|
|
|
|
const std::lock_guard<std::mutex> lock(mMutex);
|
|
|
|
mFreeItems.splice(mFreeItems.begin(), mBusyItems, iterator);
|
|
mFreeNavMeshDataSize += iterator->mSize;
|
|
}
|
|
}
|