Compare commits

...

18 Commits

Author SHA1 Message Date
Семенов Иван 01031554d1 Increase max query filter types count 2025-03-19 13:19:58 +03:00
Семенов Иван 01ff8806cd Increase max target find path iterations 2025-03-19 12:50:27 +03:00
Семенов Иван f5fc9f6c2f Optimisation: getting portal points unsafe 2025-03-06 16:45:04 +03:00
Семенов Иван 62ebb2298e Optimisation: disable position snapping for FindStraightPath 2025-03-06 16:31:06 +03:00
Семенов Иван 6aff9574c9 Disable crowd neighbours building for optimisation 2025-03-04 16:29:54 +03:00
wrenge 0b888b16fb Merge branch 'risky_optimizations' 2024-11-14 12:40:22 +03:00
wrenge 088edcd655 Intersection query alloc fix 2024-11-14 12:40:03 +03:00
wrenge 12e09475f0 Replace predicate find with simple find 2024-11-13 17:15:32 +03:00
wrenge 7de4b51135 Merge branch 'risky_optimizations' 2024-11-13 16:19:06 +03:00
wrenge 4824f29db7 Non alloc raycast 2024-11-13 15:51:02 +03:00
wrenge 2cf9d9de18 Merge branch 'risky_optimizations' 2024-11-13 14:20:49 +03:00
wrenge d2082c6586 Added clear 2024-11-13 14:20:38 +03:00
wrenge 0876d3adcf Merge branch 'risky_optimizations' 2024-11-13 13:55:25 +03:00
wrenge 1fa0320845 precache collections 2024-11-13 13:55:02 +03:00
wrenge 815a83e3cb Use precached queue instead of linked list 2024-11-13 13:42:43 +03:00
wrenge 592ecebe1e Proximity grid list reuse 2024-11-13 13:25:13 +03:00
wrenge b2a217d4a3 Object pool 2024-11-13 13:21:41 +03:00
wrenge 5c0ba9dba1 Reuse grid instead of creating new 2024-11-13 13:14:32 +03:00
14 changed files with 157 additions and 98 deletions

View File

@ -88,7 +88,7 @@ namespace DotRecast.Core.Collections
return false; return false;
//int idx = _items.BinarySearch(item, _comparer); // don't use this! Because reference types can be reused externally. //int idx = _items.BinarySearch(item, _comparer); // don't use this! Because reference types can be reused externally.
int idx = _items.FindLastIndex(x => item.Equals(x)); int idx = _items.LastIndexOf(item);
if (0 > idx) if (0 > idx)
return false; return false;

View File

