mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-25 01:23:52 +00:00
201 lines
4.6 KiB
C++
201 lines
4.6 KiB
C++
|
//
|
||
|
// Copyright (c) 2009-2010 Mikko Mononen memon@inside.org
|
||
|
//
|
||
|
// This software is provided 'as-is', without any express or implied
|
||
|
// warranty. In no event will the authors be held liable for any damages
|
||
|
// arising from the use of this software.
|
||
|
// Permission is granted to anyone to use this software for any purpose,
|
||
|
// including commercial applications, and to alter it and redistribute it
|
||
|
// freely, subject to the following restrictions:
|
||
|
// 1. The origin of this software must not be misrepresented; you must not
|
||
|
// claim that you wrote the original software. If you use this software
|
||
|
// in a product, an acknowledgment in the product documentation would be
|
||
|
// appreciated but is not required.
|
||
|
// 2. Altered source versions must be plainly marked as such, and must not be
|
||
|
// misrepresented as being the original software.
|
||
|
// 3. This notice may not be removed or altered from any source distribution.
|
||
|
//
|
||
|
|
||
|
#include <string.h>
|
||
|
#include "DetourPathQueue.h"
|
||
|
#include "DetourNavMesh.h"
|
||
|
#include "DetourNavMeshQuery.h"
|
||
|
#include "DetourAlloc.h"
|
||
|
#include "DetourCommon.h"
|
||
|
|
||
|
|
||
|
dtPathQueue::dtPathQueue() :
|
||
|
m_nextHandle(1),
|
||
|
m_maxPathSize(0),
|
||
|
m_queueHead(0),
|
||
|
m_navquery(0)
|
||
|
{
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
m_queue[i].path = 0;
|
||
|
}
|
||
|
|
||
|
dtPathQueue::~dtPathQueue()
|
||
|
{
|
||
|
purge();
|
||
|
}
|
||
|
|
||
|
void dtPathQueue::purge()
|
||
|
{
|
||
|
dtFreeNavMeshQuery(m_navquery);
|
||
|
m_navquery = 0;
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
{
|
||
|
dtFree(m_queue[i].path);
|
||
|
m_queue[i].path = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool dtPathQueue::init(const int maxPathSize, const int maxSearchNodeCount, dtNavMesh* nav)
|
||
|
{
|
||
|
purge();
|
||
|
|
||
|
m_navquery = dtAllocNavMeshQuery();
|
||
|
if (!m_navquery)
|
||
|
return false;
|
||
|
if (dtStatusFailed(m_navquery->init(nav, maxSearchNodeCount)))
|
||
|
return false;
|
||
|
|
||
|
m_maxPathSize = maxPathSize;
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
{
|
||
|
m_queue[i].ref = DT_PATHQ_INVALID;
|
||
|
m_queue[i].path = (dtPolyRef*)dtAlloc(sizeof(dtPolyRef)*m_maxPathSize, DT_ALLOC_PERM);
|
||
|
if (!m_queue[i].path)
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
m_queueHead = 0;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void dtPathQueue::update(const int maxIters)
|
||
|
{
|
||
|
static const int MAX_KEEP_ALIVE = 2; // in update ticks.
|
||
|
|
||
|
// Update path request until there is nothing to update
|
||
|
// or upto maxIters pathfinder iterations has been consumed.
|
||
|
int iterCount = maxIters;
|
||
|
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
{
|
||
|
PathQuery& q = m_queue[m_queueHead % MAX_QUEUE];
|
||
|
|
||
|
// Skip inactive requests.
|
||
|
if (q.ref == DT_PATHQ_INVALID)
|
||
|
{
|
||
|
m_queueHead++;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Handle completed request.
|
||
|
if (dtStatusSucceed(q.status) || dtStatusFailed(q.status))
|
||
|
{
|
||
|
// If the path result has not been read in few frames, free the slot.
|
||
|
q.keepAlive++;
|
||
|
if (q.keepAlive > MAX_KEEP_ALIVE)
|
||
|
{
|
||
|
q.ref = DT_PATHQ_INVALID;
|
||
|
q.status = 0;
|
||
|
}
|
||
|
|
||
|
m_queueHead++;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Handle query start.
|
||
|
if (q.status == 0)
|
||
|
{
|
||
|
q.status = m_navquery->initSlicedFindPath(q.startRef, q.endRef, q.startPos, q.endPos, q.filter);
|
||
|
}
|
||
|
// Handle query in progress.
|
||
|
if (dtStatusInProgress(q.status))
|
||
|
{
|
||
|
int iters = 0;
|
||
|
q.status = m_navquery->updateSlicedFindPath(iterCount, &iters);
|
||
|
iterCount -= iters;
|
||
|
}
|
||
|
if (dtStatusSucceed(q.status))
|
||
|
{
|
||
|
q.status = m_navquery->finalizeSlicedFindPath(q.path, &q.npath, m_maxPathSize);
|
||
|
}
|
||
|
|
||
|
if (iterCount <= 0)
|
||
|
break;
|
||
|
|
||
|
m_queueHead++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dtPathQueueRef dtPathQueue::request(dtPolyRef startRef, dtPolyRef endRef,
|
||
|
const float* startPos, const float* endPos,
|
||
|
const dtQueryFilter* filter)
|
||
|
{
|
||
|
// Find empty slot
|
||
|
int slot = -1;
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
{
|
||
|
if (m_queue[i].ref == DT_PATHQ_INVALID)
|
||
|
{
|
||
|
slot = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
// Could not find slot.
|
||
|
if (slot == -1)
|
||
|
return DT_PATHQ_INVALID;
|
||
|
|
||
|
dtPathQueueRef ref = m_nextHandle++;
|
||
|
if (m_nextHandle == DT_PATHQ_INVALID) m_nextHandle++;
|
||
|
|
||
|
PathQuery& q = m_queue[slot];
|
||
|
q.ref = ref;
|
||
|
dtVcopy(q.startPos, startPos);
|
||
|
q.startRef = startRef;
|
||
|
dtVcopy(q.endPos, endPos);
|
||
|
q.endRef = endRef;
|
||
|
|
||
|
q.status = 0;
|
||
|
q.npath = 0;
|
||
|
q.filter = filter;
|
||
|
q.keepAlive = 0;
|
||
|
|
||
|
return ref;
|
||
|
}
|
||
|
|
||
|
dtStatus dtPathQueue::getRequestStatus(dtPathQueueRef ref) const
|
||
|
{
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
{
|
||
|
if (m_queue[i].ref == ref)
|
||
|
return m_queue[i].status;
|
||
|
}
|
||
|
return DT_FAILURE;
|
||
|
}
|
||
|
|
||
|
dtStatus dtPathQueue::getPathResult(dtPathQueueRef ref, dtPolyRef* path, int* pathSize, const int maxPath)
|
||
|
{
|
||
|
for (int i = 0; i < MAX_QUEUE; ++i)
|
||
|
{
|
||
|
if (m_queue[i].ref == ref)
|
||
|
{
|
||
|
PathQuery& q = m_queue[i];
|
||
|
dtStatus details = q.status & DT_STATUS_DETAIL_MASK;
|
||
|
// Free request for reuse.
|
||
|
q.ref = DT_PATHQ_INVALID;
|
||
|
q.status = 0;
|
||
|
// Copy path
|
||
|
int n = dtMin(q.npath, maxPath);
|
||
|
memcpy(path, q.path, sizeof(dtPolyRef)*n);
|
||
|
*pathSize = n;
|
||
|
return details | DT_SUCCESS;
|
||
|
}
|
||
|
}
|
||
|
return DT_FAILURE;
|
||
|
}
|