mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 16:03:23 -07:00 
			
		
		
		
	* Add AddressTable<T>
* Use AddressTable<T> for dispatch
* Remove JumpTable & co.
* Add fallback for out of range addresses
* Add PPTC support
* Add documentation to `AddressTable<T>`
* Make AddressTable<T> configurable
* Fix table walk
* Fix IsMapped check
* Remove CountTableCapacity
* Add PPTC support for fast path
* Rename IsMapped to IsValid
* Remove stale comment
* Change format of address in exception message
* Add TranslatorStubs
* Split DispatchStub
Avoids recompilation of stubs during tests.
* Add hint for 64bit or 32bit
* Add documentation to `Symbol`
* Add documentation to `TranslatorStubs`
Make `TranslatorStubs` disposable as well.
* Add documentation to `SymbolType`
* Add `AddressTableEventSource` to monitor function table size
Add an EventSource which measures the amount of unmanaged bytes
allocated by AddressTable<T> instances.
 dotnet-counters monitor -n Ryujinx --counters ARMeilleure
* Add `AllowLcqInFunctionTable` optimization toggle
This is to reduce the impact this change has on the test duration.
Before everytime a test was ran, the FunctionTable would be initialized
and populated so that the newly compiled test would get registered to
it.
* Implement unmanaged dispatcher
Uses the DispatchStub to dispatch into the next translation, which
allows execution to stay in unmanaged for longer and skips a
ConcurrentDictionary look up when the target translation has been
registered to the FunctionTable.
* Remove redundant null check
* Tune levels of FunctionTable
Uses 5 levels instead of 4 and change unit of AddressTableEventSource
from KB to MB.
* Use 64-bit function table
Improves codegen for direct branches:
    mov qword [rax+0x408],0x10603560
 -  mov rcx,sub_10603560_OFFSET
 -  mov ecx,[rcx]
 -  mov ecx,ecx
 -  mov rdx,JIT_CACHE_BASE
 -  add rdx,rcx
 +  mov rcx,sub_10603560
 +  mov rdx,[rcx]
    mov rcx,rax
Improves codegen for dispatch stub:
    and rax,byte +0x1f
 -  mov eax,[rcx+rax*4]
 -  mov eax,eax
 -  mov rcx,JIT_CACHE_BASE
 -  lea rax,[rcx+rax]
 +  mov rax,[rcx+rax*8]
    mov rcx,rbx
* Remove `JitCacheSymbol` & `JitCache.Offset`
* Turn `Translator.Translate` into an instance method
We do not have to add more parameter to this method and related ones as
new structures are added & needed for translation.
* Add symbol only when PTC is enabled
Address LDj3SNuD's feedback
* Change `NativeContext.Running` to a 32-bit integer
* Fix PageTable symbol for host mapped
		
	
		
			
				
	
	
		
			249 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			249 lines
		
	
	
		
			9.4 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using ARMeilleure.Instructions;
 | |
| using ARMeilleure.IntermediateRepresentation;
 | |
| using ARMeilleure.State;
 | |
| using ARMeilleure.Translation.Cache;
 | |
| using System;
 | |
| using System.Reflection;
 | |
| using System.Runtime.InteropServices;
 | |
| using static ARMeilleure.IntermediateRepresentation.OperandHelper;
 | |
| 
 | |
| namespace ARMeilleure.Translation
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Represents a stub manager.
 | |
|     /// </summary>
 | |
|     class TranslatorStubs : IDisposable
 | |
