mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 13:52:29 -07:00 
			
		
		
		
	Add multi-level function table (#2228)
* 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
			
			
This commit is contained in:
		
							
								
								
									
										248
									
								
								ARMeilleure/Translation/TranslatorStubs.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										248
									
								
								ARMeilleure/Translation/TranslatorStubs.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,248 @@ | ||||
| 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); | ||||
|         } | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user