// Copyright (c) 2011-2016 The Cryptonote developers // Distributed under the MIT/X11 software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include "Timer.h" #include #include #ifndef WIN32_LEAN_AND_MEAN #define WIN32_LEAN_AND_MEAN #endif #include #include #include "Dispatcher.h" namespace System { namespace { struct TimerContext { uint64_t time; NativeContext* context; bool interrupted; }; } Timer::Timer() : dispatcher(nullptr) { } Timer::Timer(Dispatcher& dispatcher) : dispatcher(&dispatcher), context(nullptr) { } Timer::Timer(Timer&& other) : dispatcher(other.dispatcher) { if (dispatcher != nullptr) { assert(other.context == nullptr); context = nullptr; other.dispatcher = nullptr; } } Timer::~Timer() { assert(dispatcher == nullptr || context == nullptr); } Timer& Timer::operator=(Timer&& other) { assert(dispatcher == nullptr || context == nullptr); dispatcher = other.dispatcher; if (dispatcher != nullptr) { assert(other.context == nullptr); context = nullptr; other.dispatcher = nullptr; } return *this; } void Timer::sleep(std::chrono::nanoseconds duration) { assert(dispatcher != nullptr); assert(context == nullptr); if (dispatcher->interrupted()) { throw InterruptedException(); } LARGE_INTEGER frequency; LARGE_INTEGER ticks; QueryPerformanceCounter(&ticks); QueryPerformanceFrequency(&frequency); uint64_t currentTime = ticks.QuadPart / (frequency.QuadPart / 1000); uint64_t time = currentTime + duration.count() / 1000000; TimerContext timerContext{ time, dispatcher->getCurrentContext(), false }; context = &timerContext; dispatcher->addTimer(time, dispatcher->getCurrentContext()); dispatcher->getCurrentContext()->interruptProcedure = [&]() { assert(dispatcher != nullptr); assert(context != nullptr); TimerContext* timerContext = static_cast(context); if (!timerContext->interrupted) { dispatcher->interruptTimer(timerContext->time, timerContext->context); timerContext->interrupted = true; } }; dispatcher->dispatch(); dispatcher->getCurrentContext()->interruptProcedure = nullptr; assert(timerContext.context == dispatcher->getCurrentContext()); assert(dispatcher != nullptr); assert(context == &timerContext); context = nullptr; if (timerContext.interrupted) { throw InterruptedException(); } } }