mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 07:22:25 -07:00 
			
		
		
		
	* Add background translation to the CPU * Do not use a separate thread for translation, implement 2 tiers translation * Remove unnecessary usings * Lower MinCallCountForReJit * Remove unused variable
		
			
				
	
	
		
			134 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using ChocolArm64.Memory;
 | |
| using ChocolArm64.State;
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Collections.ObjectModel;
 | |
| using System.Reflection;
 | |
| using System.Reflection.Emit;
 | |
| 
 | |
| namespace ChocolArm64
 | |
| {
 | |
|     class ATranslatedSub
 | |
|     {
 | |
|         private delegate long AA64Subroutine(AThreadState Register, AMemory Memory);
 | |
| 
 | |
|         private AA64Subroutine ExecDelegate;
 | |
| 
 | |
|         public static int StateArgIdx  { get; private set; }
 | |
|         public static int MemoryArgIdx { get; private set; }
 | |
| 
 | |
|         public static Type[] FixedArgTypes { get; private set; }
 | |
| 
 | |
|         public DynamicMethod Method { get; private set; }
 | |
| 
 | |
|         public ReadOnlyCollection<ARegister> Params { get; private set; }
 | |
| 
 | |
|         private HashSet<long> Callees;
 | |
| 
 | |
|         private ATranslatedSubType Type;
 | |
| 
 | |
|         private int CallCount;
 | |
| 
 | |
|         private bool NeedsReJit;
 | |
| 
 | |
|         private int MinCallCountForReJit = 250;
 | |
| 
 | |
|         public ATranslatedSub(DynamicMethod Method, List<ARegister> Params, HashSet<long> Callees)
 | |
|         {
 | |
|             if (Method == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(Method));
 | |
|             }
 | |
| 
 | |
|             if (Params == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(Params));
 | |
|             }
 | |
| 
 | |
|             if (Callees == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(Callees));
 | |
|             }
 | |
| 
 | |
|             this.Method  = Method;
 | |
|             this.Params  = Params.AsReadOnly();
 | |
|             this.Callees = Callees;
 | |
| 
 | |
|             PrepareDelegate();
 | |
|         }
 | |
| 
 | |
|         static ATranslatedSub()
 | |
|         {
 | |
|             MethodInfo MthdInfo = typeof(AA64Subroutine).GetMethod("Invoke");
 | |
| 
 | |
|             ParameterInfo[] Params = MthdInfo.GetParameters();
 | |
| 
 | |
|             FixedArgTypes = new Type[Params.Length];
 | |
| 
 | |
|             for (int Index = 0; Index < Params.Length; Index++)
 | |
|             {
 | |
|                 Type ParamType = Params[Index].ParameterType;
 | |
| 
 | |
|                 FixedArgTypes[Index] = ParamType;
 | |
| 
 | |
|                 if (ParamType == typeof(AThreadState))
 | |
|                 {
 | |
|                     StateArgIdx = Index;
 | |
|                 }
 | |
|                 else if (ParamType == typeof(AMemory))
 | |
|                 {
 | |
|                     MemoryArgIdx = Index;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void PrepareDelegate()
 | |
|         {
 | |
|             string Name = $"{Method.Name}_Dispatch";
 | |
| 
 | |
|             DynamicMethod Mthd = new DynamicMethod(Name, typeof(long), FixedArgTypes);
 | |
| 
 | |
|             ILGenerator Generator = Mthd.GetILGenerator();
 | |
| 
 | |
|             Generator.EmitLdargSeq(FixedArgTypes.Length);
 | |
| 
 | |
|             foreach (ARegister Reg in Params)
 | |
|             {
 | |
|                 Generator.EmitLdarg(StateArgIdx);
 | |
| 
 | |
|                 Generator.Emit(OpCodes.Ldfld, Reg.GetField());
 | |
|             }
 | |
| 
 | |
|             Generator.Emit(OpCodes.Call, Method);
 | |
|             Generator.Emit(OpCodes.Ret);
 | |
| 
 | |
|             ExecDelegate = (AA64Subroutine)Mthd.CreateDelegate(typeof(AA64Subroutine));
 | |
|         }
 | |
| 
 | |
|         public bool ShouldReJit()
 | |
|         {
 | |
|             if (Type == ATranslatedSubType.SubTier0)
 | |
|             {
 | |
|                 if (CallCount < MinCallCountForReJit)
 | |
|                 {
 | |
|                     CallCount++;
 | |
|                 }
 | |
| 
 | |
|                 return CallCount == MinCallCountForReJit;
 | |
|             }
 | |
| 
 | |
|             return Type == ATranslatedSubType.SubTier1 && NeedsReJit;
 | |
|         }
 | |
| 
 | |
|         public long Execute(AThreadState ThreadState, AMemory Memory)
 | |
|         {
 | |
|             return ExecDelegate(ThreadState, Memory);
 | |
|         }
 | |
| 
 | |
|         public void SetType(ATranslatedSubType Type) => this.Type = Type;
 | |
| 
 | |
|         public bool HasCallee(long Position) => Callees.Contains(Position);
 | |
| 
 | |
|         public void MarkForReJit() => NeedsReJit = true;        
 | |
|     }
 | |
| } |