diff --git a/ARMeilleure/Common/Counter.cs b/ARMeilleure/Common/Counter.cs index d3d28730e8..f8c5d27325 100644 --- a/ARMeilleure/Common/Counter.cs +++ b/ARMeilleure/Common/Counter.cs @@ -1,20 +1,23 @@ -namespace ARMeilleure.Common +using System; + +namespace ARMeilleure.Common { /// - /// Represents an 8-bit counter. + /// Represents a numeric counter. /// - class Counter + /// Type of the counter + class Counter where T : unmanaged { private readonly int _index; - private readonly EntryTable _countTable; + private readonly EntryTable _countTable; /// - /// Initializes a new instance of the class from the specified - /// instance and index. + /// Initializes a new instance of the class from the specified + /// instance and index. /// /// instance - /// Index in the - private Counter(EntryTable countTable, int index) + /// Index in the + private Counter(EntryTable countTable, int index) { _countTable = countTable; _index = index; @@ -23,7 +26,7 @@ /// /// Gets a reference to the value of the counter. /// - public ref byte Value => ref _countTable.GetValue(_index); + public ref T Value => ref _countTable.GetValue(_index); /// /// Tries to create a instance from the specified instance. @@ -31,11 +34,28 @@ /// from which to create the /// instance if success; otherwise /// if success; otherwise - public static bool TryCreate(EntryTable countTable, out Counter counter) + /// is + /// is unsupported + public static bool TryCreate(EntryTable countTable, out Counter counter) { + if (countTable == null) + { + throw new ArgumentNullException(nameof(countTable)); + } + + if (typeof(T) != typeof(byte) && typeof(T) != typeof(sbyte) && + typeof(T) != typeof(short) && typeof(T) != typeof(ushort) && + typeof(T) != typeof(int) && typeof(T) != typeof(uint) && + typeof(T) != typeof(long) && typeof(T) != typeof(ulong) && + typeof(T) != typeof(nint) && typeof(T) != typeof(nuint) && + typeof(T) != typeof(float) && typeof(T) != typeof(double)) + { + throw new ArgumentException("Counter does not support the specified type", nameof(countTable)); + } + if (countTable.TryAllocate(out int index)) { - counter = new Counter(countTable, index); + counter = new Counter(countTable, index); return true; } diff --git a/ARMeilleure/Translation/ArmEmitterContext.cs b/ARMeilleure/Translation/ArmEmitterContext.cs index ac24a45c95..ad44b0cfb8 100644 --- a/ARMeilleure/Translation/ArmEmitterContext.cs +++ b/ARMeilleure/Translation/ArmEmitterContext.cs @@ -42,7 +42,7 @@ namespace ARMeilleure.Translation public IMemoryManager Memory { get; } public JumpTable JumpTable { get; } - public EntryTable CountTable { get; } + public EntryTable CountTable { get; } public ulong EntryAddress { get; } public bool HighCq { get; } @@ -51,7 +51,7 @@ namespace ARMeilleure.Translation public ArmEmitterContext( IMemoryManager memory, JumpTable jumpTable, - EntryTable countTable, + EntryTable countTable, ulong entryAddress, bool highCq, Aarch32Mode mode) diff --git a/ARMeilleure/Translation/PTC/Ptc.cs b/ARMeilleure/Translation/PTC/Ptc.cs index d643086e32..0142934eb5 100644 --- a/ARMeilleure/Translation/PTC/Ptc.cs +++ b/ARMeilleure/Translation/PTC/Ptc.cs @@ -540,7 +540,8 @@ namespace ARMeilleure.Translation.PTC } } - internal static void LoadTranslations(ConcurrentDictionary funcs, IMemoryManager memory, JumpTable jumpTable, EntryTable countTable) + internal static void LoadTranslations(ConcurrentDictionary funcs, IMemoryManager memory, + JumpTable jumpTable, EntryTable countTable) { if (AreCarriersEmpty()) { @@ -569,7 +570,7 @@ namespace ARMeilleure.Translation.PTC { byte[] code = ReadCode(index, infoEntry.CodeLength); - Counter callCounter = null; + Counter callCounter = null; if (infoEntry.RelocEntriesCount != 0) { @@ -679,7 +680,8 @@ namespace ARMeilleure.Translation.PTC return relocEntries; } - private static bool PatchCode(Span code, RelocEntry[] relocEntries, IntPtr pageTablePointer, JumpTable jumpTable, EntryTable countTable, out Counter callCounter) + private static bool PatchCode(Span code, RelocEntry[] relocEntries, IntPtr pageTablePointer, + JumpTable jumpTable, EntryTable countTable, out Counter callCounter) { callCounter = null; @@ -702,7 +704,7 @@ namespace ARMeilleure.Translation.PTC else if (relocEntry.Index == CountTableIndex) { // If we could not allocate an entry on the count table we dip. - if (!Counter.TryCreate(countTable, out Counter counter)) + if (!Counter.TryCreate(countTable, out Counter counter)) { return false; } @@ -745,7 +747,8 @@ namespace ARMeilleure.Translation.PTC return new UnwindInfo(pushEntries, prologueSize); } - private static TranslatedFunction FastTranslate(byte[] code, Counter callCounter, ulong guestSize, UnwindInfo unwindInfo, bool highCq) + private static TranslatedFunction FastTranslate(byte[] code, Counter callCounter, ulong guestSize, + UnwindInfo unwindInfo, bool highCq) { CompiledFunction cFunc = new CompiledFunction(code, unwindInfo); @@ -794,7 +797,8 @@ namespace ARMeilleure.Translation.PTC } } - internal static void MakeAndSaveTranslations(ConcurrentDictionary funcs, IMemoryManager memory, JumpTable jumpTable, EntryTable countTable) + internal static void MakeAndSaveTranslations(ConcurrentDictionary funcs, IMemoryManager memory, + JumpTable jumpTable, EntryTable countTable) { var profiledFuncsToTranslate = PtcProfiler.GetProfiledFuncsToTranslate(funcs); diff --git a/ARMeilleure/Translation/TranslatedFunction.cs b/ARMeilleure/Translation/TranslatedFunction.cs index c8adbe4cb3..04dd769c1c 100644 --- a/ARMeilleure/Translation/TranslatedFunction.cs +++ b/ARMeilleure/Translation/TranslatedFunction.cs @@ -8,12 +8,12 @@ namespace ARMeilleure.Translation { private readonly GuestFunction _func; // Ensure that this delegate will not be garbage collected. - public Counter CallCounter { get; } + public Counter CallCounter { get; } public ulong GuestSize { get; } public bool HighCq { get; } public IntPtr FuncPtr { get; } - public TranslatedFunction(GuestFunction func, Counter callCounter, ulong guestSize, bool highCq) + public TranslatedFunction(GuestFunction func, Counter callCounter, ulong guestSize, bool highCq) { _func = func; CallCounter = callCounter; diff --git a/ARMeilleure/Translation/Translator.cs b/ARMeilleure/Translation/Translator.cs index 50d363cd8d..f3ab15be05 100644 --- a/ARMeilleure/Translation/Translator.cs +++ b/ARMeilleure/Translation/Translator.cs @@ -37,7 +37,7 @@ namespace ARMeilleure.Translation private JumpTable _jumpTable; internal JumpTable JumpTable => _jumpTable; - internal EntryTable CountTable { get; } + internal EntryTable CountTable { get; } private volatile int _threadCount; @@ -59,7 +59,7 @@ namespace ARMeilleure.Translation _backgroundTranslatorEvent = new AutoResetEvent(false); _backgroundTranslatorLock = new ReaderWriterLock(); - CountTable = new EntryTable(capacity: 16 * 1024 * 1024); + CountTable = new EntryTable(capacity: 4 * 1024 * 1024); JitCache.Initialize(allocator); @@ -239,7 +239,7 @@ namespace ARMeilleure.Translation internal static TranslatedFunction Translate( IMemoryManager memory, JumpTable jumpTable, - EntryTable countTable, + EntryTable countTable, ulong address, ExecutionMode mode, bool highCq) @@ -256,7 +256,7 @@ namespace ARMeilleure.Translation Logger.StartPass(PassName.Translation); - Counter counter = null; + Counter counter = null; if (!context.HighCq) { @@ -415,29 +415,23 @@ namespace ARMeilleure.Translation return context.GetControlFlowGraph(); } - internal static void EmitRejitCheck(ArmEmitterContext context, out Counter counter) + internal static void EmitRejitCheck(ArmEmitterContext context, out Counter counter) { - if (!Counter.TryCreate(context.CountTable, out counter)) + if (!Counter.TryCreate(context.CountTable, out counter)) { return; } - Operand lblRejit = Label(); - Operand lblAdd = Label(); Operand lblEnd = Label(); Operand address = Const(ref counter.Value, Ptc.CountTableIndex); - Operand count = context.Load8(address); - context.BranchIf(lblAdd, count, Const(100), Comparison.LessUI); - context.BranchIf(lblRejit, count, Const(100), Comparison.Equal); - context.Branch(lblEnd); + Operand curCount = context.Load(OperandType.I32, address); + Operand count = context.Add(curCount, Const(1)); + context.Store(address, count); + context.BranchIf(lblEnd, curCount, Const(100), Comparison.NotEqual, BasicBlockFrequency.Cold); - context.MarkLabel(lblRejit, BasicBlockFrequency.Cold); context.Call(typeof(NativeInterface).GetMethod(nameof(NativeInterface.EnqueueForRejit)), Const(context.EntryAddress)); - context.MarkLabel(lblAdd, BasicBlockFrequency.Cold); - context.Store8(address, context.Add(count, Const(1))); - context.MarkLabel(lblEnd); }