@ -205,7 +205,7 @@ namespace DotRecast.Core.Numerics
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)] [MethodImpl(MethodImplOptions.AggressiveInlining)]
public static float Dist2DSqr(RcVec3f p, float[] verts, int i) public static float Dist2DSqr(RcVec3f p, Span<float> verts, int i)
{ {
float dx = verts[i] - p.X; float dx = verts[i] - p.X;
float dz = verts[i + 2] - p.Z; float dz = verts[i + 2] - p.Z;

View File

@ -130,7 +130,7 @@ namespace DotRecast.Detour.Crowd
private readonly DtObstacleAvoidanceParams[] _obstacleQueryParams; private readonly DtObstacleAvoidanceParams[] _obstacleQueryParams;
private readonly DtObstacleAvoidanceQuery _obstacleQuery; private readonly DtObstacleAvoidanceQuery _obstacleQuery;
private DtProximityGrid _grid; private readonly DtProximityGrid _grid;
private int _maxPathResult; private int _maxPathResult;
private readonly RcVec3f _agentPlacementHalfExtents; private readonly RcVec3f _agentPlacementHalfExtents;
@ -175,6 +175,7 @@ namespace DotRecast.Detour.Crowd
_agentIdx = new RcAtomicInteger(0); _agentIdx = new RcAtomicInteger(0);
_agents = new Dictionary<int, DtCrowdAgent>(); _agents = new Dictionary<int, DtCrowdAgent>();
_activeAgents = new List<DtCrowdAgent>(); _activeAgents = new List<DtCrowdAgent>();
_grid = new DtProximityGrid(_config.maxAgentRadius * 3);
// The navQuery is mostly used for local searches, no need for large node pool. // The navQuery is mostly used for local searches, no need for large node pool.
SetNavMesh(nav); SetNavMesh(nav);
@ -424,11 +425,13 @@ namespace DotRecast.Detour.Crowd
// Optimize path topology. // Optimize path topology.
UpdateTopologyOptimization(agents, dt); UpdateTopologyOptimization(agents, dt);
#region Disabled for performance issues
// Register agents to proximity grid. // Register agents to proximity grid.
BuildProximityGrid(agents); // BuildProximityGrid(agents);
// Get nearby navmesh segments and agents to collide with. // Get nearby navmesh segments and agents to collide with.
BuildNeighbours(agents); // BuildNeighbours(agents);
#endregion
// Find next corner to steer to. // Find next corner to steer to.
FindCorners(agents, debug); FindCorners(agents, debug);
@ -565,14 +568,18 @@ namespace DotRecast.Detour.Crowd
} }
} }
private readonly RcSortedQueue<DtCrowdAgent> UpdateMoveRequest_queue = new RcSortedQueue<DtCrowdAgent>((a1, a2) => a2.targetReplanTime.CompareTo(a1.targetReplanTime));
private readonly List<long> UpdateMoveRequest_reqPath = new List<long>();
private void UpdateMoveRequest(IList<DtCrowdAgent> agents, float dt) private void UpdateMoveRequest(IList<DtCrowdAgent> agents, float dt)
{ {
using var timer = _telemetry.ScopedTimer(DtCrowdTimerLabel.UpdateMoveRequest); using var timer = _telemetry.ScopedTimer(DtCrowdTimerLabel.UpdateMoveRequest);
RcSortedQueue<DtCrowdAgent> queue = new RcSortedQueue<DtCrowdAgent>((a1, a2) => a2.targetReplanTime.CompareTo(a1.targetReplanTime)); RcSortedQueue<DtCrowdAgent> queue = UpdateMoveRequest_queue;
queue.Clear();
// Fire off new requests. // Fire off new requests.
List<long> reqPath = new List<long>(); List<long> reqPath = UpdateMoveRequest_reqPath;
reqPath.Clear();
for (var i = 0; i < agents.Count; i++) for (var i = 0; i < agents.Count; i++)
{ {
var ag = agents[i]; var ag = agents[i];
@ -865,7 +872,7 @@ namespace DotRecast.Detour.Crowd
{ {
using var timer = _telemetry.ScopedTimer(DtCrowdTimerLabel.BuildProximityGrid); using var timer = _telemetry.ScopedTimer(DtCrowdTimerLabel.BuildProximityGrid);
_grid = new DtProximityGrid(_config.maxAgentRadius * 3); _grid.Clear();
for (var i = 0; i < agents.Count; i++) for (var i = 0; i < agents.Count; i++)
{ {

View File

@ -25,7 +25,7 @@ namespace DotRecast.Detour.Crowd
public int pathQueueSize = 32; // Max number of path requests in the queue public int pathQueueSize = 32; // Max number of path requests in the queue
public int maxFindPathIterations = 100; // Max number of sliced path finding iterations executed per update (used to handle longer paths and replans) public int maxFindPathIterations = 100; // Max number of sliced path finding iterations executed per update (used to handle longer paths and replans)
public int maxTargetFindPathIterations = 20; // Max number of sliced path finding iterations executed per agent to find the initial path to target public int maxTargetFindPathIterations = 100; // Max number of sliced path finding iterations executed per agent to find the initial path to target
public float topologyOptimizationTimeThreshold = 0.5f; // Min time between topology optimizations (in seconds) public float topologyOptimizationTimeThreshold = 0.5f; // Min time between topology optimizations (in seconds)
public int checkLookAhead = 10; // The number of polygons from the beginning of the corridor to check to ensure path validity public int checkLookAhead = 10; // The number of polygons from the beginning of the corridor to check to ensure path validity
public float targetReplanDelay = 1.0f; // Min time between target re-planning (in seconds) public float targetReplanDelay = 1.0f; // Min time between target re-planning (in seconds)

View File

@ -25,7 +25,7 @@ namespace DotRecast.Detour.Crowd
/// @ingroup crowd /// @ingroup crowd
/// @see dtQueryFilter, dtCrowd::GetFilter() dtCrowd::GetEditableFilter(), /// @see dtQueryFilter, dtCrowd::GetFilter() dtCrowd::GetEditableFilter(),
/// dtCrowdAgentParams::queryFilterType /// dtCrowdAgentParams::queryFilterType
public const int DT_CROWD_MAX_QUERY_FILTER_TYPE = 16; public const int DT_CROWD_MAX_QUERY_FILTER_TYPE = 32;
public const int MAX_ITERS_PER_UPDATE = 100; public const int MAX_ITERS_PER_UPDATE = 100;
public const int MAX_PATHQUEUE_NODES = 4096; public const int MAX_PATHQUEUE_NODES = 4096;

View File

@ -22,6 +22,7 @@ using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Runtime.CompilerServices; using System.Runtime.CompilerServices;
using DotRecast.Core.Buffers;
namespace DotRecast.Detour.Crowd namespace DotRecast.Detour.Crowd
{ {
@ -30,12 +31,14 @@ namespace DotRecast.Detour.Crowd
private readonly float _cellSize; private readonly float _cellSize;
private readonly float _invCellSize; private readonly float _invCellSize;
private readonly Dictionary<long, List<DtCrowdAgent>> _items; private readonly Dictionary<long, List<DtCrowdAgent>> _items;
private readonly RcObjectPool<List<DtCrowdAgent>> _listPool;
public DtProximityGrid(float cellSize) public DtProximityGrid(float cellSize)
{ {
_cellSize = cellSize; _cellSize = cellSize;
_invCellSize = 1.0f / cellSize; _invCellSize = 1.0f / cellSize;
_items = new Dictionary<long, List<DtCrowdAgent>>(); _items = new Dictionary<long, List<DtCrowdAgent>>();
_listPool = new RcObjectPool<List<DtCrowdAgent>>(() => new List<DtCrowdAgent>());
} }
[MethodImpl(MethodImplOptions.AggressiveInlining)] [MethodImpl(MethodImplOptions.AggressiveInlining)]
@ -57,6 +60,8 @@ namespace DotRecast.Detour.Crowd
public void Clear() public void Clear()
{ {
foreach (var pair in _items)
_listPool.Return(pair.Value);
_items.Clear(); _items.Clear();
} }
@ -74,7 +79,8 @@ namespace DotRecast.Detour.Crowd
long key = CombineKey(x, y); long key = CombineKey(x, y);
if (!_items.TryGetValue(key, out var ids)) if (!_items.TryGetValue(key, out var ids))
{ {
ids = new List<DtCrowdAgent>(); ids = _listPool.Get();
ids.Clear();
_items.Add(key, ids); _items.Add(key, ids);
} }

View File

@ -27,7 +27,7 @@ namespace DotRecast.Detour
{ {
private const float EPSILON = 0.0001f; private const float EPSILON = 0.0001f;
public static float[] Intersect(Span<float> p, Span<float> q) public static Span<float> Intersect(Span<float> p, Span<float> q, Span<float> buffer)
{ {
int n = p.Length / 3; int n = p.Length / 3;
int m = q.Length / 3; int m = q.Length / 3;
@ -95,7 +95,7 @@ namespace DotRecast.Detour
/* Special case: A & B parallel and separated. */ /* Special case: A & B parallel and separated. */
if (parallel && aHB < 0f && bHA < 0f) if (parallel && aHB < 0f && bHA < 0f)
{ {
return null; return Span<float>.Empty;
} }
/* Special case: A & B collinear. */ /* Special case: A & B collinear. */
else if (parallel && MathF.Abs(aHB) < EPSILON && MathF.Abs(bHA) < EPSILON) else if (parallel && MathF.Abs(aHB) < EPSILON && MathF.Abs(bHA) < EPSILON)
@ -168,8 +168,9 @@ namespace DotRecast.Detour
return null; return null;
} }
float[] copied = inters.Slice(0, ii).ToArray(); Span<float> result = buffer.Slice(0, ii);
return copied; inters.Slice(0, ii).CopyTo(result);
return result;
} }
private static int AddVertex(Span<float> inters, int ii, RcVec3f p) private static int AddVertex(Span<float> inters, int ii, RcVec3f p)

View File

@ -23,6 +23,7 @@ using System.Collections.Generic;
using DotRecast.Core; using DotRecast.Core;
using DotRecast.Core.Buffers; using DotRecast.Core.Buffers;
using DotRecast.Core.Numerics; using DotRecast.Core.Numerics;
using CollectionExtensions = DotRecast.Core.Collections.CollectionExtensions;
namespace DotRecast.Detour namespace DotRecast.Detour
{ {
@ -33,6 +34,8 @@ namespace DotRecast.Detour
/// @ingroup detour /// @ingroup detour
public class DtNavMeshQuery public class DtNavMeshQuery
{ {
public const int MAX_PATH_LENGTH = 256;
protected readonly DtNavMesh m_nav; //< Pointer to navmesh data. protected readonly DtNavMesh m_nav; //< Pointer to navmesh data.
protected DtQueryData m_query; //< Sliced query state. protected DtQueryData m_query; //< Sliced query state.
@ -230,6 +233,7 @@ namespace DotRecast.Detour
IDtQueryFilter filter, IRcRand frand, IDtPolygonByCircleConstraint constraint, IDtQueryFilter filter, IRcRand frand, IDtPolygonByCircleConstraint constraint,
out long randomRef, out RcVec3f randomPt) out long randomRef, out RcVec3f randomPt)
{ {
const int MAX_VERT_BUFFER_SIZE = 128;
randomRef = startRef; randomRef = startRef;
randomPt = centerPos; randomPt = centerPos;
@ -262,10 +266,14 @@ namespace DotRecast.Detour
float radiusSqr = maxRadius * maxRadius; float radiusSqr = maxRadius * maxRadius;
float areaSum = 0.0f; float areaSum = 0.0f;
using RcRentedArray<float> polyVertsBuffer = RcRentedArray.Rent<float>(MAX_VERT_BUFFER_SIZE);
using RcRentedArray<float> randomPolyVertsBuffer = RcRentedArray.Rent<float>(MAX_VERT_BUFFER_SIZE);
using RcRentedArray<float> constrainedVertsBuffer = RcRentedArray.Rent<float>(MAX_VERT_BUFFER_SIZE);
DtPoly randomPoly = null; DtPoly randomPoly = null;
long randomPolyRef = 0; long randomPolyRef = 0;
float[] randomPolyVerts = null; Span<float> randomPolyVerts = Span<float>.Empty;
while (!m_openList.IsEmpty()) while (!m_openList.IsEmpty())
{ {
@ -283,14 +291,14 @@ namespace DotRecast.Detour
{ {
// Calc area of the polygon. // Calc area of the polygon.
float polyArea = 0.0f; float polyArea = 0.0f;
float[] polyVerts = new float[bestPoly.vertCount * 3]; Span<float> polyVerts = polyVertsBuffer.AsSpan().Slice(0, bestPoly.vertCount * 3);
for (int j = 0; j < bestPoly.vertCount; ++j) for (int j = 0; j < bestPoly.vertCount; ++j)
{ {
RcArrays.Copy(bestTile.data.verts, bestPoly.verts[j] * 3, polyVerts, j * 3, 3); RcSpans.Copy(bestTile.data.verts, bestPoly.verts[j] * 3, polyVerts, j * 3, 3);
} }
float[] constrainedVerts = constraint.Apply(polyVerts, centerPos, maxRadius); Span<float> constrainedVerts = constraint.Apply(polyVerts, centerPos, maxRadius, constrainedVertsBuffer.AsSpan());
if (constrainedVerts != null) if (!constrainedVerts.IsEmpty)
{ {
int vertCount = constrainedVerts.Length / 3; int vertCount = constrainedVerts.Length / 3;
for (int j = 2; j < vertCount; ++j) for (int j = 2; j < vertCount; ++j)
@ -308,7 +316,8 @@ namespace DotRecast.Detour
{ {
randomPoly = bestPoly; randomPoly = bestPoly;
randomPolyRef = bestRef; randomPolyRef = bestRef;
randomPolyVerts = constrainedVerts; randomPolyVerts = randomPolyVertsBuffer.AsSpan().Slice(0, constrainedVerts.Length);
constrainedVerts.CopyTo(randomPolyVerts);
} }
} }
} }
@ -882,7 +891,8 @@ namespace DotRecast.Detour
float lastBestNodeCost = startNode.total; float lastBestNodeCost = startNode.total;
DtRaycastHit rayHit = new DtRaycastHit(); DtRaycastHit rayHit = new DtRaycastHit();
rayHit.path = new List<long>(); using var pathRent = RcRentedArray.Rent<long>(MAX_PATH_LENGTH);
rayHit.path = pathRent.AsArray();
while (!m_openList.IsEmpty()) while (!m_openList.IsEmpty())
{ {
// Remove node from open list and put it in closed list. // Remove node from open list and put it in closed list.
@ -1171,7 +1181,8 @@ namespace DotRecast.Detour
} }
var rayHit = new DtRaycastHit(); var rayHit = new DtRaycastHit();
rayHit.path = new List<long>(); using var pathRent = RcRentedArray.Rent<long>(MAX_PATH_LENGTH);
rayHit.path = pathRent.AsArray();
int iter = 0; int iter = 0;
while (iter < maxIter && !m_openList.IsEmpty()) while (iter < maxIter && !m_openList.IsEmpty())
@ -1626,20 +1637,20 @@ namespace DotRecast.Detour
DtStatus stat = DtStatus.DT_STATUS_NOTHING; DtStatus stat = DtStatus.DT_STATUS_NOTHING;
// TODO: Should this be callers responsibility? // TODO: Should this be callers responsibility?
var closestStartPosRes = ClosestPointOnPolyBoundary(path[0], startPos, out var closestStartPos); // var closestStartPosRes = ClosestPointOnPolyBoundary(path[0], startPos, out var closestStartPos);
if (closestStartPosRes.Failed()) // if (closestStartPosRes.Failed())
{ // {
return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM; // return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM;
} // }
//
var closestEndPosRes = ClosestPointOnPolyBoundary(path[pathSize - 1], endPos, out var closestEndPos); // var closestEndPosRes = ClosestPointOnPolyBoundary(path[pathSize - 1], endPos, out var closestEndPos);
if (closestEndPosRes.Failed()) // if (closestEndPosRes.Failed())
{ // {
return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM; // return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM;
} // }
// Add start point. // Add start point.
stat = AppendVertex(closestStartPos, DtStraightPathFlags.DT_STRAIGHTPATH_START, path[0], straightPath, ref straightPathCount, maxStraightPath); stat = AppendVertex(startPos, DtStraightPathFlags.DT_STRAIGHTPATH_START, path[0], straightPath, ref straightPathCount, maxStraightPath);
if (!stat.InProgress()) if (!stat.InProgress())
{ {
return stat; return stat;
@ -1647,7 +1658,7 @@ namespace DotRecast.Detour
if (pathSize > 1) if (pathSize > 1)
{ {
RcVec3f portalApex = closestStartPos; RcVec3f portalApex = startPos;
RcVec3f portalLeft = portalApex; RcVec3f portalLeft = portalApex;
RcVec3f portalRight = portalApex; RcVec3f portalRight = portalApex;
int apexIndex = 0; int apexIndex = 0;
@ -1676,21 +1687,21 @@ namespace DotRecast.Detour
{ {
// Failed to get portal points, in practice this means that path[i+1] is invalid polygon. // Failed to get portal points, in practice this means that path[i+1] is invalid polygon.
// Clamp the end point to path[i], and return the path so far. // Clamp the end point to path[i], and return the path so far.
var cpStatus = ClosestPointOnPolyBoundary(path[i], endPos, out closestEndPos); // var cpStatus = ClosestPointOnPolyBoundary(path[i], endPos, out closestEndPos);
if (cpStatus.Failed()) // if (cpStatus.Failed())
{ // {
return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM; // return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM;
} // }
// Append portals along the current straight path segment. // Append portals along the current straight path segment.
if ((options & (DtStraightPathOptions.DT_STRAIGHTPATH_AREA_CROSSINGS | DtStraightPathOptions.DT_STRAIGHTPATH_ALL_CROSSINGS)) != 0) if ((options & (DtStraightPathOptions.DT_STRAIGHTPATH_AREA_CROSSINGS | DtStraightPathOptions.DT_STRAIGHTPATH_ALL_CROSSINGS)) != 0)
{ {
// Ignore status return value as we're just about to return anyway. // Ignore status return value as we're just about to return anyway.
AppendPortals(apexIndex, i, closestEndPos, path, straightPath, ref straightPathCount, maxStraightPath, options); AppendPortals(apexIndex, i, endPos, path, straightPath, ref straightPathCount, maxStraightPath, options);
} }
// Ignore status return value as we're just about to return anyway. // Ignore status return value as we're just about to return anyway.
AppendVertex(closestEndPos, 0, path[i], straightPath, ref straightPathCount, maxStraightPath); AppendVertex(endPos, 0, path[i], straightPath, ref straightPathCount, maxStraightPath);
return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM | (straightPathCount >= maxStraightPath ? DtStatus.DT_BUFFER_TOO_SMALL : DtStatus.DT_STATUS_NOTHING); return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM | (straightPathCount >= maxStraightPath ? DtStatus.DT_BUFFER_TOO_SMALL : DtStatus.DT_STATUS_NOTHING);
} }
@ -1708,8 +1719,8 @@ namespace DotRecast.Detour
else else
{ {
// End of the path. // End of the path.
left = closestEndPos; left = endPos;
right = closestEndPos; right = endPos;
toType = DtPolyTypes.DT_POLYTYPE_GROUND; toType = DtPolyTypes.DT_POLYTYPE_GROUND;
} }
@ -1829,7 +1840,7 @@ namespace DotRecast.Detour
// Append portals along the current straight path segment. // Append portals along the current straight path segment.
if ((options & (DtStraightPathOptions.DT_STRAIGHTPATH_AREA_CROSSINGS | DtStraightPathOptions.DT_STRAIGHTPATH_ALL_CROSSINGS)) != 0) if ((options & (DtStraightPathOptions.DT_STRAIGHTPATH_AREA_CROSSINGS | DtStraightPathOptions.DT_STRAIGHTPATH_ALL_CROSSINGS)) != 0)
{ {
stat = AppendPortals(apexIndex, pathSize - 1, closestEndPos, path, straightPath, ref straightPathCount, maxStraightPath, options); stat = AppendPortals(apexIndex, pathSize - 1, endPos, path, straightPath, ref straightPathCount, maxStraightPath, options);
if (!stat.InProgress()) if (!stat.InProgress())
{ {
return stat; return stat;
@ -1838,10 +1849,11 @@ namespace DotRecast.Detour
} }
// Ignore status return value as we're just about to return anyway. // Ignore status return value as we're just about to return anyway.
AppendVertex(closestEndPos, DtStraightPathFlags.DT_STRAIGHTPATH_END, 0, straightPath, ref straightPathCount, maxStraightPath); AppendVertex(endPos, DtStraightPathFlags.DT_STRAIGHTPATH_END, 0, straightPath, ref straightPathCount, maxStraightPath);
return DtStatus.DT_SUCCESS | (straightPathCount >= maxStraightPath ? DtStatus.DT_BUFFER_TOO_SMALL : DtStatus.DT_STATUS_NOTHING); return DtStatus.DT_SUCCESS | (straightPathCount >= maxStraightPath ? DtStatus.DT_BUFFER_TOO_SMALL : DtStatus.DT_STATUS_NOTHING);
} }
private readonly Queue<DtNode> MoveAlongSurface_queue = new Queue<DtNode>();
/// @par /// @par
/// ///
/// This method is optimized for small delta movement and a small number of /// This method is optimized for small delta movement and a small number of
@ -1897,8 +1909,8 @@ namespace DotRecast.Detour
startNode.total = 0; startNode.total = 0;
startNode.id = startRef; startNode.id = startRef;
startNode.flags = DtNodeFlags.DT_NODE_CLOSED; startNode.flags = DtNodeFlags.DT_NODE_CLOSED;
LinkedList<DtNode> stack = new LinkedList<DtNode>(); MoveAlongSurface_queue.Clear();
stack.AddLast(startNode); MoveAlongSurface_queue.Enqueue(startNode);
RcVec3f bestPos = new RcVec3f(); RcVec3f bestPos = new RcVec3f();
float bestDist = float.MaxValue; float bestDist = float.MaxValue;
@ -1914,11 +1926,10 @@ namespace DotRecast.Detour
const int MAX_NEIS = 8; const int MAX_NEIS = 8;
Span<long> neis = stackalloc long[MAX_NEIS]; Span<long> neis = stackalloc long[MAX_NEIS];
while (0 < stack.Count) while (0 < MoveAlongSurface_queue.Count)
{ {
// Pop front. // Pop front.
DtNode curNode = stack.First?.Value; DtNode curNode = MoveAlongSurface_queue.Dequeue();
stack.RemoveFirst();
// Get poly and tile. // Get poly and tile.
// The API input has been checked already, skip checking internal data. // The API input has been checked already, skip checking internal data.
@ -2017,7 +2028,7 @@ namespace DotRecast.Detour
// Mark as the node as visited and push to queue. // Mark as the node as visited and push to queue.
neighbourNode.pidx = m_tinyNodePool.GetNodeIdx(curNode); neighbourNode.pidx = m_tinyNodePool.GetNodeIdx(curNode);
neighbourNode.flags |= DtNodeFlags.DT_NODE_CLOSED; neighbourNode.flags |= DtNodeFlags.DT_NODE_CLOSED;
stack.AddLast(neighbourNode); MoveAlongSurface_queue.Enqueue(neighbourNode);
} }
} }
} }
@ -2066,19 +2077,11 @@ namespace DotRecast.Detour
fromType = 0; fromType = 0;
toType = 0; toType = 0;
var status = m_nav.GetTileAndPolyByRef(from, out var fromTile, out var fromPoly); m_nav.GetTileAndPolyByRefUnsafe(from, out var fromTile, out var fromPoly);
if (status.Failed())
{
return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM;
}
fromType = fromPoly.GetPolyType(); fromType = fromPoly.GetPolyType();
status = m_nav.GetTileAndPolyByRef(to, out var toTile, out var toPoly); m_nav.GetTileAndPolyByRefUnsafe(to, out var toTile, out var toPoly);
if (status.Failed())
{
return DtStatus.DT_FAILURE | DtStatus.DT_INVALID_PARAM;
}
toType = toPoly.GetPolyType(); toType = toPoly.GetPolyType();
@ -2263,13 +2266,15 @@ namespace DotRecast.Detour
out float t, out RcVec3f hitNormal, ref List<long> path) out float t, out RcVec3f hitNormal, ref List<long> path)
{ {
DtRaycastHit hit = new DtRaycastHit(); DtRaycastHit hit = new DtRaycastHit();
hit.path = path; using var pathRent = RcRentedArray.Rent<long>(MAX_PATH_LENGTH);
hit.path = pathRent.AsArray();
DtStatus status = Raycast(startRef, startPos, endPos, filter, 0, ref hit, 0); DtStatus status = Raycast(startRef, startPos, endPos, filter, 0, ref hit, 0);
t = hit.t; t = hit.t;
hitNormal = hit.hitNormal; hitNormal = hit.hitNormal;
path = hit.path; path.Clear();
CollectionExtensions.AddRange(path, new Span<long>(hit.path, 0, hit.pathCount));
return status; return status;
} }
@ -2335,7 +2340,7 @@ namespace DotRecast.Detour
} }
hit.t = 0; hit.t = 0;
hit.path.Clear(); hit.pathCount = 0;
hit.pathCost = 0; hit.pathCost = 0;
Span<RcVec3f> verts = stackalloc RcVec3f[m_nav.GetMaxVertsPerPoly() + 1]; Span<RcVec3f> verts = stackalloc RcVec3f[m_nav.GetMaxVertsPerPoly() + 1];
@ -2388,7 +2393,7 @@ namespace DotRecast.Detour
} }
// Store visited polygons. // Store visited polygons.
hit.path.Add(curRef); hit.AddPathNode(curRef);
// Ray end is completely inside the polygon. // Ray end is completely inside the polygon.
if (segMax == -1) if (segMax == -1)
@ -2908,6 +2913,7 @@ namespace DotRecast.Detour
return DtStatus.DT_SUCCESS; return DtStatus.DT_SUCCESS;
} }
private readonly Queue<DtNode> FindLocalNeighbourhood_queue = new Queue<DtNode>();
/// @par /// @par
/// ///
/// This method is optimized for a small search radius and small number of result /// This method is optimized for a small search radius and small number of result
@ -2959,8 +2965,8 @@ namespace DotRecast.Detour
startNode.pidx = 0; startNode.pidx = 0;
startNode.id = startRef; startNode.id = startRef;
startNode.flags = DtNodeFlags.DT_NODE_CLOSED; startNode.flags = DtNodeFlags.DT_NODE_CLOSED;
LinkedList<DtNode> stack = new LinkedList<DtNode>(); FindLocalNeighbourhood_queue.Clear();
stack.AddLast(startNode); FindLocalNeighbourhood_queue.Enqueue(startNode);
resultRef.Add(startNode.id); resultRef.Add(startNode.id);
resultParent.Add(0L); resultParent.Add(0L);
@ -2970,11 +2976,11 @@ namespace DotRecast.Detour
Span<float> pa = stackalloc float[m_nav.GetMaxVertsPerPoly() * 3]; Span<float> pa = stackalloc float[m_nav.GetMaxVertsPerPoly() * 3];
Span<float> pb = stackalloc float[m_nav.GetMaxVertsPerPoly() * 3]; Span<float> pb = stackalloc float[m_nav.GetMaxVertsPerPoly() * 3];
while (0 < stack.Count) while (0 < FindLocalNeighbourhood_queue.Count)
{ {
// Pop front. // Pop front.
DtNode curNode = stack.First?.Value;
stack.RemoveFirst(); DtNode curNode = FindLocalNeighbourhood_queue.Dequeue();
// Get poly and tile. // Get poly and tile.
// The API input has been checked already, skip checking internal data. // The API input has been checked already, skip checking internal data.
@ -3087,7 +3093,7 @@ namespace DotRecast.Detour
resultRef.Add(neighbourRef); resultRef.Add(neighbourRef);
resultParent.Add(curRef); resultParent.Add(curRef);
stack.AddLast(neighbourNode); FindLocalNeighbourhood_queue.Enqueue(neighbourNode);
} }
} }

View File

@ -1,3 +1,4 @@
using System;
using DotRecast.Core.Numerics; using DotRecast.Core.Numerics;
namespace DotRecast.Detour namespace DotRecast.Detour
@ -10,9 +11,11 @@ namespace DotRecast.Detour
{ {
} }
public float[] Apply(float[] polyVerts, RcVec3f circleCenter, float radius) public Span<float> Apply(Span<float> polyVerts, RcVec3f circleCenter, float radius, Span<float> resultBuffer)
{ {
return polyVerts; var result = resultBuffer.Slice(0, polyVerts.Length);
polyVerts.CopyTo(result);
return result;
} }
} }
} }

View File

@ -38,9 +38,16 @@ namespace DotRecast.Detour
public int hitEdgeIndex; public int hitEdgeIndex;
/// Pointer to an array of reference ids of the visited polygons. [opt] /// Pointer to an array of reference ids of the visited polygons. [opt]
public List<long> path; public long[] path;
public int pathCount;
/// The cost of the path until hit. /// The cost of the path until hit.
public float pathCost; public float pathCost;
public void AddPathNode(long nodeRef)
{
path[pathCount++] = nodeRef;
}
} }
} }