|     {
 | |
|         private static readonly Lazy<IntPtr> _slowDispatchStub = new(GenerateSlowDispatchStub, isThreadSafe: true);
 | |
| 
 | |
|         private bool _disposed;
 | |
| 
 | |
|         private readonly Translator _translator;
 | |
|         private readonly Lazy<IntPtr> _dispatchStub;
 | |
|         private readonly Lazy<DispatcherFunction> _dispatchLoop;
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets the dispatch stub.
 | |
|         /// </summary>
 | |
|         /// <exception cref="ObjectDisposedException"><see cref="TranslatorStubs"/> instance was disposed</exception>
 | |
|         public IntPtr DispatchStub
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (_disposed)
 | |
|                 {
 | |
|                     throw new ObjectDisposedException(null);
 | |
|                 }
 | |
| 
 | |
|                 return _dispatchStub.Value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets the slow dispatch stub.
 | |
|         /// </summary>
 | |
|         /// <exception cref="ObjectDisposedException"><see cref="TranslatorStubs"/> instance was disposed</exception>
 | |
|         public IntPtr SlowDispatchStub
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (_disposed)
 | |
|                 {
 | |
|                     throw new ObjectDisposedException(null);
 | |
|                 }
 | |
| 
 | |
|                 return _slowDispatchStub.Value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Gets the dispatch loop function.
 | |
|         /// </summary>
 | |
|         /// <exception cref="ObjectDisposedException"><see cref="TranslatorStubs"/> instance was disposed</exception>
 | |
|         public DispatcherFunction DispatchLoop
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (_disposed)
 | |
|                 {
 | |
|                     throw new ObjectDisposedException(null);
 | |
|                 }
 | |
| 
 | |
|                 return _dispatchLoop.Value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Initializes a new instance of the <see cref="TranslatorStubs"/> class with the specified
 | |
|         /// <see cref="Translator"/> instance.
 | |
|         /// </summary>
 | |
|         /// <param name="translator"><see cref="Translator"/> instance to use</param>
 | |
|         /// <exception cref="ArgumentNullException"><paramref name="translator"/> is null</exception>
 | |
|         public TranslatorStubs(Translator translator)
 | |
|         {
 | |
|             _translator = translator ?? throw new ArgumentNullException(nameof(translator));
 | |
|             _dispatchStub = new(GenerateDispatchStub, isThreadSafe: true);
 | |
|             _dispatchLoop = new(GenerateDispatchLoop, isThreadSafe: true);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Releases all resources used by the <see cref="TranslatorStubs"/> instance.
 | |
|         /// </summary>
 | |
|         public void Dispose()
 | |
|         {
 | |
|             Dispose(true);
 | |
|             GC.SuppressFinalize(this);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Releases all unmanaged and optionally managed resources used by the <see cref="TranslatorStubs"/> instance.
 | |
|         /// </summary>
 | |
|         /// <param name="disposing"><see langword="true"/> to dispose managed resources also; otherwise just unmanaged resouces</param>
 | |
|         protected virtual void Dispose(bool disposing)
 | |
|         {
 | |
|             if (!_disposed)
 | |
|             {
 | |
|                 if (_dispatchStub.IsValueCreated)
 | |
|                 {
 | |
|                     JitCache.Unmap(_dispatchStub.Value);
 | |
|                 }
 | |
| 
 | |
|                 if (_dispatchLoop.IsValueCreated)
 | |
|                 {
 | |
|                     JitCache.Unmap(Marshal.GetFunctionPointerForDelegate(_dispatchLoop.Value));
 | |
|                 }
 | |
| 
 | |
|                 _disposed = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Frees resources used by the <see cref="TranslatorStubs"/> instance.
 | |
|         /// </summary>
 | |
|         ~TranslatorStubs()
 | |
|         {
 | |
|             Dispose(false);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Generates a <see cref="DispatchStub"/>.
 | |
|         /// </summary>
 | |
|         /// <returns>Generated <see cref="DispatchStub"/></returns>
 | |
|         private IntPtr GenerateDispatchStub()
 | |
|         {
 | |
|             var context = new EmitterContext();
 | |
| 
 | |
|             Operand lblFallback = Label();
 | |
|             Operand lblEnd = Label();
 | |
| 
 | |
|             // Load the target guest address from the native context.
 | |
|             Operand nativeContext = context.LoadArgument(OperandType.I64, 0);
 | |
|             Operand guestAddress = context.Load(OperandType.I64,
 | |
|                 context.Add(nativeContext, Const((ulong)NativeContext.GetDispatchAddressOffset())));
 | |
| 
 | |
|             // Check if guest address is within range of the AddressTable.
 | |
|             Operand masked = context.BitwiseAnd(guestAddress, Const(~_translator.FunctionTable.Mask));
 | |
|             context.BranchIfTrue(lblFallback, masked);
 | |
| 
 | |
|             Operand index = null;
 | |
|             Operand page = Const((long)_translator.FunctionTable.Base);
 | |
| 
 | |
|             for (int i = 0; i < _translator.FunctionTable.Levels.Length; i++)
 | |
|             {
 | |
|                 ref var level = ref _translator.FunctionTable.Levels[i];
 | |
| 
 | |
|                 // level.Mask is not used directly because it is more often bigger than 32-bits, so it will not
 | |
|                 // be encoded as an immediate on x86's bitwise and operation.
 | |
|                 Operand mask = Const(level.Mask >> level.Index);
 | |
| 
 | |
|                 index = context.BitwiseAnd(context.ShiftRightUI(guestAddress, Const(level.Index)), mask);
 | |
| 
 | |
|                 if (i < _translator.FunctionTable.Levels.Length - 1)
 | |
|                 {
 | |
|                     page = context.Load(OperandType.I64, context.Add(page, context.ShiftLeft(index, Const(3))));
 | |
|                     context.BranchIfFalse(lblFallback, page);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             Operand hostAddress;
 | |
|             Operand hostAddressAddr = context.Add(page, context.ShiftLeft(index, Const(3)));
 | |
|             hostAddress = context.Load(OperandType.I64, hostAddressAddr);
 | |
|             context.Tailcall(hostAddress, nativeContext);
 | |
| 
 | |
|             context.MarkLabel(lblFallback);
 | |
|             hostAddress = context.Call(typeof(NativeInterface).GetMethod(nameof(NativeInterface.GetFunctionAddress)), guestAddress);
 | |
|             context.Tailcall(hostAddress, nativeContext);
 | |
| 
 | |
|             var cfg = context.GetControlFlowGraph();
 | |
|             var retType = OperandType.I64;
 | |
|             var argTypes = new[] { OperandType.I64 };
 | |
| 
 | |
|             var func = Compiler.Compile<GuestFunction>(cfg, argTypes, retType, CompilerOptions.HighCq);
 | |
| 
 | |
|             return Marshal.GetFunctionPointerForDelegate(func);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Generates a <see cref="SlowDispatchStub"/>.
 | |
|         /// </summary>
 | |
|         /// <returns>Generated <see cref="SlowDispatchStub"/></returns>
 | |
|         private static IntPtr GenerateSlowDispatchStub()
 | |
|         {
 | |
|             var context = new EmitterContext();
 | |
| 
 | |
|             // Load the target guest address from the native context.
 | |
|             Operand nativeContext = context.LoadArgument(OperandType.I64, 0);
 | |
|             Operand guestAddress = context.Load(OperandType.I64,
 | |
|                 context.Add(nativeContext, Const((ulong)NativeContext.GetDispatchAddressOffset())));
 | |
| 
 | |
|             MethodInfo getFuncAddress = typeof(NativeInterface).GetMethod(nameof(NativeInterface.GetFunctionAddress));
 | |
|             Operand hostAddress = context.Call(getFuncAddress, guestAddress);
 | |
|             context.Tailcall(hostAddress, nativeContext);
 | |
| 
 | |
|             var cfg = context.GetControlFlowGraph();
 | |
|             var retType = OperandType.I64;
 | |
|             var argTypes = new[] { OperandType.I64 };
 | |
| 
 | |
|             var func = Compiler.Compile<GuestFunction>(cfg, argTypes, retType, CompilerOptions.HighCq);
 | |
| 
 | |
|             return Marshal.GetFunctionPointerForDelegate(func);
 | |
|         }
 | |
| 
 | |
|         /// <summary>
 | |
|         /// Generates a <see cref="DispatchLoop"/> function.
 | |
|         /// </summary>
 | |
|         /// <returns><see cref="DispatchLoop"/> function</returns>
 | |
|         private DispatcherFunction GenerateDispatchLoop()
 | |
|         {
 | |
|             var context = new EmitterContext();
 | |
| 
 | |
|             Operand beginLbl = Label();
 | |
|             Operand endLbl = Label();
 | |
| 
 | |
|             Operand nativeContext = context.LoadArgument(OperandType.I64, 0);
 | |
|             Operand guestAddress = context.Copy(
 | |
|                 context.AllocateLocal(OperandType.I64),
 | |
|                 context.LoadArgument(OperandType.I64, 1));
 | |
| 
 | |
|             Operand runningAddress = context.Add(nativeContext, Const((ulong)NativeContext.GetRunningOffset()));
 | |
|             Operand dispatchAddress = context.Add(nativeContext, Const((ulong)NativeContext.GetDispatchAddressOffset()));
 | |
| 
 | |
|             context.MarkLabel(beginLbl);
 | |
|             context.Store(dispatchAddress, guestAddress);
 | |
|             context.Copy(guestAddress, context.Call(Const((ulong)DispatchStub), OperandType.I64, nativeContext));
 | |
|             context.BranchIfFalse(endLbl, guestAddress);
 | |
|             context.BranchIfFalse(endLbl, context.Load(OperandType.I32, runningAddress));
 | |
|             context.Branch(beginLbl);
 | |
| 
 | |
|             context.MarkLabel(endLbl);
 | |
|             context.Return();
 | |
| 
 | |
|             var cfg = context.GetControlFlowGraph();
 | |
|             var retType = OperandType.None;
 | |
|             var argTypes = new[] { OperandType.I64, OperandType.I64 };
 | |
| 
 | |
|             return Compiler.Compile<DispatcherFunction>(cfg, argTypes, retType, CompilerOptions.HighCq);
 | |
|         }
 | |
|     }
 | |
| }
 |