df4e4949c8
* Fix crash in CTaskMngr::startFrame() * Fix crash in EventsMngr::executeEvents() * Fix loops * Fix loop iterators signess
287 lines
5.2 KiB
C++
Executable File
287 lines
5.2 KiB
C++
Executable File
// vim: set ts=4 sw=4 tw=99 noet:
|
|
//
|
|
// AMX Mod X, based on AMX Mod by Aleksander Naszko ("OLO").
|
|
// Copyright (C) The AMX Mod X Development Team.
|
|
//
|
|
// This software is licensed under the GNU General Public License, version 3 or higher.
|
|
// Additional exceptions apply. For full license details, see LICENSE.txt or visit:
|
|
// https://alliedmods.net/amxmodx-license
|
|
|
|
#include "amxmodx.h"
|
|
#include "CTask.h"
|
|
|
|
/*********************** CTask ***********************/
|
|
|
|
void CTaskMngr::CTask::set(CPluginMngr::CPlugin *pPlugin, int iFunc, int iFlags, cell iId, float fBase, int iParamsLen, const cell *pParams, int iRepeat, float fCurrentTime)
|
|
{
|
|
clear();
|
|
m_bFree = false;
|
|
|
|
m_pPlugin = pPlugin;
|
|
m_iFunc = iFunc;
|
|
m_iId = iId;
|
|
m_fBase = fBase;
|
|
m_bInExecute = false;
|
|
|
|
if (iFlags & 2)
|
|
{
|
|
m_bLoop = true;
|
|
m_iRepeat = -1;
|
|
}
|
|
else if (iFlags & 1)
|
|
{
|
|
m_bLoop = true;
|
|
m_iRepeat = iRepeat;
|
|
}
|
|
|
|
m_bAfterStart = (iFlags & 4) ? true : false;
|
|
m_bBeforeEnd = (iFlags & 8) ? true : false;
|
|
|
|
m_fNextExecTime = fCurrentTime + m_fBase;
|
|
|
|
if (iParamsLen)
|
|
{
|
|
m_iParamLen = iParamsLen + 1;
|
|
m_pParams = new cell[m_iParamLen];
|
|
memcpy(m_pParams, pParams, sizeof(cell)*iParamsLen);
|
|
m_pParams[iParamsLen] = 0;
|
|
} else {
|
|
m_iParamLen = 0;
|
|
m_pParams = NULL;
|
|
}
|
|
}
|
|
|
|
void CTaskMngr::CTask::clear()
|
|
{
|
|
m_bFree = true;
|
|
|
|
if (m_iFunc >= 0)
|
|
{
|
|
unregisterSPForward(m_iFunc);
|
|
m_iFunc = -1;
|
|
}
|
|
|
|
if (m_pParams)
|
|
{
|
|
delete [] m_pParams;
|
|
m_pParams = NULL;
|
|
}
|
|
|
|
m_pPlugin = NULL;
|
|
m_iId = 0;
|
|
m_fBase = 0.0f;
|
|
|
|
m_iRepeat = 0;
|
|
m_bLoop = false;
|
|
m_bAfterStart = false;
|
|
m_bBeforeEnd = false;
|
|
|
|
m_fNextExecTime = 0.0f;
|
|
}
|
|
|
|
bool CTaskMngr::CTask::isFree() const
|
|
{
|
|
return m_bFree;
|
|
}
|
|
|
|
void CTaskMngr::CTask::changeBase(float fNewBase)
|
|
{
|
|
m_fBase = fNewBase;
|
|
}
|
|
|
|
void CTaskMngr::CTask::resetNextExecTime(float fCurrentTime)
|
|
{
|
|
// If we're here while we're executing we would add m_fBase twice
|
|
if (!m_bInExecute)
|
|
m_fNextExecTime = fCurrentTime + m_fBase;
|
|
}
|
|
|
|
void CTaskMngr::CTask::executeIfRequired(float fCurrentTime, float fTimeLimit, float fTimeLeft)
|
|
{
|
|
bool execute = false;
|
|
bool done = false;
|
|
|
|
if (m_bAfterStart)
|
|
{
|
|
if (fCurrentTime - fTimeLeft + 1.0f >= m_fBase)
|
|
execute = true;
|
|
}
|
|
else if (m_bBeforeEnd)
|
|
{
|
|
if (fTimeLimit != 0.0f && (fTimeLeft + fTimeLimit * 60.0f) - fCurrentTime - 1.0f <= m_fBase)
|
|
execute = true;
|
|
}
|
|
else if (m_fNextExecTime <= fCurrentTime)
|
|
{
|
|
execute = true;
|
|
}
|
|
|
|
if (execute)
|
|
{
|
|
//only bother calling if we have something to call
|
|
if (!(m_bLoop && !m_iRepeat))
|
|
{
|
|
m_bInExecute = true;
|
|
if (m_iParamLen) // call with parameters
|
|
{
|
|
cell arr = prepareCellArray(m_pParams, m_iParamLen);
|
|
executeForwards(m_iFunc, arr, m_iId);
|
|
} else {
|
|
executeForwards(m_iFunc, m_iId);
|
|
}
|
|
m_bInExecute = false;
|
|
}
|
|
|
|
if (isFree())
|
|
return;
|
|
|
|
// set new exec time OR remove the task if needed
|
|
if (m_bLoop)
|
|
{
|
|
if (m_iRepeat != -1 && --m_iRepeat <= 0)
|
|
done = true;
|
|
} else {
|
|
done = true;
|
|
}
|
|
|
|
if (done)
|
|
{
|
|
clear();
|
|
} else {
|
|
m_fNextExecTime += m_fBase;
|
|
}
|
|
}
|
|
}
|
|
|
|
CTaskMngr::CTask::CTask()
|
|
{
|
|
m_bFree = true;
|
|
|
|
m_pPlugin = NULL;
|
|
m_iFunc = -1;
|
|
m_iId = 0;
|
|
m_fBase = 0.0f;
|
|
|
|
m_iRepeat = 0;
|
|
m_bLoop = false;
|
|
m_bAfterStart = false;
|
|
m_bBeforeEnd = false;
|
|
m_bInExecute = false;
|
|
|
|
m_fNextExecTime = 0.0f;
|
|
|
|
m_iParamLen = 0;
|
|
m_pParams = NULL;
|
|
}
|
|
|
|
CTaskMngr::CTask::~CTask()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
/*********************** CTaskMngr ***********************/
|
|
|
|
CTaskMngr::CTaskMngr()
|
|
{
|
|
m_pTmr_CurrentTime = NULL;
|
|
m_pTmr_TimeLimit = NULL;
|
|
m_pTmr_TimeLeft = NULL;
|
|
}
|
|
|
|
CTaskMngr::~CTaskMngr()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
void CTaskMngr::registerTimers(float *pCurrentTime, float *pTimeLimit, float *pTimeLeft)
|
|
{
|
|
m_pTmr_CurrentTime = pCurrentTime;
|
|
m_pTmr_TimeLimit = pTimeLimit;
|
|
m_pTmr_TimeLeft = pTimeLeft;
|
|
}
|
|
|
|
void CTaskMngr::registerTask(CPluginMngr::CPlugin *pPlugin, int iFunc, int iFlags, cell iId, float fBase, int iParamsLen, const cell *pParams, int iRepeat)
|
|
{
|
|
// first, search for free tasks
|
|
for (auto &task : m_Tasks)
|
|
{
|
|
if (task->isFree() && !task->inExecute())
|
|
{
|
|
// found: reuse it
|
|
task->set(pPlugin, iFunc, iFlags, iId, fBase, iParamsLen, pParams, iRepeat, *m_pTmr_CurrentTime);
|
|
return;
|
|
}
|
|
}
|
|
// not found: make a new one
|
|
auto task = ke::AutoPtr<CTask>(new CTask);
|
|
|
|
if (!task)
|
|
return;
|
|
|
|
task->set(pPlugin, iFunc, iFlags, iId, fBase, iParamsLen, pParams, iRepeat, *m_pTmr_CurrentTime);
|
|
m_Tasks.append(ke::Move(task));
|
|
}
|
|
|
|
int CTaskMngr::removeTasks(int iId, AMX *pAmx)
|
|
{
|
|
int i = 0;
|
|
|
|
for (auto &task : m_Tasks)
|
|
{
|
|
if (task->match(iId, pAmx))
|
|
{
|
|
task->clear();
|
|
++i;
|
|
}
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
int CTaskMngr::changeTasks(int iId, AMX *pAmx, float fNewBase)
|
|
{
|
|
int i = 0;
|
|
|
|
for (auto &task : m_Tasks)
|
|
{
|
|
if (task->match(iId, pAmx))
|
|
{
|
|
task->changeBase(fNewBase);
|
|
task->resetNextExecTime(*m_pTmr_CurrentTime);
|
|
++i;
|
|
}
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
bool CTaskMngr::taskExists(int iId, AMX *pAmx)
|
|
{
|
|
for (auto &task : m_Tasks)
|
|
{
|
|
if (task->match(iId, pAmx))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CTaskMngr::startFrame()
|
|
{
|
|
auto lastSize = m_Tasks.length();
|
|
for(auto i = 0u; i < lastSize; i++)
|
|
{
|
|
auto &task = m_Tasks[i];
|
|
|
|
if (task->isFree())
|
|
continue;
|
|
task->executeIfRequired(*m_pTmr_CurrentTime, *m_pTmr_TimeLimit, *m_pTmr_TimeLeft);
|
|
}
|
|
}
|
|
|
|
void CTaskMngr::clear()
|
|
{
|
|
m_Tasks.clear();
|
|
}
|