2021-10-02 10:28:49 +00:00
|
|
|
/***
|
|
|
|
Copyright (C) 2021 J Reece Wilson (a/k/a "Reece"). All rights reserved.
|
|
|
|
|
|
|
|
File: LSTimer.NT.cpp
|
|
|
|
Date: 2021-10-1
|
|
|
|
Author: Reece
|
|
|
|
***/
|
|
|
|
#include <Source/RuntimeInternal.hpp>
|
|
|
|
#include "LSTimer.hpp"
|
2022-04-18 23:51:36 +00:00
|
|
|
#include <Source/Time/Time.hpp>
|
2021-10-02 10:28:49 +00:00
|
|
|
|
2022-06-11 23:52:46 +00:00
|
|
|
namespace Aurora::IO::Loop
|
2021-10-02 10:28:49 +00:00
|
|
|
{
|
2022-05-04 13:00:26 +00:00
|
|
|
LSTimer::LSTimer(AuUInt32 reschedStepMsOrZero, AuUInt32 maxIterationsOrZero, bool bSingleshot, HANDLE handle) :
|
2022-04-18 23:51:36 +00:00
|
|
|
LSHandle(AuReinterpretCast<AuUInt>(handle)),
|
2022-05-03 17:42:14 +00:00
|
|
|
reschedStepNsOrZero_(AuMSToNS<AuUInt64>(reschedStepMsOrZero)),
|
2022-05-04 13:00:26 +00:00
|
|
|
bSingleshot(bSingleshot),
|
2022-04-18 23:51:36 +00:00
|
|
|
maxIterationsOrZero_(maxIterationsOrZero)
|
|
|
|
{
|
|
|
|
this->targetTime_ = AuTime::ConvertTimestamp(AuTime::CurrentClockMS());
|
|
|
|
}
|
|
|
|
|
|
|
|
LSTimer::~LSTimer()
|
|
|
|
{
|
|
|
|
auto handle = AuReinterpretCast<HANDLE>(this->handle);
|
|
|
|
AuWin32CloseHandle(handle);
|
|
|
|
this->handle = kInvalidHandle;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LSTimer::UpdateTime(AuUInt64 absTimeMs)
|
|
|
|
{
|
|
|
|
this->targetTime_ = AuTime::ConvertTimestamp(absTimeMs);
|
|
|
|
UpdateTimeInternal(this->targetTime_);
|
|
|
|
}
|
|
|
|
|
2022-05-03 17:42:14 +00:00
|
|
|
void LSTimer::UpdateTimeNs(AuUInt64 absTimeNs)
|
|
|
|
{
|
|
|
|
this->targetTime_ = AuTime::ConvertTimestampNs(absTimeNs);
|
|
|
|
UpdateTimeInternal(this->targetTime_);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LSTimer::UpdateTimeInternal(AuUInt64 absTimeNs)
|
2022-04-18 23:51:36 +00:00
|
|
|
{
|
|
|
|
LARGE_INTEGER i;
|
2022-05-03 17:42:14 +00:00
|
|
|
this->targetTime_ = absTimeNs;
|
|
|
|
i.QuadPart = absTimeNs;
|
2022-04-18 23:51:36 +00:00
|
|
|
SysAssert(::SetWaitableTimer(AuReinterpretCast<HANDLE>(this->handle), &i, 0, nullptr, nullptr, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
void LSTimer::UpdateTickRateIfAny(AuUInt32 reschedStepMsOrZero, AuUInt32 maxIterationsOrZero)
|
|
|
|
{
|
2022-05-03 17:42:14 +00:00
|
|
|
this->reschedStepNsOrZero_ = AuMSToNS<AuUInt64>(reschedStepMsOrZero);
|
|
|
|
this->maxIterationsOrZero_ = maxIterationsOrZero;
|
|
|
|
this->count_ = 0;
|
2022-06-11 23:01:27 +00:00
|
|
|
UpdateTimeInternal(this->targetTime_);
|
2022-05-03 17:42:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LSTimer::UpdateTickRateIfAnyNs(AuUInt64 reschedStepNsOrZero, AuUInt32 maxIterationsOrZero)
|
|
|
|
{
|
|
|
|
this->reschedStepNsOrZero_ = reschedStepNsOrZero;
|
2022-04-18 23:51:36 +00:00
|
|
|
this->maxIterationsOrZero_ = maxIterationsOrZero;
|
|
|
|
this->count_ = 0;
|
2022-06-11 23:01:27 +00:00
|
|
|
UpdateTimeInternal(this->targetTime_);
|
2022-04-18 23:51:36 +00:00
|
|
|
}
|
2021-10-03 12:47:16 +00:00
|
|
|
|
2022-04-18 23:51:36 +00:00
|
|
|
bool LSTimer::OnTrigger(AuUInt handle)
|
|
|
|
{
|
2022-05-03 17:42:14 +00:00
|
|
|
SysAssert(this->targetTime_ <= AuTime::ConvertTimestampNs(AuTime::CurrentClockNS()));
|
2022-05-04 13:00:26 +00:00
|
|
|
|
2022-05-03 17:42:14 +00:00
|
|
|
if (!this->reschedStepNsOrZero_)
|
2022-04-18 23:51:36 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this->maxIterationsOrZero_)
|
|
|
|
{
|
2022-05-03 17:42:14 +00:00
|
|
|
this->UpdateTimeInternal(this->targetTime_ + (AuUInt64(this->reschedStepNsOrZero_) / 100ULL));
|
2022-04-18 23:51:36 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-04-19 01:52:11 +00:00
|
|
|
bool ok = AuAtomicAdd<AuUInt32>(&this->count_, 1) <= this->maxIterationsOrZero_;
|
2022-04-18 23:51:36 +00:00
|
|
|
if (ok)
|
|
|
|
{
|
2022-05-03 17:42:14 +00:00
|
|
|
this->UpdateTimeInternal(this->targetTime_ + (AuUInt64(this->reschedStepNsOrZero_) / 100ULL));
|
2022-04-18 23:51:36 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Stop();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LSTimer::Stop()
|
|
|
|
{
|
|
|
|
bool bSuccess = ::CancelWaitableTimer(AuReinterpretCast<HANDLE>(handle));
|
|
|
|
SysAssertDbg(bSuccess);
|
|
|
|
LARGE_INTEGER i;
|
|
|
|
i.QuadPart = AuNumericLimits<AuInt64>::max();
|
|
|
|
SysAssert(::SetWaitableTimer(AuReinterpretCast<HANDLE>(this->handle), &i, 0, nullptr, nullptr, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LSTimer::IsSignaled()
|
|
|
|
{
|
|
|
|
return LSHandle::IsSignaled();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LSTimer::WaitOn(AuUInt32 timeout)
|
|
|
|
{
|
|
|
|
return LSHandle::WaitOn(timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
ELoopSource LSTimer::GetType()
|
|
|
|
{
|
|
|
|
return ELoopSource::eSourceTimer;
|
|
|
|
}
|
|
|
|
|
2022-05-04 13:00:26 +00:00
|
|
|
AUKN_SYM AuSPtr<ITimer> NewLSTimer(AuUInt64 absStartTimeMs, AuUInt32 reschedStepMsOrZero, AuUInt32 maxIterationsOrZero, bool bSingleshot)
|
2022-04-18 23:51:36 +00:00
|
|
|
{
|
|
|
|
// https://docs.microsoft.com/en-us/windows/win32/api/threadpoolapiset/nf-threadpoolapiset-setthreadpoolwait
|
|
|
|
// TODO: Is that any better with an event?
|
|
|
|
|
2022-05-04 13:00:26 +00:00
|
|
|
auto handle = CreateWaitableTimerW(nullptr, !bSingleshot, nullptr);
|
2022-04-18 23:51:36 +00:00
|
|
|
if (!handle)
|
|
|
|
{
|
|
|
|
SysPushErrorIO();
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-05-04 13:00:26 +00:00
|
|
|
auto object = AuMakeShared<LSTimer>(reschedStepMsOrZero, maxIterationsOrZero, bSingleshot, handle);
|
2022-04-18 23:51:36 +00:00
|
|
|
if (!object)
|
|
|
|
{
|
|
|
|
SysPushErrorMem();
|
|
|
|
AuWin32CloseHandle(handle);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2022-05-04 13:00:26 +00:00
|
|
|
if (absStartTimeMs)
|
|
|
|
{
|
|
|
|
object->UpdateTime(absStartTimeMs);
|
|
|
|
}
|
2022-04-18 23:51:36 +00:00
|
|
|
|
|
|
|
return object;
|
|
|
|
}
|
2021-10-02 10:28:49 +00:00
|
|
|
}
|