mirror of
https://github.com/OpenMW/openmw.git
synced 2025-04-03 21:06:41 +00:00
Set proper max tiles on initializing navmesh settings
This commit is contained in:
parent
55c5f2112b
commit
a1438f65fe
6 changed files with 80 additions and 66 deletions
|
@ -180,13 +180,10 @@ namespace DetourNavigator
|
||||||
if (!playerTileChanged && changedTiles.empty())
|
if (!playerTileChanged && changedTiles.empty())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
const int maxTiles
|
|
||||||
= std::min(mSettings.get().mMaxTilesNumber, navMeshCacheItem->lockConst()->getImpl().getParams()->maxTiles);
|
|
||||||
|
|
||||||
std::unique_lock lock(mMutex);
|
std::unique_lock lock(mMutex);
|
||||||
|
|
||||||
if (playerTileChanged)
|
if (playerTileChanged)
|
||||||
updateJobs(mWaiting, playerTile, maxTiles);
|
updateJobs(mWaiting, playerTile, mSettings.get().mMaxTilesNumber);
|
||||||
|
|
||||||
for (const auto& [changedTile, changeType] : changedTiles)
|
for (const auto& [changedTile, changeType] : changedTiles)
|
||||||
{
|
{
|
||||||
|
@ -221,7 +218,7 @@ namespace DetourNavigator
|
||||||
lock.unlock();
|
lock.unlock();
|
||||||
|
|
||||||
if (playerTileChanged && mDbWorker != nullptr)
|
if (playerTileChanged && mDbWorker != nullptr)
|
||||||
mDbWorker->updateJobs(playerTile, maxTiles);
|
mDbWorker->updateJobs(playerTile, mSettings.get().mMaxTilesNumber);
|
||||||
}
|
}
|
||||||
|
|
||||||
void AsyncNavMeshUpdater::wait(WaitConditionType waitConditionType, Loading::Listener* listener)
|
void AsyncNavMeshUpdater::wait(WaitConditionType waitConditionType, Loading::Listener* listener)
|
||||||
|
@ -376,10 +373,8 @@ namespace DetourNavigator
|
||||||
return JobStatus::Done;
|
return JobStatus::Done;
|
||||||
|
|
||||||
const auto playerTile = *mPlayerTile.lockConst();
|
const auto playerTile = *mPlayerTile.lockConst();
|
||||||
const int maxTiles
|
|
||||||
= std::min(mSettings.get().mMaxTilesNumber, navMeshCacheItem->lockConst()->getImpl().getParams()->maxTiles);
|
|
||||||
|
|
||||||
if (!shouldAddTile(job.mChangedTile, playerTile, maxTiles))
|
if (!shouldAddTile(job.mChangedTile, playerTile, mSettings.get().mMaxTilesNumber))
|
||||||
{
|
{
|
||||||
Log(Debug::Debug) << "Ignore add tile by job " << job.mId << ": too far from player";
|
Log(Debug::Debug) << "Ignore add tile by job " << job.mId << ": too far from player";
|
||||||
navMeshCacheItem->lock()->removeTile(job.mChangedTile);
|
navMeshCacheItem->lock()->removeTile(job.mChangedTile);
|
||||||
|
|
|
@ -480,15 +480,6 @@ namespace DetourNavigator
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <class T>
|
|
||||||
unsigned long getMinValuableBitsNumber(const T value)
|
|
||||||
{
|
|
||||||
unsigned long power = 0;
|
|
||||||
while (power < sizeof(T) * 8 && (static_cast<T>(1) << power) < value)
|
|
||||||
++power;
|
|
||||||
return power;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::pair<float, float> getBoundsByZ(
|
std::pair<float, float> getBoundsByZ(
|
||||||
const RecastMesh& recastMesh, float agentHalfExtentsZ, const RecastSettings& settings)
|
const RecastMesh& recastMesh, float agentHalfExtentsZ, const RecastSettings& settings)
|
||||||
{
|
{
|
||||||
|
@ -528,10 +519,7 @@ namespace DetourNavigator
|
||||||
return { minZ, maxZ };
|
return { minZ, maxZ };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} // namespace DetourNavigator
|
|
||||||
|
|
||||||
namespace DetourNavigator
|
|
||||||
{
|
|
||||||
std::unique_ptr<PreparedNavMeshData> prepareNavMeshTileData(const RecastMesh& recastMesh,
|
std::unique_ptr<PreparedNavMeshData> prepareNavMeshTileData(const RecastMesh& recastMesh,
|
||||||
std::string_view worldspace, const TilePosition& tilePosition, const AgentBounds& agentBounds,
|
std::string_view worldspace, const TilePosition& tilePosition, const AgentBounds& agentBounds,
|
||||||
const RecastSettings& settings)
|
const RecastSettings& settings)
|
||||||
|
@ -621,22 +609,12 @@ namespace DetourNavigator
|
||||||
|
|
||||||
void initEmptyNavMesh(const Settings& settings, dtNavMesh& navMesh)
|
void initEmptyNavMesh(const Settings& settings, dtNavMesh& navMesh)
|
||||||
{
|
{
|
||||||
// Max tiles and max polys affect how the tile IDs are caculated.
|
|
||||||
// There are 22 bits available for identifying a tile and a polygon.
|
|
||||||
const int polysAndTilesBits = 22;
|
|
||||||
const auto polysBits = getMinValuableBitsNumber(settings.mDetour.mMaxPolys);
|
|
||||||
|
|
||||||
if (polysBits >= polysAndTilesBits)
|
|
||||||
throw InvalidArgument("Too many polygons per tile");
|
|
||||||
|
|
||||||
const auto tilesBits = polysAndTilesBits - polysBits;
|
|
||||||
|
|
||||||
dtNavMeshParams params;
|
dtNavMeshParams params;
|
||||||
std::fill_n(params.orig, 3, 0.0f);
|
std::fill_n(params.orig, 3, 0.0f);
|
||||||
params.tileWidth = settings.mRecast.mTileSize * settings.mRecast.mCellSize;
|
params.tileWidth = settings.mRecast.mTileSize * settings.mRecast.mCellSize;
|
||||||
params.tileHeight = settings.mRecast.mTileSize * settings.mRecast.mCellSize;
|
params.tileHeight = settings.mRecast.mTileSize * settings.mRecast.mCellSize;
|
||||||
params.maxTiles = 1 << tilesBits;
|
params.maxTiles = settings.mMaxTilesNumber;
|
||||||
params.maxPolys = 1 << polysBits;
|
params.maxPolys = settings.mDetour.mMaxPolys;
|
||||||
|
|
||||||
const auto status = navMesh.init(¶ms);
|
const auto status = navMesh.init(¶ms);
|
||||||
|
|
||||||
|
|
|
@ -181,7 +181,7 @@ namespace DetourNavigator
|
||||||
{
|
{
|
||||||
const auto locked = cached->lockConst();
|
const auto locked = cached->lockConst();
|
||||||
const auto& navMesh = locked->getImpl();
|
const auto& navMesh = locked->getImpl();
|
||||||
const auto maxTiles = std::min(mSettings.mMaxTilesNumber, navMesh.getParams()->maxTiles);
|
const int maxTiles = mSettings.mMaxTilesNumber;
|
||||||
getTilesPositions(range, [&](const TilePosition& tile) {
|
getTilesPositions(range, [&](const TilePosition& tile) {
|
||||||
if (changedTiles.find(tile) != changedTiles.end())
|
if (changedTiles.find(tile) != changedTiles.end())
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -3,41 +3,81 @@
|
||||||
#include <components/misc/constants.hpp>
|
#include <components/misc/constants.hpp>
|
||||||
#include <components/settings/values.hpp>
|
#include <components/settings/values.hpp>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
namespace DetourNavigator
|
namespace DetourNavigator
|
||||||
{
|
{
|
||||||
RecastSettings makeRecastSettingsFromSettingsManager()
|
namespace
|
||||||
{
|
{
|
||||||
RecastSettings result;
|
struct NavMeshLimits
|
||||||
|
{
|
||||||
|
int mMaxTiles;
|
||||||
|
int mMaxPolys;
|
||||||
|
};
|
||||||
|
|
||||||
result.mBorderSize = ::Settings::navigator().mBorderSize;
|
template <class T>
|
||||||
result.mCellHeight = ::Settings::navigator().mCellHeight;
|
unsigned long getMinValuableBitsNumber(const T value)
|
||||||
result.mCellSize = ::Settings::navigator().mCellSize;
|
{
|
||||||
result.mDetailSampleDist = ::Settings::navigator().mDetailSampleDist;
|
unsigned long power = 0;
|
||||||
result.mDetailSampleMaxError = ::Settings::navigator().mDetailSampleMaxError;
|
while (power < sizeof(T) * 8 && (static_cast<T>(1) << power) < value)
|
||||||
result.mMaxClimb = Constants::sStepSizeUp;
|
++power;
|
||||||
result.mMaxSimplificationError = ::Settings::navigator().mMaxSimplificationError;
|
return power;
|
||||||
result.mMaxSlope = Constants::sMaxSlope;
|
}
|
||||||
result.mRecastScaleFactor = ::Settings::navigator().mRecastScaleFactor;
|
|
||||||
result.mSwimHeightScale = 0;
|
|
||||||
result.mMaxEdgeLen = ::Settings::navigator().mMaxEdgeLen;
|
|
||||||
result.mMaxVertsPerPoly = ::Settings::navigator().mMaxVertsPerPoly;
|
|
||||||
result.mRegionMergeArea = ::Settings::navigator().mRegionMergeArea;
|
|
||||||
result.mRegionMinArea = ::Settings::navigator().mRegionMinArea;
|
|
||||||
result.mTileSize = ::Settings::navigator().mTileSize;
|
|
||||||
|
|
||||||
return result;
|
NavMeshLimits getNavMeshTileLimits(const DetourSettings& settings)
|
||||||
}
|
{
|
||||||
|
// Max tiles and max polys affect how the tile IDs are caculated.
|
||||||
|
// There are 22 bits available for identifying a tile and a polygon.
|
||||||
|
constexpr int polysAndTilesBits = 22;
|
||||||
|
const unsigned long polysBits = getMinValuableBitsNumber(settings.mMaxPolys);
|
||||||
|
|
||||||
DetourSettings makeDetourSettingsFromSettingsManager()
|
if (polysBits >= polysAndTilesBits)
|
||||||
{
|
throw std::invalid_argument("Too many polygons per tile: " + std::to_string(settings.mMaxPolys));
|
||||||
DetourSettings result;
|
|
||||||
|
|
||||||
result.mMaxNavMeshQueryNodes = ::Settings::navigator().mMaxNavMeshQueryNodes;
|
const unsigned long tilesBits = polysAndTilesBits - polysBits;
|
||||||
result.mMaxPolys = ::Settings::navigator().mMaxPolygonsPerTile;
|
|
||||||
result.mMaxPolygonPathSize = ::Settings::navigator().mMaxPolygonPathSize;
|
|
||||||
result.mMaxSmoothPathSize = ::Settings::navigator().mMaxSmoothPathSize;
|
|
||||||
|
|
||||||
return result;
|
return NavMeshLimits{
|
||||||
|
.mMaxTiles = static_cast<int>(1 << tilesBits),
|
||||||
|
.mMaxPolys = static_cast<int>(1 << polysBits),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
RecastSettings makeRecastSettingsFromSettingsManager()
|
||||||
|
{
|
||||||
|
RecastSettings result;
|
||||||
|
|
||||||
|
result.mBorderSize = ::Settings::navigator().mBorderSize;
|
||||||
|
result.mCellHeight = ::Settings::navigator().mCellHeight;
|
||||||
|
result.mCellSize = ::Settings::navigator().mCellSize;
|
||||||
|
result.mDetailSampleDist = ::Settings::navigator().mDetailSampleDist;
|
||||||
|
result.mDetailSampleMaxError = ::Settings::navigator().mDetailSampleMaxError;
|
||||||
|
result.mMaxClimb = Constants::sStepSizeUp;
|
||||||
|
result.mMaxSimplificationError = ::Settings::navigator().mMaxSimplificationError;
|
||||||
|
result.mMaxSlope = Constants::sMaxSlope;
|
||||||
|
result.mRecastScaleFactor = ::Settings::navigator().mRecastScaleFactor;
|
||||||
|
result.mSwimHeightScale = 0;
|
||||||
|
result.mMaxEdgeLen = ::Settings::navigator().mMaxEdgeLen;
|
||||||
|
result.mMaxVertsPerPoly = ::Settings::navigator().mMaxVertsPerPoly;
|
||||||
|
result.mRegionMergeArea = ::Settings::navigator().mRegionMergeArea;
|
||||||
|
result.mRegionMinArea = ::Settings::navigator().mRegionMinArea;
|
||||||
|
result.mTileSize = ::Settings::navigator().mTileSize;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
DetourSettings makeDetourSettingsFromSettingsManager()
|
||||||
|
{
|
||||||
|
DetourSettings result;
|
||||||
|
|
||||||
|
result.mMaxNavMeshQueryNodes = ::Settings::navigator().mMaxNavMeshQueryNodes;
|
||||||
|
result.mMaxPolys = ::Settings::navigator().mMaxPolygonsPerTile;
|
||||||
|
result.mMaxPolygonPathSize = ::Settings::navigator().mMaxPolygonPathSize;
|
||||||
|
result.mMaxSmoothPathSize = ::Settings::navigator().mMaxSmoothPathSize;
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Settings makeSettingsFromSettingsManager()
|
Settings makeSettingsFromSettingsManager()
|
||||||
|
@ -46,7 +86,12 @@ namespace DetourNavigator
|
||||||
|
|
||||||
result.mRecast = makeRecastSettingsFromSettingsManager();
|
result.mRecast = makeRecastSettingsFromSettingsManager();
|
||||||
result.mDetour = makeDetourSettingsFromSettingsManager();
|
result.mDetour = makeDetourSettingsFromSettingsManager();
|
||||||
result.mMaxTilesNumber = ::Settings::navigator().mMaxTilesNumber;
|
|
||||||
|
const NavMeshLimits limits = getNavMeshTileLimits(result.mDetour);
|
||||||
|
|
||||||
|
result.mDetour.mMaxPolys = limits.mMaxPolys;
|
||||||
|
|
||||||
|
result.mMaxTilesNumber = std::min(limits.mMaxTiles, ::Settings::navigator().mMaxTilesNumber.get());
|
||||||
result.mWaitUntilMinDistanceToPlayer = ::Settings::navigator().mWaitUntilMinDistanceToPlayer;
|
result.mWaitUntilMinDistanceToPlayer = ::Settings::navigator().mWaitUntilMinDistanceToPlayer;
|
||||||
result.mAsyncNavMeshUpdaterThreads = ::Settings::navigator().mAsyncNavMeshUpdaterThreads;
|
result.mAsyncNavMeshUpdaterThreads = ::Settings::navigator().mAsyncNavMeshUpdaterThreads;
|
||||||
result.mMaxNavMeshTilesCacheSize = ::Settings::navigator().mMaxNavMeshTilesCacheSize;
|
result.mMaxNavMeshTilesCacheSize = ::Settings::navigator().mMaxNavMeshTilesCacheSize;
|
||||||
|
|
|
@ -55,10 +55,6 @@ namespace DetourNavigator
|
||||||
|
|
||||||
inline constexpr std::int64_t navMeshFormatVersion = 2;
|
inline constexpr std::int64_t navMeshFormatVersion = 2;
|
||||||
|
|
||||||
RecastSettings makeRecastSettingsFromSettingsManager();
|
|
||||||
|
|
||||||
DetourSettings makeDetourSettingsFromSettingsManager();
|
|
||||||
|
|
||||||
Settings makeSettingsFromSettingsManager();
|
Settings makeSettingsFromSettingsManager();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -981,7 +981,7 @@ enable agents paths render = false
|
||||||
enable recast mesh render = false
|
enable recast mesh render = false
|
||||||
|
|
||||||
# Max number of navmesh tiles (value >= 0)
|
# Max number of navmesh tiles (value >= 0)
|
||||||
max tiles number = 512
|
max tiles number = 1024
|
||||||
|
|
||||||
# Min time duration for the same tile update in milliseconds (value >= 0)
|
# Min time duration for the same tile update in milliseconds (value >= 0)
|
||||||
min update interval ms = 250
|
min update interval ms = 250
|
||||||
|
|
Loading…
Reference in a new issue