| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172 |
- /* Copyright The kNet Project.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License. */
- #pragma once
- /** @file PolledTimer.h
- @brief PolledTimer is a timer object that can be used in "polled" or per-frame executed contexts. */
- #include "Clock.h"
- namespace kNet
- {
- class PolledTimer
- {
- public:
- /// The default ctor starts the timer when the object is created.
- PolledTimer():enabled(false)
- {
- startTime = Clock::Tick();
- }
- explicit PolledTimer(float msecs)
- :enabled(false)
- {
- StartMSecs(msecs);
- }
- /// Starts the timer in a non-periodic mode, to go off once in the given amount of milliseconds.
- void StartMSecs(float msecs)
- {
- StartTicks((tick_t)(Clock::TicksPerSec() * (msecs / 1000.f)));
- }
- /// Starts the timer in a non-periodic mode, to go off once in the given amount of high-precision Clock ticks.
- void StartTicks(tick_t ticks)
- {
- startTime = Clock::Tick();
- alarmTime = startTime + ticks;
- enabled = true;
- }
- void Stop()
- {
- enabled = false;
- }
- void Reset()
- {
- Stop();
- }
- /// Returns the amount of time in ticks this timer has been running.
- tick_t TicksElapsed() const
- {
- return Clock::Tick() - startTime;
- }
- float MSecsElapsed() const
- {
- return Clock::TicksToMillisecondsF(TicksElapsed());
- }
- /// Starts the timer from zero to run upwards without a target period, i.e. the timer will never go off,
- /// but will only report TicksElapsed/MSecsElapsed values.
- void Start()
- {
- enabled = false; // The target-timer is disabled.
- startTime = Clock::Tick(); // Start counting from now.
- }
- bool Enabled() const
- {
- return enabled;
- }
- /// Tests whether the timer has gone off, and resets it as well.
- /// Returns true if the timer has elapsed, false otherwise.
- bool Test()
- {
- if (!enabled)
- return false;
- if (Clock::IsNewer(Clock::Tick(), alarmTime))
- {
- Reset();
- return true;
- }
- return false;
- }
- bool TriggeredOrNotRunning()
- {
- return Test() || !Enabled();
- }
- /// @return The number of ticks left until the timer elapses, or (tick_t)-1 if the timer is not active.
- tick_t TicksLeft() const
- {
- if (!enabled)
- return (tick_t)(-1);
- tick_t now = Clock::Tick();
- if (Clock::IsNewer(now, alarmTime))
- return 0;
- else
- return Clock::TicksInBetween(alarmTime, now);
- }
- /// @return The number of msecs left until the timer elapses, 0.f if the timer has elapsed, or
- /// -1.0f if the timer is not even active.
- float MSecsLeft() const
- {
- if (!enabled)
- return -1.f;
- tick_t now = Clock::Tick();
- if (Clock::IsNewer(now, alarmTime))
- return 0.f;
- else
- return Clock::TimespanToMillisecondsF(now, alarmTime);
- }
- /// Waits until the timer elapses. Uses OS Clock::Sleep() to avoid excessive CPU use.
- void WaitPrecise()
- {
- if (!enabled)
- return;
- tick_t timeLeft = TicksLeft();
- while(timeLeft > 0)
- {
- if (timeLeft > Clock::TicksPerMillisecond())
- {
- float msecs = Clock::TicksToMillisecondsF(timeLeft);
- Clock::Sleep((int)msecs);
- }
- else // If there's less than one ms left, we use spinwait to return precisely as near to the correct tick as possible.
- {
- SpinWait();
- return;
- }
- timeLeft = TicksLeft();
- }
- }
- /// Waits in an empty loop until the timer elapses. Will cause the the CPU use to go to max.
- void SpinWait()
- {
- while(enabled && TicksLeft() > 0)
- ;
- }
- private:
- bool enabled;
- /// The wallclock time when this PolledTimer is supposed to go off.
- tick_t alarmTime;
- /// The wallclock time when this PolledTimer was last started.
- tick_t startTime;
- };
- } // ~kNet
|