View File

@ -39,8 +39,7 @@ namespace DotRecast.Detour
} }
} }
public Span<float> Apply(Span<float> verts, RcVec3f center, float radius, Span<float> resultBuffer)
public float[] Apply(float[] verts, RcVec3f center, float radius)
{ {
float radiusSqr = radius * radius; float radiusSqr = radius * radius;
int outsideVertex = -1; int outsideVertex = -1;
@ -56,19 +55,30 @@ namespace DotRecast.Detour
if (outsideVertex == -1) if (outsideVertex == -1)
{ {
// polygon inside circle // polygon inside circle
return verts; var result = resultBuffer.Slice(0, verts.Length);
verts.CopyTo(result);
return result;
} }
Span<float> qCircle = stackalloc float[UnitCircle.Length]; Span<float> qCircle = stackalloc float[UnitCircle.Length];
ScaleCircle(UnitCircle, center, radius, qCircle); ScaleCircle(UnitCircle, center, radius, qCircle);
float[] intersection = DtConvexConvexIntersections.Intersect(verts, qCircle); Span<float> intersection = DtConvexConvexIntersections.Intersect(verts, qCircle, resultBuffer);
if (intersection == null && DtUtils.PointInPolygon(center, verts, verts.Length / 3)) if (intersection.IsEmpty && DtUtils.PointInPolygon(center, verts, verts.Length / 3))
{ {
// circle inside polygon // circle inside polygon
return qCircle.ToArray(); var result = resultBuffer.Slice(0, qCircle.Length);
qCircle.CopyTo(result);
return result;
} }
return intersection; if(!intersection.IsEmpty)
{
var result = resultBuffer.Slice(0, intersection.Length);
// No need to copy, data is already in buffer
return result;
}
return Span<float>.Empty;
} }
} }
} }

