| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2013-10-01 19:10:47 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-08 19:11:21 -04:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2013-10-01 19:10:47 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2018-08-05 22:07:28 -04:00
										 |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2021-11-28 11:28:29 +01:00
										 |  |  | #include <condition_variable>
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include <functional>
 | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2022-04-07 16:01:26 -07:00
										 |  |  | #include <mutex>
 | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  | #include <optional>
 | 
					
						
							| 
									
										
										
										
											2015-01-05 20:17:49 -05:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | #include "common/wall_clock.h"
 | 
					
						
							| 
									
										
										
										
											2015-01-05 20:17:49 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 12:32:15 -05:00
										 |  |  | namespace Core::Timing { | 
					
						
							| 
									
										
										
										
											2018-04-30 03:24:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | /// A callback that may be scheduled for a particular core timing event.
 | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  | using TimedCallback = std::function<std::optional<std::chrono::nanoseconds>( | 
					
						
							|  |  |  |     std::uintptr_t user_data, s64 time, std::chrono::nanoseconds ns_late)>; | 
					
						
							|  |  |  | using PauseCallback = std::function<void(bool paused)>; | 
					
						
							| 
									
										
										
										
											2018-08-05 21:27:11 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | /// Contains the characteristics of a particular event.
 | 
					
						
							|  |  |  | struct EventType { | 
					
						
							| 
									
										
										
										
											2020-11-25 15:21:03 -05:00
										 |  |  |     explicit EventType(TimedCallback&& callback_, std::string&& name_) | 
					
						
							|  |  |  |         : callback{std::move(callback_)}, name{std::move(name_)} {} | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  |     /// The event's callback function.
 | 
					
						
							|  |  |  |     TimedCallback callback; | 
					
						
							|  |  |  |     /// A pointer to the name of the event.
 | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  |     const std::string name; | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2015-01-05 20:17:49 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  |  * This is a system to schedule events into the emulated machine's future. Time is measured | 
					
						
							|  |  |  |  * in main CPU clock cycles. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * To schedule an event, you first have to register its type. This is where you pass in the | 
					
						
							| 
									
										
										
										
											2020-07-15 19:14:21 -04:00
										 |  |  |  * callback. You then schedule events using the type ID you get back. | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-07-15 19:14:21 -04:00
										 |  |  |  * The s64 ns_late that the callbacks get is how many ns late it was. | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  |  * So to schedule a new event on a regular basis: | 
					
						
							|  |  |  |  * inside callback: | 
					
						
							| 
									
										
										
										
											2020-07-15 19:14:21 -04:00
										 |  |  |  *   ScheduleEvent(period_in_ns - ns_late, callback, "whatever") | 
					
						
							| 
									
										
										
										
											2017-11-25 14:56:57 +01:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | class CoreTiming { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     CoreTiming(); | 
					
						
							|  |  |  |     ~CoreTiming(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CoreTiming(const CoreTiming&) = delete; | 
					
						
							|  |  |  |     CoreTiming(CoreTiming&&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CoreTiming& operator=(const CoreTiming&) = delete; | 
					
						
							|  |  |  |     CoreTiming& operator=(CoreTiming&&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// CoreTiming begins at the boundary of timing slice -1. An initial call to Advance() is
 | 
					
						
							|  |  |  |     /// required to end slice - 1 and start slice 0 before the first cycle of code is executed.
 | 
					
						
							| 
									
										
										
										
											2020-07-15 18:30:06 -04:00
										 |  |  |     void Initialize(std::function<void()>&& on_thread_init_); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /// Tears down all timing related functionality.
 | 
					
						
							|  |  |  |     void Shutdown(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 13:09:32 -04:00
										 |  |  |     /// Sets if emulation is multicore or single core, must be set before Initialize
 | 
					
						
							| 
									
										
										
										
											2020-11-25 15:21:03 -05:00
										 |  |  |     void SetMulticore(bool is_multicore_) { | 
					
						
							|  |  |  |         is_multicore = is_multicore_; | 
					
						
							| 
									
										
										
										
											2020-03-19 13:09:32 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-29 17:37:57 -04:00
										 |  |  |     /// Check if it's using host timing.
 | 
					
						
							|  |  |  |     bool IsHostTiming() const { | 
					
						
							|  |  |  |         return is_multicore; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Pauses/Unpauses the execution of the timer thread.
 | 
					
						
							|  |  |  |     void Pause(bool is_paused); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Pauses/Unpauses the execution of the timer thread and waits until paused.
 | 
					
						
							|  |  |  |     void SyncPause(bool is_paused); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Checks if core timing is running.
 | 
					
						
							|  |  |  |     bool IsRunning() const; | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Checks if the timer thread has started.
 | 
					
						
							|  |  |  |     bool HasStarted() const { | 
					
						
							|  |  |  |         return has_started; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Checks if there are any pending time events.
 | 
					
						
							|  |  |  |     bool HasPendingEvents() const; | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Schedules an event in core timing
 | 
					
						
							| 
									
										
										
										
											2020-07-15 18:30:06 -04:00
										 |  |  |     void ScheduleEvent(std::chrono::nanoseconds ns_into_future, | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |                        const std::shared_ptr<EventType>& event_type, std::uintptr_t user_data = 0, | 
					
						
							|  |  |  |                        bool absolute_time = false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Schedules an event which will automatically re-schedule itself with the given time, until
 | 
					
						
							|  |  |  |     /// unscheduled
 | 
					
						
							|  |  |  |     void ScheduleLoopingEvent(std::chrono::nanoseconds start_time, | 
					
						
							|  |  |  |                               std::chrono::nanoseconds resched_time, | 
					
						
							|  |  |  |                               const std::shared_ptr<EventType>& event_type, | 
					
						
							|  |  |  |                               std::uintptr_t user_data = 0, bool absolute_time = false); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-27 19:00:41 -04:00
										 |  |  |     void UnscheduleEvent(const std::shared_ptr<EventType>& event_type, std::uintptr_t user_data); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// We only permit one event of each type in the queue at a time.
 | 
					
						
							|  |  |  |     void RemoveEvent(const std::shared_ptr<EventType>& event_type); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 22:14:15 -04:00
										 |  |  |     void AddTicks(u64 ticks_to_add); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-28 15:23:28 -04:00
										 |  |  |     void ResetTicks(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Idle(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s64 GetDowncount() const { | 
					
						
							|  |  |  |         return downcount; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Returns current time in emulated CPU cycles
 | 
					
						
							|  |  |  |     u64 GetCPUTicks() const; | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Returns current time in emulated in Clock cycles
 | 
					
						
							|  |  |  |     u64 GetClockTicks() const; | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Returns current time in microseconds.
 | 
					
						
							|  |  |  |     std::chrono::microseconds GetGlobalTimeUs() const; | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Returns current time in nanoseconds.
 | 
					
						
							|  |  |  |     std::chrono::nanoseconds GetGlobalTimeNs() const; | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     /// Checks for events manually and returns time in nanoseconds for next event, threadsafe.
 | 
					
						
							| 
									
										
										
										
											2020-03-03 15:50:38 -04:00
										 |  |  |     std::optional<s64> Advance(); | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |     /// Register a callback function to be called when coretiming pauses.
 | 
					
						
							|  |  |  |     void RegisterPauseCallback(PauseCallback&& callback); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | private: | 
					
						
							|  |  |  |     struct Event; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Clear all pending events. This should ONLY be done on exit.
 | 
					
						
							|  |  |  |     void ClearPendingEvents(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 01:29:24 +02:00
										 |  |  |     static void ThreadEntry(CoreTiming& instance, size_t id); | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     void ThreadLoop(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::unique_ptr<Common::WallClock> clock; | 
					
						
							| 
									
										
										
										
											2019-09-09 21:37:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |     s64 global_timer = 0; | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // The queue is a min-heap using std::make_heap/push_heap/pop_heap.
 | 
					
						
							|  |  |  |     // We don't use std::priority_queue because we need to be able to serialize, unserialize and
 | 
					
						
							|  |  |  |     // erase arbitrary events (RemoveEvent()) regardless of the queue order. These aren't
 | 
					
						
							|  |  |  |     // accomodated by the standard adaptor class.
 | 
					
						
							|  |  |  |     std::vector<Event> event_queue; | 
					
						
							|  |  |  |     u64 event_fifo_id = 0; | 
					
						
							| 
									
										
										
										
											2022-06-29 01:29:24 +02:00
										 |  |  |     std::atomic<size_t> pending_events{}; | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  |     std::shared_ptr<EventType> ev_lost; | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  |     std::atomic<bool> has_started{}; | 
					
						
							| 
									
										
										
										
											2020-07-15 18:30:06 -04:00
										 |  |  |     std::function<void()> on_thread_init{}; | 
					
						
							| 
									
										
										
										
											2020-02-24 22:04:12 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-27 16:26:48 +01:00
										 |  |  |     std::vector<std::thread> worker_threads; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::condition_variable event_cv; | 
					
						
							|  |  |  |     std::condition_variable wait_pause_cv; | 
					
						
							|  |  |  |     std::condition_variable wait_signal_cv; | 
					
						
							|  |  |  |     mutable std::mutex event_mutex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::atomic<bool> paused_state{}; | 
					
						
							|  |  |  |     bool is_paused{}; | 
					
						
							|  |  |  |     bool shutting_down{}; | 
					
						
							| 
									
										
										
										
											2020-03-19 13:09:32 -04:00
										 |  |  |     bool is_multicore{}; | 
					
						
							| 
									
										
										
										
											2021-11-27 16:26:48 +01:00
										 |  |  |     size_t pause_count{}; | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  |     s64 pause_end_time{}; | 
					
						
							| 
									
										
										
										
											2020-03-19 13:09:32 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-28 15:23:28 -04:00
										 |  |  |     /// Cycle timing
 | 
					
						
							|  |  |  |     u64 ticks{}; | 
					
						
							|  |  |  |     s64 downcount{}; | 
					
						
							| 
									
										
										
										
											2022-07-10 06:59:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     std::vector<PauseCallback> pause_callbacks{}; | 
					
						
							| 
									
										
										
										
											2019-02-14 12:42:58 -05:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2013-10-01 19:10:47 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-26 21:48:56 -05:00
										 |  |  | /// Creates a core timing event with the given name and callback.
 | 
					
						
							|  |  |  | ///
 | 
					
						
							|  |  |  | /// @param name     The name of the core timing event to create.
 | 
					
						
							|  |  |  | /// @param callback The callback to execute for the event.
 | 
					
						
							|  |  |  | ///
 | 
					
						
							|  |  |  | /// @returns An EventType instance representing the created event.
 | 
					
						
							|  |  |  | ///
 | 
					
						
							|  |  |  | std::shared_ptr<EventType> CreateEvent(std::string name, TimedCallback&& callback); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-12 12:32:15 -05:00
										 |  |  | } // namespace Core::Timing
 |