From 96faae54ea0bf72aa9bca604a4a38e374f55e03f Mon Sep 17 00:00:00 2001 From: Christian Speckner Date: Tue, 19 Sep 2023 22:57:22 +0200 Subject: [PATCH] Still defunct real time emulation worker. --- Makefile | 2 +- src/emucore/EmulationWorker.hxx | 3 +- src/os/rtstella/RTEmulationWorker.cxx | 257 ++++++++++++++++++++++++++ src/os/rtstella/RTEmulationWorker.hxx | 92 +++++++++ src/os/rtstella/Spinlock.cxx | 4 + src/os/rtstella/module.mk | 2 +- 6 files changed, 356 insertions(+), 4 deletions(-) create mode 100644 src/os/rtstella/RTEmulationWorker.cxx create mode 100644 src/os/rtstella/RTEmulationWorker.hxx diff --git a/Makefile b/Makefile index 111e9619e..93bdb007f 100644 --- a/Makefile +++ b/Makefile @@ -22,7 +22,7 @@ srcdir ?= . DEFINES := -DSDL_SUPPORT -D_GLIBCXX_USE_CXX11_ABI=1 -LDFLAGS := -pthread +LDFLAGS := -pthread -latomic INCLUDES := LIBS := OBJS := diff --git a/src/emucore/EmulationWorker.hxx b/src/emucore/EmulationWorker.hxx index eb696f18d..81ca35b4e 100644 --- a/src/emucore/EmulationWorker.hxx +++ b/src/emucore/EmulationWorker.hxx @@ -38,7 +38,6 @@ #ifndef EMULATION_WORKER_HXX #define EMULATION_WORKER_HXX -#include #include #include #include @@ -185,7 +184,7 @@ class EmulationWorker // Any pending signal (or Signal::none) Signal myPendingSignal{Signal::none}; // The initial access to myState is not synchronized -> make this atomic - std::atomic myState{State::initializing}; + State myState{State::initializing}; // Emulation parameters TIA* myTia{nullptr}; diff --git a/src/os/rtstella/RTEmulationWorker.cxx b/src/os/rtstella/RTEmulationWorker.cxx new file mode 100644 index 000000000..2026f0a7d --- /dev/null +++ b/src/os/rtstella/RTEmulationWorker.cxx @@ -0,0 +1,257 @@ +#include "RTEmulationWorker.hxx" + +#include +#include + +#include "TIA.hxx" +#include "DispatchResult.hxx" + +namespace { + constexpr uint64_t TIMESLICE_NANOSECONDS = 100000; + constexpr uint64_t MAX_LAG_NANOSECONDS = 10 * TIMESLICE_NANOSECONDS; + + Int64 timeDifferenceNanoseconds(const struct timespec& from, const struct timespec& to) + { + return (from.tv_sec - to.tv_sec) * 1000000000 + (from.tv_nsec - to.tv_nsec); + } +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +RTEmulationWorker::RTEmulationWorker() +{ + if (!myPendingSignal.is_lock_free()) { + cerr << "FATAL: atomic is not lock free" << endl << std::flush; + throw runtime_error("atomic is not lock free"); + } + + if (!myState.is_lock_free()) { + cerr << "FATAL: atomic is not lock free" << endl << std::flush; + throw runtime_error("atomic is not lock free"); + } + + myThread = std::thread( + &RTEmulationWorker::threadMain, this + ); + + while (myState == State::initializing) {} +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +RTEmulationWorker::~RTEmulationWorker() +{ + while (myPendingSignal != Signal::none) {} + + switch (myState) { + case State::exception: + break; + + case State::stopped: + { + std::unique_lock lock(myThreadIsRunningMutex); + myPendingSignal = Signal::quit; + } + + myWakeupCondition.notify_one(); + break; + + case State::running: + case State::paused: + myPendingSignal = Signal::quit; + break; + + case State::initializing: + throw runtime_error("unreachable"); + break; + } + + myThread.join(); + + rethrowPendingException(); +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +void RTEmulationWorker::suspend() +{ + while (myPendingSignal != Signal::none) {} + + if (myState != State::running || myState != State::paused) throw runtime_error("invalid state"); + + myPendingSignal = Signal::suspend; + + while (myState != State::paused) rethrowPendingException(); +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +void RTEmulationWorker::resume() +{ + while (myPendingSignal != Signal::none) {} + + if (myState != State::running || myState != State::paused) throw runtime_error("invalid state"); + + myPendingSignal = Signal::resume; + + while (myState != State::running) rethrowPendingException(); +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +void RTEmulationWorker::start(uInt32 cyclesPerSecond, DispatchResult* dispatchResult, TIA* tia) +{ + while (myPendingSignal != Signal::none) {} + + switch (myState) { + case State::exception: + rethrowPendingException(); + break; + + case State::stopped: + break; + + case State::running: + case State::paused: + stop(); + break; + + default: + throw runtime_error("unreachable"); + break; + } + + myTia = tia; + myDispatchResult = dispatchResult; + myCyclesPerSecond = cyclesPerSecond; + + myPendingSignal = Signal::start; + myWakeupCondition.notify_one(); +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +void RTEmulationWorker::stop() +{ + while (myPendingSignal != Signal::none) {} + + switch (myState) { + case State::exception: + break; + + case State::stopped: + break; + + case State::running: + case State::paused: + myPendingSignal = Signal::stop; + break; + + default: + throw runtime_error("unreachable"); + break; + } + + while (myState != State::stopped) rethrowPendingException(); +} + +void RTEmulationWorker::threadMain() +{ + std::unique_lock lock(myThreadIsRunningMutex); + + try { + while (myPendingSignal != Signal::quit) { + myState = State::stopped; + myPendingSignal = Signal::none; + myWakeupCondition.wait(lock); + + switch (myPendingSignal) { + case Signal::quit: + case Signal::none: + case Signal::stop: + continue; + + case Signal::resume: + case Signal::suspend: + throw runtime_error("invalid state"); + + case Signal::start: + break; + } + + dispatchEmulation(); + } + + myState = State::quit; + } + catch (...) { + myPendingException = std::current_exception(); + myState = State::exception; + } + + myPendingSignal = Signal::none; +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +void RTEmulationWorker::dispatchEmulation() +{ + struct timespec timeOffset; + struct timespec timesliceStart; + double virtualTimeNanoseconds = 0; + + clock_gettime(CLOCK_MONOTONIC, &timeOffset); + + myState = State::running; + myPendingSignal = Signal::none; + + while (myState == State::running) { + clock_gettime(CLOCK_MONOTONIC, ×liceStart); + + uInt64 realTimeNanoseconds = timeDifferenceNanoseconds(timesliceStart, timeOffset); + double deltaNanoseconds = realTimeNanoseconds - virtualTimeNanoseconds; + + if (deltaNanoseconds > MAX_LAG_NANOSECONDS) { + virtualTimeNanoseconds = realTimeNanoseconds; + deltaNanoseconds = 0.; + } + + const Int64 cycleGoal = (deltaNanoseconds * myCyclesPerSecond) / 1000000000; + uInt64 cyclesTotal = 0; + + while (cycleGoal > cyclesTotal && myDispatchResult->isSuccess()) { + myTia->update(*myDispatchResult, cycleGoal - cyclesTotal); + cyclesTotal += myDispatchResult->getCycles(); + } + + virtualTimeNanoseconds += cyclesTotal / myCyclesPerSecond * 1000000000; + + struct timespec now; + do { + clock_gettime(CLOCK_MONOTONIC, &now); + + switch (myPendingSignal) { + case Signal::quit: + case Signal::stop: + return; + + case Signal::none: + continue; + + case Signal::resume: + myState = State::running; + myPendingSignal = Signal::none; + break; + + case Signal::suspend: + myState = State::paused; + myPendingSignal = Signal::none; + break; + } + } while (timeDifferenceNanoseconds(now, timesliceStart) < TIMESLICE_NANOSECONDS || myState != State::running); + } +} + +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +void RTEmulationWorker::rethrowPendingException() +{ + if (myState == State::exception && myPendingException) { + const std::exception_ptr ex = myPendingException; + myPendingException = nullptr; + + std::rethrow_exception(ex); + } +} diff --git a/src/os/rtstella/RTEmulationWorker.hxx b/src/os/rtstella/RTEmulationWorker.hxx new file mode 100644 index 000000000..2c211861c --- /dev/null +++ b/src/os/rtstella/RTEmulationWorker.hxx @@ -0,0 +1,92 @@ +//============================================================================ +// +// SSSS tt lll lll +// SS SS tt ll ll +// SS tttttt eeee ll ll aaaa +// SSSS tt ee ee ll ll aa +// SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator" +// SS SS tt ee ll ll aa aa +// SSSS ttt eeeee llll llll aaaaa +// +// Copyright (c) 1995-2023 by Bradford W. Mott, Stephen Anthony +// and the Stella Team +// +// See the file "License.txt" for information on usage and redistribution of +// this file, and for a DISCLAIMER OF ALL WARRANTIES. +//============================================================================ + +#ifndef RT_EMULATION_WORKER_HXX +#define RT_EMULATION_WORKER_HXX + +#include +#include +#include +#include +#include + +#include "bspf.hxx" + +class TIA; +class DispatchResult; + +class RTEmulationWorker { + public: + + enum class Signal: uInt32 { + start, + suspend, + resume, + stop, + quit, + none + }; + + enum class State: uInt32 { + initializing, + stopped, + running, + paused, + quit, + exception + }; + + public: + + RTEmulationWorker(); + + ~RTEmulationWorker(); + + void start(uInt32 cyclesPerSecond, DispatchResult* dispatchResult, TIA* tia); + + void suspend(); + + void resume(); + + void stop(); + + private: + + void threadMain(); + + void rethrowPendingException(); + + void dispatchEmulation(); + + private: + + std::thread myThread; + + std::condition_variable myWakeupCondition; + std::mutex myThreadIsRunningMutex; + + std::atomic myPendingSignal{Signal::none}; + std::atomic myState{State::initializing}; + + std::exception_ptr myPendingException; + + TIA* myTia{nullptr}; + uInt64 myCyclesPerSecond{0}; + DispatchResult* myDispatchResult{nullptr}; +}; + +#endif // RT_EMULATION_WORKER_HXX diff --git a/src/os/rtstella/Spinlock.cxx b/src/os/rtstella/Spinlock.cxx index 0c06d1897..08bcfbbb7 100644 --- a/src/os/rtstella/Spinlock.cxx +++ b/src/os/rtstella/Spinlock.cxx @@ -18,21 +18,25 @@ #include "bspf.hxx" #include "Spinlock.hxx" +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Spinlock::Spinlock() { pthread_spin_init(&mySpinlock, PTHREAD_PROCESS_PRIVATE); } +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Spinlock::~Spinlock() { pthread_spin_destroy(&mySpinlock); } +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Spinlock::lock() { pthread_spin_lock(&mySpinlock); } +// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Spinlock::unlock() { pthread_spin_unlock(&mySpinlock); diff --git a/src/os/rtstella/module.mk b/src/os/rtstella/module.mk index 2048ec2cb..e3a48e334 100644 --- a/src/os/rtstella/module.mk +++ b/src/os/rtstella/module.mk @@ -1,7 +1,7 @@ MODULE := src/os/rtstella MODULE_OBJS := \ - src/os/rtstella/Spinlock.o + src/os/rtstella/Spinlock.o src/os/rtstella/RTEmulationWorker.o MODULE_DIRS += \ src/os/rtstella