View File

@ -24,6 +24,6 @@ namespace DotRecast.Detour
{ {
public interface IDtPolygonByCircleConstraint public interface IDtPolygonByCircleConstraint
{ {
float[] Apply(float[] polyVerts, RcVec3f circleCenter, float radius); Span<float> Apply(Span<float> polyVerts, RcVec3f circleCenter, float radius, Span<float> resultBuffer);
} }
} }

View File

@ -17,6 +17,7 @@ freely, subject to the following restrictions:
3. This notice may not be removed or altered from any source distribution. 3. This notice may not be removed or altered from any source distribution.
*/ */
using System;
using NUnit.Framework; using NUnit.Framework;
namespace DotRecast.Detour.Test; namespace DotRecast.Detour.Test;
@ -29,9 +30,10 @@ public class ConvexConvexIntersectionTest
{ {
float[] p = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 }; float[] p = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 };
float[] q = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 }; float[] q = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 };
float[] intersection = DtConvexConvexIntersections.Intersect(p, q); float[] buffer = new float[128];
Span<float> intersection = DtConvexConvexIntersections.Intersect(p, q, buffer);
Assert.That(intersection.Length, Is.EqualTo(5 * 3)); Assert.That(intersection.Length, Is.EqualTo(5 * 3));
Assert.That(intersection, Is.EqualTo(p)); Assert.That(intersection.ToArray(), Is.EquivalentTo(p));
} }
[Test] [Test]
@ -39,8 +41,9 @@ public class ConvexConvexIntersectionTest
{ {
float[] p = { -5, 0, -5, -5, 0, 4, 1, 0, 4, 1, 0, -5 }; float[] p = { -5, 0, -5, -5, 0, 4, 1, 0, 4, 1, 0, -5 };
float[] q = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 }; float[] q = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 };
float[] intersection = DtConvexConvexIntersections.Intersect(p, q); float[] buffer = new float[128];
Span<float> intersection = DtConvexConvexIntersections.Intersect(p, q, buffer);
Assert.That(intersection.Length, Is.EqualTo(5 * 3)); Assert.That(intersection.Length, Is.EqualTo(5 * 3));
Assert.That(intersection, Is.EqualTo(new[] { 1, 0, 3, 1, 0, -3.4f, -2, 0, -4, -4, 0, 0, -3, 0, 3 })); Assert.That(intersection.ToArray(), Is.EquivalentTo(new[] { 1, 0, 3, 1, 0, -3.4f, -2, 0, -4, -4, 0, 0, -3, 0, 3 }));
} }
} }

