mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-19 21:23:54 +00:00
333 lines
13 KiB
C++
333 lines
13 KiB
C++
#include "makenavmesh.hpp"
|
|
#include "chunkytrimesh.hpp"
|
|
#include "debug.hpp"
|
|
#include "dtstatus.hpp"
|
|
#include "exceptions.hpp"
|
|
#include "recastmesh.hpp"
|
|
#include "settings.hpp"
|
|
#include "settingsutils.hpp"
|
|
#include "sharednavmesh.hpp"
|
|
|
|
#include <DetourNavMesh.h>
|
|
#include <DetourNavMeshBuilder.h>
|
|
#include <Recast.h>
|
|
#include <RecastAlloc.h>
|
|
|
|
#include <iomanip>
|
|
#include <limits>
|
|
|
|
namespace
|
|
{
|
|
using namespace DetourNavigator;
|
|
|
|
void initPolyMeshDetail(rcPolyMeshDetail& value)
|
|
{
|
|
value.meshes = nullptr;
|
|
value.verts = nullptr;
|
|
value.tris = nullptr;
|
|
}
|
|
|
|
struct PolyMeshDetailStackDeleter
|
|
{
|
|
void operator ()(rcPolyMeshDetail* value) const
|
|
{
|
|
rcFree(value->meshes);
|
|
rcFree(value->verts);
|
|
rcFree(value->tris);
|
|
}
|
|
};
|
|
|
|
using PolyMeshDetailStackPtr = std::unique_ptr<rcPolyMeshDetail, PolyMeshDetailStackDeleter>;
|
|
|
|
struct NavMeshDataValueDeleter
|
|
{
|
|
void operator ()(unsigned char* value) const
|
|
{
|
|
dtFree(value);
|
|
}
|
|
};
|
|
|
|
using NavMeshDataValue = std::unique_ptr<unsigned char, NavMeshDataValueDeleter>;
|
|
|
|
struct NavMeshData
|
|
{
|
|
NavMeshDataValue mValue;
|
|
int mSize;
|
|
|
|
NavMeshData() = default;
|
|
|
|
NavMeshData(unsigned char* value, int size)
|
|
: mValue(value)
|
|
, mSize(size)
|
|
{}
|
|
};
|
|
|
|
NavMeshData makeNavMeshTileData(const osg::Vec3f& agentHalfExtents, const RecastMesh& recastMesh,
|
|
const int tileX, const int tileY, const osg::Vec3f& boundsMin, const osg::Vec3f& boundsMax,
|
|
const Settings& settings)
|
|
{
|
|
rcContext context;
|
|
rcConfig config;
|
|
|
|
config.cs = settings.mCellSize;
|
|
config.ch = settings.mCellHeight;
|
|
config.walkableSlopeAngle = settings.mMaxSlope;
|
|
config.walkableHeight = static_cast<int>(std::ceil(getHeight(settings, agentHalfExtents) / config.ch));
|
|
config.walkableClimb = static_cast<int>(std::floor(getMaxClimb(settings) / config.ch));
|
|
config.walkableRadius = static_cast<int>(std::ceil(getRadius(settings, agentHalfExtents) / config.cs));
|
|
config.maxEdgeLen = static_cast<int>(std::round(settings.mMaxEdgeLen / config.cs));
|
|
config.maxSimplificationError = settings.mMaxSimplificationError;
|
|
config.minRegionArea = settings.mRegionMinSize * settings.mRegionMinSize;
|
|
config.mergeRegionArea = settings.mRegionMergeSize * settings.mRegionMergeSize;
|
|
config.maxVertsPerPoly = settings.mMaxVertsPerPoly;
|
|
config.detailSampleDist = settings.mDetailSampleDist < 0.9f ? 0 : config.cs * settings.mDetailSampleDist;
|
|
config.detailSampleMaxError = config.ch * settings.mDetailSampleMaxError;
|
|
config.borderSize = settings.mBorderSize;
|
|
config.width = settings.mTileSize + config.borderSize * 2;
|
|
config.height = settings.mTileSize + config.borderSize * 2;
|
|
rcVcopy(config.bmin, boundsMin.ptr());
|
|
rcVcopy(config.bmax, boundsMax.ptr());
|
|
config.bmin[0] -= getBorderSize(settings);
|
|
config.bmin[2] -= getBorderSize(settings);
|
|
config.bmax[0] += getBorderSize(settings);
|
|
config.bmax[2] += getBorderSize(settings);
|
|
|
|
rcHeightfield solid;
|
|
OPENMW_CHECK_DT_RESULT(rcCreateHeightfield(nullptr, solid, config.width, config.height,
|
|
config.bmin, config.bmax, config.cs, config.ch));
|
|
|
|
{
|
|
const auto& chunkyMesh = recastMesh.getChunkyTriMesh();
|
|
std::vector<unsigned char> areas(chunkyMesh.getMaxTrisPerChunk(), 0);
|
|
const osg::Vec2f tileBoundsMin(config.bmin[0], config.bmin[2]);
|
|
const osg::Vec2f tileBoundsMax(config.bmax[0], config.bmax[2]);
|
|
std::vector<std::size_t> cids;
|
|
chunkyMesh.getChunksOverlappingRect(Rect {tileBoundsMin, tileBoundsMax}, std::back_inserter(cids));
|
|
|
|
if (cids.empty())
|
|
return NavMeshData();
|
|
|
|
for (const auto cid : cids)
|
|
{
|
|
const auto chunk = chunkyMesh.getChunk(cid);
|
|
|
|
std::fill(
|
|
areas.begin(),
|
|
std::min(areas.begin() + static_cast<std::ptrdiff_t>(chunk.mSize),
|
|
areas.end()),
|
|
0
|
|
);
|
|
|
|
rcMarkWalkableTriangles(
|
|
&context,
|
|
config.walkableSlopeAngle,
|
|
recastMesh.getVertices().data(),
|
|
static_cast<int>(recastMesh.getVerticesCount()),
|
|
chunk.mIndices,
|
|
static_cast<int>(chunk.mSize),
|
|
areas.data()
|
|
);
|
|
|
|
OPENMW_CHECK_DT_RESULT(rcRasterizeTriangles(
|
|
&context,
|
|
recastMesh.getVertices().data(),
|
|
static_cast<int>(recastMesh.getVerticesCount()),
|
|
chunk.mIndices,
|
|
areas.data(),
|
|
static_cast<int>(chunk.mSize),
|
|
solid,
|
|
config.walkableClimb
|
|
));
|
|
}
|
|
}
|
|
|
|
rcFilterLowHangingWalkableObstacles(&context, config.walkableClimb, solid);
|
|
rcFilterLedgeSpans(&context, config.walkableHeight, config.walkableClimb, solid);
|
|
rcFilterWalkableLowHeightSpans(&context, config.walkableHeight, solid);
|
|
|
|
rcPolyMesh polyMesh;
|
|
rcPolyMeshDetail polyMeshDetail;
|
|
initPolyMeshDetail(polyMeshDetail);
|
|
const PolyMeshDetailStackPtr polyMeshDetailPtr(&polyMeshDetail);
|
|
{
|
|
rcCompactHeightfield compact;
|
|
|
|
OPENMW_CHECK_DT_RESULT(rcBuildCompactHeightfield(&context, config.walkableHeight, config.walkableClimb,
|
|
solid, compact));
|
|
OPENMW_CHECK_DT_RESULT(rcErodeWalkableArea(&context, config.walkableRadius, compact));
|
|
OPENMW_CHECK_DT_RESULT(rcBuildDistanceField(&context, compact));
|
|
OPENMW_CHECK_DT_RESULT(rcBuildRegions(&context, compact, config.borderSize, config.minRegionArea,
|
|
config.mergeRegionArea));
|
|
|
|
rcContourSet contourSet;
|
|
OPENMW_CHECK_DT_RESULT(rcBuildContours(&context, compact, config.maxSimplificationError, config.maxEdgeLen,
|
|
contourSet));
|
|
|
|
if (contourSet.nconts == 0)
|
|
return NavMeshData();
|
|
|
|
OPENMW_CHECK_DT_RESULT(rcBuildPolyMesh(&context, contourSet, config.maxVertsPerPoly, polyMesh));
|
|
OPENMW_CHECK_DT_RESULT(rcBuildPolyMeshDetail(&context, polyMesh, compact, config.detailSampleDist,
|
|
config.detailSampleMaxError, polyMeshDetail));
|
|
}
|
|
|
|
for (int i = 0; i < polyMesh.npolys; ++i)
|
|
if (polyMesh.areas[i] == RC_WALKABLE_AREA)
|
|
polyMesh.flags[i] = 1;
|
|
|
|
dtNavMeshCreateParams params;
|
|
params.verts = polyMesh.verts;
|
|
params.vertCount = polyMesh.nverts;
|
|
params.polys = polyMesh.polys;
|
|
params.polyAreas = polyMesh.areas;
|
|
params.polyFlags = polyMesh.flags;
|
|
params.polyCount = polyMesh.npolys;
|
|
params.nvp = polyMesh.nvp;
|
|
params.detailMeshes = polyMeshDetail.meshes;
|
|
params.detailVerts = polyMeshDetail.verts;
|
|
params.detailVertsCount = polyMeshDetail.nverts;
|
|
params.detailTris = polyMeshDetail.tris;
|
|
params.detailTriCount = polyMeshDetail.ntris;
|
|
params.offMeshConVerts = nullptr;
|
|
params.offMeshConRad = nullptr;
|
|
params.offMeshConDir = nullptr;
|
|
params.offMeshConAreas = nullptr;
|
|
params.offMeshConFlags = nullptr;
|
|
params.offMeshConUserID = nullptr;
|
|
params.offMeshConCount = 0;
|
|
params.walkableHeight = getHeight(settings, agentHalfExtents);
|
|
params.walkableRadius = getRadius(settings, agentHalfExtents);
|
|
params.walkableClimb = getMaxClimb(settings);
|
|
rcVcopy(params.bmin, polyMesh.bmin);
|
|
rcVcopy(params.bmax, polyMesh.bmax);
|
|
params.cs = config.cs;
|
|
params.ch = config.ch;
|
|
params.buildBvTree = true;
|
|
params.userId = 0;
|
|
params.tileX = tileX;
|
|
params.tileY = tileY;
|
|
params.tileLayer = 0;
|
|
|
|
unsigned char* navMeshData;
|
|
int navMeshDataSize;
|
|
OPENMW_CHECK_DT_RESULT(dtCreateNavMeshData(¶ms, &navMeshData, &navMeshDataSize));
|
|
|
|
return NavMeshData(navMeshData, navMeshDataSize);
|
|
}
|
|
|
|
struct AutoIncrementRevision
|
|
{
|
|
std::atomic_size_t& mNavMeshRevision;
|
|
bool mNavMeshChanged;
|
|
|
|
AutoIncrementRevision(std::atomic_size_t& navMeshRevision)
|
|
: mNavMeshRevision(navMeshRevision)
|
|
, mNavMeshChanged(false)
|
|
{}
|
|
|
|
~AutoIncrementRevision()
|
|
{
|
|
if (mNavMeshChanged)
|
|
++mNavMeshRevision;
|
|
}
|
|
};
|
|
|
|
UpdateNavMeshStatus makeUpdateNavMeshStatus(bool removed, bool add)
|
|
{
|
|
if (removed && add)
|
|
return UpdateNavMeshStatus::replaced;
|
|
else if (removed)
|
|
return UpdateNavMeshStatus::removed;
|
|
else if (add)
|
|
return UpdateNavMeshStatus::add;
|
|
else
|
|
return UpdateNavMeshStatus::ignore;
|
|
}
|
|
}
|
|
|
|
namespace DetourNavigator
|
|
{
|
|
NavMeshPtr makeEmptyNavMesh(const Settings& 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.
|
|
const auto tileBits = 10;
|
|
const auto polyBits = 22 - tileBits;
|
|
const auto maxTiles = 1 << tileBits;
|
|
const auto maxPolysPerTile = 1 << polyBits;
|
|
|
|
dtNavMeshParams params;
|
|
std::fill_n(params.orig, 3, 0.0f);
|
|
params.tileWidth = settings.mTileSize * settings.mCellSize;
|
|
params.tileHeight = settings.mTileSize * settings.mCellSize;
|
|
params.maxTiles = maxTiles;
|
|
params.maxPolys = maxPolysPerTile;
|
|
|
|
NavMeshPtr navMesh(dtAllocNavMesh(), &dtFreeNavMesh);
|
|
OPENMW_CHECK_DT_STATUS(navMesh->init(¶ms));
|
|
|
|
return navMesh;
|
|
}
|
|
|
|
UpdateNavMeshStatus updateNavMesh(const osg::Vec3f& agentHalfExtents, const RecastMesh& recastMesh,
|
|
const TilePosition& changedTile, const Settings& settings, NavMeshCacheItem& navMeshCacheItem)
|
|
{
|
|
log("update NavMesh with mutiple tiles:",
|
|
" agentHeight=", std::setprecision(std::numeric_limits<float>::max_exponent10),
|
|
getHeight(settings, agentHalfExtents),
|
|
" agentMaxClimb=", std::setprecision(std::numeric_limits<float>::max_exponent10),
|
|
getMaxClimb(settings),
|
|
" agentRadius=", std::setprecision(std::numeric_limits<float>::max_exponent10),
|
|
getRadius(settings, agentHalfExtents),
|
|
" changedTile=", changedTile);
|
|
|
|
auto& navMesh = navMeshCacheItem.mValue;
|
|
const auto& params = *navMesh.lock()->getParams();
|
|
const osg::Vec3f origin(params.orig[0], params.orig[1], params.orig[2]);
|
|
|
|
const auto x = changedTile.x();
|
|
const auto y = changedTile.y();
|
|
|
|
AutoIncrementRevision incRev(navMeshCacheItem.mNavMeshRevision);
|
|
bool removed = false;
|
|
|
|
{
|
|
const auto locked = navMesh.lock();
|
|
removed = dtStatusSucceed(locked->removeTile(locked->getTileRefAt(x, y, 0), nullptr, nullptr));
|
|
}
|
|
|
|
incRev.mNavMeshChanged = removed;
|
|
|
|
const auto& boundsMin = recastMesh.getBoundsMin();
|
|
const auto& boundsMax = recastMesh.getBoundsMax();
|
|
|
|
if (boundsMin == boundsMax)
|
|
{
|
|
log("ignore add tile: recastMesh is empty");
|
|
return makeUpdateNavMeshStatus(removed, false);
|
|
}
|
|
|
|
const auto tileBounds = makeTileBounds(settings, changedTile);
|
|
const osg::Vec3f tileBorderMin(tileBounds.mMin.x(), boundsMin.y() - 1, tileBounds.mMin.y());
|
|
const osg::Vec3f tileBorderMax(tileBounds.mMax.x(), boundsMax.y() + 1, tileBounds.mMax.y());
|
|
|
|
auto navMeshData = makeNavMeshTileData(agentHalfExtents, recastMesh, x, y,
|
|
tileBorderMin, tileBorderMax, settings);
|
|
|
|
if (!navMeshData.mValue)
|
|
{
|
|
log("ignore add tile: NavMeshData is null");
|
|
return makeUpdateNavMeshStatus(removed, false);
|
|
}
|
|
|
|
const auto status = navMesh.lock()->addTile(navMeshData.mValue.get(), navMeshData.mSize,
|
|
DT_TILE_FREE_DATA, 0, 0);
|
|
if (dtStatusSucceed(status))
|
|
incRev.mNavMeshChanged = true;
|
|
else
|
|
log("failed to add tile with status=", WriteDtStatus {status});
|
|
navMeshData.mValue.release();
|
|
|
|
return makeUpdateNavMeshStatus(removed, dtStatusSucceed(status));
|
|
}
|
|
}
|