| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | // Copyright 2020 yuzu Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  | #include <array>
 | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  | #include "common/atomic_ops.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-10 11:20:40 -04:00
										 |  |  | #include "common/uint128.h"
 | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | #include "common/x64/native_clock.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 EstimateRDTSCFrequency() { | 
					
						
							| 
									
										
										
										
											2021-12-02 16:12:23 -05:00
										 |  |  |     // Discard the first result measuring the rdtsc.
 | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     _mm_mfence(); | 
					
						
							| 
									
										
										
										
											2021-12-02 16:12:23 -05:00
										 |  |  |     __rdtsc(); | 
					
						
							|  |  |  |     std::this_thread::sleep_for(std::chrono::milliseconds{1}); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     _mm_mfence(); | 
					
						
							| 
									
										
										
										
											2021-12-02 16:12:23 -05:00
										 |  |  |     __rdtsc(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Get the current time.
 | 
					
						
							|  |  |  |     const auto start_time = std::chrono::steady_clock::now(); | 
					
						
							|  |  |  |     _mm_mfence(); | 
					
						
							|  |  |  |     const u64 tsc_start = __rdtsc(); | 
					
						
							|  |  |  |     // Wait for 200 milliseconds.
 | 
					
						
							|  |  |  |     std::this_thread::sleep_for(std::chrono::milliseconds{200}); | 
					
						
							|  |  |  |     const auto end_time = std::chrono::steady_clock::now(); | 
					
						
							|  |  |  |     _mm_mfence(); | 
					
						
							|  |  |  |     const u64 tsc_end = __rdtsc(); | 
					
						
							|  |  |  |     // Calculate differences.
 | 
					
						
							|  |  |  |     const u64 timer_diff = static_cast<u64>( | 
					
						
							|  |  |  |         std::chrono::duration_cast<std::chrono::nanoseconds>(end_time - start_time).count()); | 
					
						
							|  |  |  |     const u64 tsc_diff = tsc_end - tsc_start; | 
					
						
							| 
									
										
										
										
											2020-02-10 11:20:40 -04:00
										 |  |  |     const u64 tsc_freq = MultiplyAndDivide64(tsc_diff, 1000000000ULL, timer_diff); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     return tsc_freq; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace X64 { | 
					
						
							| 
									
										
										
										
											2020-11-25 15:21:03 -05:00
										 |  |  | NativeClock::NativeClock(u64 emulated_cpu_frequency_, u64 emulated_clock_frequency_, | 
					
						
							|  |  |  |                          u64 rtsc_frequency_) | 
					
						
							|  |  |  |     : WallClock(emulated_cpu_frequency_, emulated_clock_frequency_, true), rtsc_frequency{ | 
					
						
							|  |  |  |                                                                                rtsc_frequency_} { | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  |     _mm_mfence(); | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  |     time_point.inner.last_measure = __rdtsc(); | 
					
						
							|  |  |  |     time_point.inner.accumulated_ticks = 0U; | 
					
						
							| 
									
										
										
										
											2022-01-30 12:36:56 -05:00
										 |  |  |     ns_rtsc_factor = GetFixedPoint64Factor(NS_RATIO, rtsc_frequency); | 
					
						
							|  |  |  |     us_rtsc_factor = GetFixedPoint64Factor(US_RATIO, rtsc_frequency); | 
					
						
							|  |  |  |     ms_rtsc_factor = GetFixedPoint64Factor(MS_RATIO, rtsc_frequency); | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  |     clock_rtsc_factor = GetFixedPoint64Factor(emulated_clock_frequency, rtsc_frequency); | 
					
						
							|  |  |  |     cpu_rtsc_factor = GetFixedPoint64Factor(emulated_cpu_frequency, rtsc_frequency); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 NativeClock::GetRTSC() { | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  |     TimePoint new_time_point{}; | 
					
						
							|  |  |  |     TimePoint current_time_point{}; | 
					
						
							|  |  |  |     do { | 
					
						
							| 
									
										
										
										
											2022-03-28 23:06:04 +01:00
										 |  |  |         current_time_point.pack = Common::AtomicLoad128(time_point.pack.data()); | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  |         _mm_mfence(); | 
					
						
							|  |  |  |         const u64 current_measure = __rdtsc(); | 
					
						
							|  |  |  |         u64 diff = current_measure - current_time_point.inner.last_measure; | 
					
						
							|  |  |  |         diff = diff & ~static_cast<u64>(static_cast<s64>(diff) >> 63); // max(diff, 0)
 | 
					
						
							|  |  |  |         new_time_point.inner.last_measure = current_measure > current_time_point.inner.last_measure | 
					
						
							|  |  |  |                                                 ? current_measure | 
					
						
							|  |  |  |                                                 : current_time_point.inner.last_measure; | 
					
						
							|  |  |  |         new_time_point.inner.accumulated_ticks = current_time_point.inner.accumulated_ticks + diff; | 
					
						
							|  |  |  |     } while (!Common::AtomicCompareAndSwap(time_point.pack.data(), new_time_point.pack, | 
					
						
							|  |  |  |                                            current_time_point.pack)); | 
					
						
							| 
									
										
										
										
											2020-03-21 12:23:13 -04:00
										 |  |  |     /// The clock cannot be more precise than the guest timer, remove the lower bits
 | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  |     return new_time_point.inner.accumulated_ticks & inaccuracy_mask; | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-25 12:28:55 -04:00
										 |  |  | void NativeClock::Pause(bool is_paused) { | 
					
						
							|  |  |  |     if (!is_paused) { | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  |         TimePoint current_time_point{}; | 
					
						
							|  |  |  |         TimePoint new_time_point{}; | 
					
						
							|  |  |  |         do { | 
					
						
							| 
									
										
										
										
											2022-03-28 23:06:04 +01:00
										 |  |  |             current_time_point.pack = Common::AtomicLoad128(time_point.pack.data()); | 
					
						
							| 
									
										
										
										
											2021-01-02 02:24:49 +01:00
										 |  |  |             new_time_point.pack = current_time_point.pack; | 
					
						
							|  |  |  |             _mm_mfence(); | 
					
						
							|  |  |  |             new_time_point.inner.last_measure = __rdtsc(); | 
					
						
							|  |  |  |         } while (!Common::AtomicCompareAndSwap(time_point.pack.data(), new_time_point.pack, | 
					
						
							|  |  |  |                                                current_time_point.pack)); | 
					
						
							| 
									
										
										
										
											2020-02-25 12:28:55 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | std::chrono::nanoseconds NativeClock::GetTimeNS() { | 
					
						
							|  |  |  |     const u64 rtsc_value = GetRTSC(); | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  |     return std::chrono::nanoseconds{MultiplyHigh(rtsc_value, ns_rtsc_factor)}; | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::chrono::microseconds NativeClock::GetTimeUS() { | 
					
						
							|  |  |  |     const u64 rtsc_value = GetRTSC(); | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  |     return std::chrono::microseconds{MultiplyHigh(rtsc_value, us_rtsc_factor)}; | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::chrono::milliseconds NativeClock::GetTimeMS() { | 
					
						
							|  |  |  |     const u64 rtsc_value = GetRTSC(); | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  |     return std::chrono::milliseconds{MultiplyHigh(rtsc_value, ms_rtsc_factor)}; | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 NativeClock::GetClockCycles() { | 
					
						
							|  |  |  |     const u64 rtsc_value = GetRTSC(); | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  |     return MultiplyHigh(rtsc_value, clock_rtsc_factor); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 NativeClock::GetCPUCycles() { | 
					
						
							|  |  |  |     const u64 rtsc_value = GetRTSC(); | 
					
						
							| 
									
										
										
										
											2021-01-01 23:28:55 +01:00
										 |  |  |     return MultiplyHigh(rtsc_value, cpu_rtsc_factor); | 
					
						
							| 
									
										
										
										
											2020-02-09 16:53:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace X64
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Common
 |