View File

@ -17,6 +17,7 @@ freely, subject to the following restrictions:
3. This notice may not be removed or altered from any source distribution. 3. This notice may not be removed or altered from any source distribution.
*/ */
using System;
using DotRecast.Core.Numerics; using DotRecast.Core.Numerics;
using NUnit.Framework; using NUnit.Framework;
@ -32,9 +33,12 @@ public class PolygonByCircleConstraintTest
{ {
float[] polygon = { -2, 0, 2, 2, 0, 2, 2, 0, -2, -2, 0, -2 }; float[] polygon = { -2, 0, 2, 2, 0, 2, 2, 0, -2, -2, 0, -2 };
RcVec3f center = new RcVec3f(1, 0, 1); RcVec3f center = new RcVec3f(1, 0, 1);
float[] constrained = _constraint.Apply(polygon, center, 6); var radius = 6;
float[] buffer = new float[128];
Span<float> constrained = _constraint.Apply(polygon, center, radius, buffer);
Assert.That(constrained, Is.EqualTo(polygon)); Assert.That(constrained.ToArray(), Is.EquivalentTo(polygon));
} }
[Test] [Test]
@ -43,10 +47,13 @@ public class PolygonByCircleConstraintTest
int expectedSize = 21; int expectedSize = 21;
float[] polygon = { -2, 0, 2, 2, 0, 2, 2, 0, -2, -2, 0, -2 }; float[] polygon = { -2, 0, 2, 2, 0, 2, 2, 0, -2, -2, 0, -2 };
RcVec3f center = new RcVec3f(2, 0, 0); RcVec3f center = new RcVec3f(2, 0, 0);
var radius = 3;
float[] buffer = new float[128];
Span<float> constrained = _constraint.Apply(polygon, center, radius, buffer);
float[] constrained = _constraint.Apply(polygon, center, 3);
Assert.That(constrained.Length, Is.EqualTo(expectedSize)); Assert.That(constrained.Length, Is.EqualTo(expectedSize));
Assert.That(constrained, Is.SupersetOf(new[] { 2f, 0f, 2f, 2f, 0f, -2f })); Assert.That(constrained.ToArray(), Is.SupersetOf(new[] { 2f, 0f, 2f, 2f, 0f, -2f }));
} }
[Test] [Test]
@ -55,7 +62,10 @@ public class PolygonByCircleConstraintTest
int expectedSize = 12 * 3; int expectedSize = 12 * 3;
float[] polygon = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 }; float[] polygon = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 };
RcVec3f center = new RcVec3f(-1, 0, -1); RcVec3f center = new RcVec3f(-1, 0, -1);
float[] constrained = _constraint.Apply(polygon, center, 2); var radius = 2;
float[] buffer = new float[128];
Span<float> constrained = _constraint.Apply(polygon, center, radius, buffer);
Assert.That(constrained.Length, Is.EqualTo(expectedSize)); Assert.That(constrained.Length, Is.EqualTo(expectedSize));
@ -73,10 +83,13 @@ public class PolygonByCircleConstraintTest
int expectedSize = 9 * 3; int expectedSize = 9 * 3;
float[] polygon = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 }; float[] polygon = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 };
RcVec3f center = new RcVec3f(-2, 0, -1); RcVec3f center = new RcVec3f(-2, 0, -1);
float[] constrained = _constraint.Apply(polygon, center, 3); var radius = 3;
float[] buffer = new float[128];
Span<float> constrained = _constraint.Apply(polygon, center, radius, buffer);
Assert.That(constrained.Length, Is.EqualTo(expectedSize)); Assert.That(constrained.Length, Is.EqualTo(expectedSize));
Assert.That(constrained, Is.SupersetOf(new[] { -2f, 0f, -4f, -4f, 0f, 0f, -3.4641016f, 0.0f, 1.60769534f, -2.0f, 0.0f, 2.0f })); Assert.That(constrained.ToArray(), Is.SupersetOf(new[] { -2f, 0f, -4f, -4f, 0f, 0f, -3.4641016f, 0.0f, 1.60769534f, -2.0f, 0.0f, 2.0f }));
} }
[Test] [Test]
@ -85,9 +98,12 @@ public class PolygonByCircleConstraintTest
int expectedSize = 7 * 3; int expectedSize = 7 * 3;
float[] polygon = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 }; float[] polygon = { -4, 0, 0, -3, 0, 3, 2, 0, 3, 3, 0, -3, -2, 0, -4 };
RcVec3f center = new RcVec3f(4, 0, 0); RcVec3f center = new RcVec3f(4, 0, 0);
float[] constrained = _constraint.Apply(polygon, center, 4); var radius = 4;
float[] buffer = new float[128];
Span<float> constrained = _constraint.Apply(polygon, center, radius, buffer);
Assert.That(constrained.Length, Is.EqualTo(expectedSize)); Assert.That(constrained.Length, Is.EqualTo(expectedSize));
Assert.That(constrained, Is.SupersetOf(new[] { 1.53589869f, 0f, 3f, 2f, 0f, 3f, 3f, 0f, -3f })); Assert.That(constrained.ToArray(), Is.SupersetOf(new[] { 1.53589869f, 0f, 3f, 2f, 0f, 3f, 3f, 0f, -3f }));
} }
} }