/* ============================================================================== This file is part of the JUCE library. Copyright (c) 2022 - Raw Material Software Limited JUCE is an open source library subject to commercial or open-source licensing. The code included in this file is provided under the terms of the ISC license http://www.isc.org/downloads/software-support-policy/isc-license. Permission To use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted provided that the above copyright notice and this permission notice appear in all copies. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE DISCLAIMED. ============================================================================== */ namespace juce { //============================================================================== class HighResolutionTimer::Impl : private PlatformTimerListener { public: explicit Impl (HighResolutionTimer& o) : owner { o } {} void startTimer (int newIntervalMs) { shouldCancelCallbacks.store (true); const auto shouldWaitForPendingCallbacks = [&] { const std::scoped_lock lock { timerMutex }; if (timer.getIntervalMs() > 0) timer.cancelTimer(); jassert (timer.getIntervalMs() == 0); if (newIntervalMs > 0) timer.startTimer (jmax (0, newIntervalMs)); return callbackThreadId != std::this_thread::get_id() && timer.getIntervalMs() <= 0; }(); if (shouldWaitForPendingCallbacks) std::scoped_lock lock { callbackMutex }; } int getIntervalMs() const { const std::scoped_lock lock { timerMutex }; return timer.getIntervalMs(); } bool isTimerRunning() const { return getIntervalMs() > 0; } private: void onTimerExpired() final { callbackThreadId.store (std::this_thread::get_id()); { std::scoped_lock lock { callbackMutex }; if (isTimerRunning()) { try { owner.hiResTimerCallback(); } catch (...) { // Exceptions thrown in a timer callback won't be // propagated to the main thread, it's best to find // a way to avoid them if possible jassertfalse; } } } callbackThreadId.store ({}); } HighResolutionTimer& owner; mutable std::mutex timerMutex; std::mutex callbackMutex; std::atomic callbackThreadId{}; std::atomic shouldCancelCallbacks { false }; PlatformTimer timer { *this }; JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Impl) JUCE_DECLARE_NON_MOVEABLE (Impl) }; //============================================================================== HighResolutionTimer::HighResolutionTimer() : impl (std::make_unique (*this)) {} HighResolutionTimer::~HighResolutionTimer() { // You *must* call stopTimer from the derived class destructor to // avoid data races on the timer's vtable jassert (! isTimerRunning()); stopTimer(); } void HighResolutionTimer::startTimer (int newIntervalMs) { impl->startTimer (newIntervalMs); } void HighResolutionTimer::stopTimer() { impl->startTimer (0); } int HighResolutionTimer::getTimerInterval() const noexcept { return impl->getIntervalMs(); } bool HighResolutionTimer::isTimerRunning() const noexcept { return impl->isTimerRunning(); } //============================================================================== #if JUCE_UNIT_TESTS class HighResolutionTimerTests final : public UnitTest { public: HighResolutionTimerTests() : UnitTest ("HighResolutionTimer", UnitTestCategories::threads) {} void runTest() override { constexpr int maximumTimeoutMs {30'000}; beginTest ("Start/stop a timer"); { WaitableEvent timerFiredOnce; WaitableEvent timerFiredTwice; Timer timer {[&, callbackCount = 0]() mutable { switch (++callbackCount) { case 1: timerFiredOnce.signal(); return; case 2: timerFiredTwice.signal(); return; default: return; } }}; expect (! timer.isTimerRunning()); expect (timer.getTimerInterval() == 0); timer.startTimer (1); expect (timer.isTimerRunning()); expect (timer.getTimerInterval() == 1); expect (timerFiredOnce.wait (maximumTimeoutMs)); expect (timerFiredTwice.wait (maximumTimeoutMs)); timer.stopTimer(); expect (! timer.isTimerRunning()); expect (timer.getTimerInterval() == 0); } beginTest ("Stop a timer from the timer callback"); { WaitableEvent stoppedTimer; auto timerCallback = [&] (Timer& timer) { expect (timer.isTimerRunning()); timer.stopTimer(); expect (! timer.isTimerRunning()); stoppedTimer.signal(); }; Timer timer {[&]{ timerCallback (timer); }}; timer.startTimer (1); expect (stoppedTimer.wait (maximumTimeoutMs)); } beginTest ("Restart a timer from the timer callback"); { WaitableEvent restartTimer; WaitableEvent timerRestarted; WaitableEvent timerFiredAfterRestart; Timer timer {[&, callbackCount = 0]() mutable { switch (++callbackCount) { case 1: expect (restartTimer.wait (maximumTimeoutMs)); expect (timer.getTimerInterval() == 1); timer.startTimer (2); expect (timer.getTimerInterval() == 2); timerRestarted.signal(); return; case 2: expect (timer.getTimerInterval() == 2); timerFiredAfterRestart.signal(); return; default: return; } }}; timer.startTimer (1); expect (timer.getTimerInterval() == 1); restartTimer.signal(); expect (timerRestarted.wait (maximumTimeoutMs)); expect (timer.getTimerInterval() == 2); expect (timerFiredAfterRestart.wait (maximumTimeoutMs)); timer.stopTimer(); } beginTest ("Calling stopTimer on a timer, waits for any timer callbacks to finish"); { WaitableEvent timerCallbackStarted; WaitableEvent stoppingTimer; std::atomic timerCallbackFinished { false }; Timer timer {[&, callbackCount = 0]() mutable { switch (++callbackCount) { case 1: timerCallbackStarted.signal(); expect (stoppingTimer.wait (maximumTimeoutMs)); Thread::sleep (10); timerCallbackFinished = true; return; default: return; } }}; timer.startTimer (1); expect (timerCallbackStarted.wait (maximumTimeoutMs)); stoppingTimer.signal(); timer.stopTimer(); expect (timerCallbackFinished); } beginTest ("Calling stopTimer on a timer, waits for any timer callbacks to finish, even if the timer callback calls stopTimer first"); { WaitableEvent stoppedFromInsideTimerCallback; WaitableEvent stoppingFromOutsideTimerCallback; std::atomic timerCallbackFinished { false }; Timer timer {[&]() { timer.stopTimer(); stoppedFromInsideTimerCallback.signal(); expect (stoppingFromOutsideTimerCallback.wait (maximumTimeoutMs)); Thread::sleep (10); timerCallbackFinished = true; }}; timer.startTimer (1); expect (stoppedFromInsideTimerCallback.wait (maximumTimeoutMs)); stoppingFromOutsideTimerCallback.signal(); timer.stopTimer(); expect (timerCallbackFinished); } beginTest ("Adjusting a timer period from outside the timer callback doesn't cause data races"); { WaitableEvent timerCallbackStarted; WaitableEvent timerRestarted; WaitableEvent timerFiredAfterRestart; std::atomic lastCallbackCount {0}; Timer timer {[&, callbackCount = 0]() mutable { switch (++callbackCount) { case 1: expect (timer.getTimerInterval() == 1); timerCallbackStarted.signal(); Thread::sleep (10); lastCallbackCount = 1; return; case 2: expect (timerRestarted.wait (maximumTimeoutMs)); expect (timer.getTimerInterval() == 2); lastCallbackCount = 2; timerFiredAfterRestart.signal(); return; default: return; } }}; timer.startTimer (1); expect (timerCallbackStarted.wait (maximumTimeoutMs)); timer.startTimer (2); timerRestarted.signal(); expect (timerFiredAfterRestart.wait (maximumTimeoutMs)); expect (lastCallbackCount == 2); timer.stopTimer(); expect (lastCallbackCount == 2); } beginTest ("A timer can be restarted externally, after being stopped internally"); { WaitableEvent timerStopped; WaitableEvent timerFiredAfterRestart; Timer timer {[&, callbackCount = 0]() mutable { switch (++callbackCount) { case 1: timer.stopTimer(); timerStopped.signal(); return; case 2: timerFiredAfterRestart.signal(); return; default: return; } }}; expect (! timer.isTimerRunning()); timer.startTimer (1); expect (timer.isTimerRunning()); expect (timerStopped.wait (maximumTimeoutMs)); expect (! timer.isTimerRunning()); timer.startTimer (1); expect (timer.isTimerRunning()); expect (timerFiredAfterRestart.wait (maximumTimeoutMs)); } beginTest ("Calls to `startTimer` and `getTimerInterval` succeed while a callback is blocked"); { WaitableEvent timerBlocked; WaitableEvent unblockTimer; Timer timer {[&] { timerBlocked.signal(); unblockTimer.wait(); timer.stopTimer(); }}; timer.startTimer (1); timerBlocked.wait(); expect (timer.getTimerInterval() == 1); timer.startTimer (2); expect (timer.getTimerInterval() == 2); unblockTimer.signal(); timer.stopTimer(); } beginTest ("Stress test"); { constexpr auto maxNumTimers { 100 }; std::vector> timers; timers.reserve (maxNumTimers); for (int i = 0; i < maxNumTimers; ++i) { auto timer = std::make_unique ([]{}); timer->startTimer (1); if (! timer->isTimerRunning()) break; timers.push_back (std::move (timer)); } expect (timers.size() >= 16); } } class Timer final : public HighResolutionTimer { public: explicit Timer (std::function fn) : callback (std::move (fn)) {} ~Timer() override { stopTimer(); } void hiResTimerCallback() override { callback(); } private: std::function callback; }; }; static HighResolutionTimerTests highResolutionTimerTests; #endif } // namespace juce