mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 18:33:57 -07:00 
			
		
		
		
	* Add a "Pause Emulation" option and hotkey Closes Ryujinx#1604 * Refactoring how pause is handled * Applied suggested changes from review * Applied suggested fixes * Pass correct suspend type to threads for suspend/resume * Fix NRE after stoping emulation * Removing SimulateWakeUpMessage call after resuming emulation * Skip suspending non game process * Pause the tickCounter in the ExecutionContext * Refactoring tickCounter pause/resume as suggested * Fix Config migration to add pause hotkey * Fixed pausing only application threads * Fix exiting emulator while paused * Avoid pause/resume while already paused/resumed * Cleanup unused code * Avoid restarting audio if stopping emulation while in pause. * Added suggested changes * Fix ConfigurationState
		
			
				
	
	
		
			163 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			163 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using ARMeilleure.Memory;
 | |
| using System;
 | |
| using System.Diagnostics;
 | |
| 
 | |
| namespace ARMeilleure.State
 | |
| {
 | |
|     public class ExecutionContext
 | |
|     {
 | |
|         private const int MinCountForCheck = 4000;
 | |
| 
 | |
|         private NativeContext _nativeContext;
 | |
| 
 | |
|         internal IntPtr NativeContextPtr => _nativeContext.BasePtr;
 | |
| 
 | |
|         private bool _interrupted;
 | |
| 
 | |
|         private static Stopwatch _tickCounter;
 | |
| 
 | |
|         private static double _hostTickFreq;
 | |
| 
 | |
|         public uint CtrEl0   => 0x8444c004;
 | |
|         public uint DczidEl0 => 0x00000004;
 | |
| 
 | |
|         public ulong CntfrqEl0 { get; set; }
 | |
|         public ulong CntpctEl0
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 double ticks = _tickCounter.ElapsedTicks * _hostTickFreq;
 | |
| 
 | |
|                 return (ulong)(ticks * CntfrqEl0);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // CNTVCT_EL0 = CNTPCT_EL0 - CNTVOFF_EL2
 | |
|         // Since EL2 isn't implemented, CNTVOFF_EL2 = 0
 | |
|         public ulong CntvctEl0 => CntpctEl0;
 | |
| 
 | |
|         public static TimeSpan ElapsedTime => _tickCounter.Elapsed;
 | |
|         public static long ElapsedTicks => _tickCounter.ElapsedTicks;
 | |
|         public static double TickFrequency => _hostTickFreq;
 | |
| 
 | |
|         public long TpidrEl0 { get; set; }
 | |
|         public long Tpidr    { get; set; }
 | |
| 
 | |
|         public FPCR Fpcr { get; set; }
 | |
|         public FPSR Fpsr { get; set; }
 | |
|         public FPCR StandardFpcrValue => (Fpcr & (FPCR.Ahp)) | FPCR.Dn | FPCR.Fz;
 | |
| 
 | |
|         public bool IsAarch32 { get; set; }
 | |
| 
 | |
|         internal ExecutionMode ExecutionMode
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (IsAarch32)
 | |
|                 {
 | |
|                     return GetPstateFlag(PState.TFlag)
 | |
|                         ? ExecutionMode.Aarch32Thumb
 | |
|                         : ExecutionMode.Aarch32Arm;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     return ExecutionMode.Aarch64;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool Running
 | |
|         {
 | |
|             get => _nativeContext.GetRunning();
 | |
|             private set => _nativeContext.SetRunning(value);
 | |
|         }
 | |
| 
 | |
|         public event EventHandler<EventArgs>              Interrupt;
 | |
|         public event EventHandler<InstExceptionEventArgs> Break;
 | |
|         public event EventHandler<InstExceptionEventArgs> SupervisorCall;
 | |
|         public event EventHandler<InstUndefinedEventArgs> Undefined;
 | |
| 
 | |
|         static ExecutionContext()
 | |
|         {
 | |
|             _hostTickFreq = 1.0 / Stopwatch.Frequency;
 | |
| 
 | |
|             _tickCounter = new Stopwatch();
 | |
|             _tickCounter.Start();
 | |
|         }
 | |
| 
 | |
|         public ExecutionContext(IJitMemoryAllocator allocator)
 | |
|         {
 | |
|             _nativeContext = new NativeContext(allocator);
 | |
| 
 | |
|             Running = true;
 | |
| 
 | |
|             _nativeContext.SetCounter(MinCountForCheck);
 | |
|         }
 | |
| 
 | |
|         public ulong GetX(int index)              => _nativeContext.GetX(index);
 | |
|         public void  SetX(int index, ulong value) => _nativeContext.SetX(index, value);
 | |
| 
 | |
|         public V128 GetV(int index)             => _nativeContext.GetV(index);
 | |
|         public void SetV(int index, V128 value) => _nativeContext.SetV(index, value);
 | |
| 
 | |
|         public bool GetPstateFlag(PState flag)             => _nativeContext.GetPstateFlag(flag);
 | |
|         public void SetPstateFlag(PState flag, bool value) => _nativeContext.SetPstateFlag(flag, value);
 | |
| 
 | |
|         public bool GetFPstateFlag(FPState flag) => _nativeContext.GetFPStateFlag(flag);
 | |
|         public void SetFPstateFlag(FPState flag, bool value) => _nativeContext.SetFPStateFlag(flag, value);
 | |
| 
 | |
|         internal void CheckInterrupt()
 | |
|         {
 | |
|             if (_interrupted)
 | |
|             {
 | |
|                 _interrupted = false;
 | |
| 
 | |
|                 Interrupt?.Invoke(this, EventArgs.Empty);
 | |
|             }
 | |
| 
 | |
|             _nativeContext.SetCounter(MinCountForCheck);
 | |
|         }
 | |
| 
 | |
|         public void RequestInterrupt()
 | |
|         {
 | |
|             _interrupted = true;
 | |
|         }
 | |
| 
 | |
|         internal void OnBreak(ulong address, int imm)
 | |
|         {
 | |
|             Break?.Invoke(this, new InstExceptionEventArgs(address, imm));
 | |
|         }
 | |
| 
 | |
|         internal void OnSupervisorCall(ulong address, int imm)
 | |
|         {
 | |
|             SupervisorCall?.Invoke(this, new InstExceptionEventArgs(address, imm));
 | |
|         }
 | |
| 
 | |
|         internal void OnUndefined(ulong address, int opCode)
 | |
|         {
 | |
|             Undefined?.Invoke(this, new InstUndefinedEventArgs(address, opCode));
 | |
|         }
 | |
| 
 | |
|         public void StopRunning()
 | |
|         {
 | |
|             Running = false;
 | |
| 
 | |
|             _nativeContext.SetCounter(0);
 | |
|         }
 | |
| 
 | |
|         public static void SuspendCounter()
 | |
|         {
 | |
|             _tickCounter.Stop();
 | |
|         }
 | |
| 
 | |
|         public static void ResumeCounter()
 | |
|         {
 | |
|             _tickCounter.Start();
 | |
|         }
 | |
| 
 | |
|         public void Dispose()
 | |
|         {
 | |
|             _nativeContext.Dispose();
 | |
|         }
 | |
|     }
 | |
| } |