| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "common/uint128.h"
 | 
					
						
							|  |  |  | #include "common/wall_clock.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef ARCHITECTURE_x86_64
 | 
					
						
							|  |  |  | #include "common/x64/cpu_detect.h"
 | 
					
						
							|  |  |  | #include "common/x64/native_clock.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using base_timer = std::chrono::steady_clock; | 
					
						
							|  |  |  | using base_time_point = std::chrono::time_point<base_timer>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 16:19:37 -03:00
										 |  |  | class StandardWallClock final : public WallClock { | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2020-11-25 15:21:03 -05:00
										 |  |  |     explicit StandardWallClock(u64 emulated_cpu_frequency_, u64 emulated_clock_frequency_) | 
					
						
							| 
									
										
										
										
											2021-02-19 18:04:23 -08:00
										 |  |  |         : WallClock(emulated_cpu_frequency_, emulated_clock_frequency_, false) { | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |         start_time = base_timer::now(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::chrono::nanoseconds GetTimeNS() override { | 
					
						
							|  |  |  |         base_time_point current = base_timer::now(); | 
					
						
							|  |  |  |         auto elapsed = current - start_time; | 
					
						
							|  |  |  |         return std::chrono::duration_cast<std::chrono::nanoseconds>(elapsed); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::chrono::microseconds GetTimeUS() override { | 
					
						
							|  |  |  |         base_time_point current = base_timer::now(); | 
					
						
							|  |  |  |         auto elapsed = current - start_time; | 
					
						
							|  |  |  |         return std::chrono::duration_cast<std::chrono::microseconds>(elapsed); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::chrono::milliseconds GetTimeMS() override { | 
					
						
							|  |  |  |         base_time_point current = base_timer::now(); | 
					
						
							|  |  |  |         auto elapsed = current - start_time; | 
					
						
							|  |  |  |         return std::chrono::duration_cast<std::chrono::milliseconds>(elapsed); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u64 GetClockCycles() override { | 
					
						
							| 
									
										
										
										
											2021-02-19 18:04:23 -08:00
										 |  |  |         std::chrono::nanoseconds time_now = GetTimeNS(); | 
					
						
							|  |  |  |         const u128 temporary = | 
					
						
							|  |  |  |             Common::Multiply64Into128(time_now.count(), emulated_clock_frequency); | 
					
						
							|  |  |  |         return Common::Divide128On32(temporary, 1000000000).first; | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u64 GetCPUCycles() override { | 
					
						
							| 
									
										
										
										
											2021-02-19 18:04:23 -08:00
										 |  |  |         std::chrono::nanoseconds time_now = GetTimeNS(); | 
					
						
							|  |  |  |         const u128 temporary = Common::Multiply64Into128(time_now.count(), emulated_cpu_frequency); | 
					
						
							|  |  |  |         return Common::Divide128On32(temporary, 1000000000).first; | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-30 15:02:02 -04:00
										 |  |  |     void Pause([[maybe_unused]] bool is_paused) override { | 
					
						
							| 
									
										
										
										
											2020-02-25 12:28:55 -04:00
										 |  |  |         // Do nothing in this clock type.
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | private: | 
					
						
							|  |  |  |     base_time_point start_time; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef ARCHITECTURE_x86_64
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-30 12:57:23 -05:00
										 |  |  | std::unique_ptr<WallClock> CreateBestMatchingClock(u64 emulated_cpu_frequency, | 
					
						
							|  |  |  |                                                    u64 emulated_clock_frequency) { | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     const auto& caps = GetCPUCaps(); | 
					
						
							|  |  |  |     u64 rtsc_frequency = 0; | 
					
						
							|  |  |  |     if (caps.invariant_tsc) { | 
					
						
							| 
									
										
										
										
											2022-07-06 12:42:01 -05:00
										 |  |  |         rtsc_frequency = caps.tsc_frequency ? caps.tsc_frequency : EstimateRDTSCFrequency(); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-01-27 16:53:58 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-30 12:57:23 -05:00
										 |  |  |     // Fallback to StandardWallClock if the hardware TSC does not have the precision greater than:
 | 
					
						
							|  |  |  |     // - A nanosecond
 | 
					
						
							|  |  |  |     // - The emulated CPU frequency
 | 
					
						
							|  |  |  |     // - The emulated clock counter frequency (CNTFRQ)
 | 
					
						
							|  |  |  |     if (rtsc_frequency <= WallClock::NS_RATIO || rtsc_frequency <= emulated_cpu_frequency || | 
					
						
							|  |  |  |         rtsc_frequency <= emulated_clock_frequency) { | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  |         return std::make_unique<StandardWallClock>(emulated_cpu_frequency, | 
					
						
							|  |  |  |                                                    emulated_clock_frequency); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-02-10 13:33:13 -04:00
										 |  |  |         return std::make_unique<X64::NativeClock>(emulated_cpu_frequency, emulated_clock_frequency, | 
					
						
							|  |  |  |                                                   rtsc_frequency); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-30 12:57:23 -05:00
										 |  |  | std::unique_ptr<WallClock> CreateBestMatchingClock(u64 emulated_cpu_frequency, | 
					
						
							|  |  |  |                                                    u64 emulated_clock_frequency) { | 
					
						
							| 
									
										
										
										
											2020-02-10 11:20:40 -04:00
										 |  |  |     return std::make_unique<StandardWallClock>(emulated_cpu_frequency, emulated_clock_frequency); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Common
 |