mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-24 16:30:30 -07:00 
			
		
		
		
	Add inlined on translation call counting (#2190)
* Add EntryTable<TEntry>
* Add on translation call counting
* Add Counter
* Add PPTC support
* Make Counter a generic & use a 32-bit counter instead
* Return false on overflow
* Set PPTC version
* Print more information about the rejit queue
* Make Counter<T> disposable
* Remove Block.TailCall since it is not used anymore
* Apply suggestions from code review
Address gdkchan's feedback
Co-authored-by: gdkchan <gab.dark.100@gmail.com>
* Fix more stale docs
* Remove rejit requests queue logging
* Make Counter<T> finalizable
Most certainly quite an odd use case.
* Make EntryTable<T>.TryAllocate set entry to default
* Re-trigger CI
* Dispose Counters before they hit the finalizer queue
* Re-trigger CI
Just for good measure...
* Make EntryTable<T> expandable
* EntryTable is now expandable instead of being a fixed slab.
* Remove EntryTable<T>.TryAllocate
* Remove Counter<T>.TryCreate
Address LDj3SNuD's feedback
* Apply suggestions from code review
Address LDj3SNuD's feedback
Co-authored-by: LDj3SNuD <35856442+LDj3SNuD@users.noreply.github.com>
* Remove useless return
* POH approach, but the sequel
* Revert "POH approach, but the sequel"
This reverts commit 5f5abaa247.
The sequel got shelved
* Add extra documentation
Co-authored-by: gdkchan <gab.dark.100@gmail.com>
Co-authored-by: LDj3SNuD <35856442+LDj3SNuD@users.noreply.github.com>
			
			
This commit is contained in:
		
							
								
								
									
										99
									
								
								ARMeilleure/Common/Counter.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								ARMeilleure/Common/Counter.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | ||||
| using System; | ||||
|  | ||||
| namespace ARMeilleure.Common | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Represents a numeric counter which can be used for instrumentation of compiled code. | ||||
|     /// </summary> | ||||
|     /// <typeparam name="T">Type of the counter</typeparam> | ||||
|     class Counter<T> : IDisposable where T : unmanaged | ||||
|     { | ||||
|         private bool _disposed; | ||||
|         private readonly int _index; | ||||
|         private readonly EntryTable<T> _countTable; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes a new instance of the <see cref="Counter{T}"/> class from the specified | ||||
|         /// <see cref="EntryTable{T}"/> instance and index. | ||||
|         /// </summary> | ||||
|         /// <param name="countTable"><see cref="EntryTable{T}"/> instance</param> | ||||
|         /// <param name="index">Index in the <see cref="EntryTable{T}"/></param> | ||||
|         /// <exception cref="ArgumentNullException"><paramref name="countTable"/> is <see langword="null"/></exception> | ||||
|         /// <exception cref="ArgumentException"><typeparamref name="T"/> is unsupported</exception> | ||||
|         public Counter(EntryTable<T> 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."); | ||||
|             } | ||||
|  | ||||
|             _countTable = countTable ?? throw new ArgumentNullException(nameof(countTable)); | ||||
|             _index = countTable.Allocate(); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets a reference to the value of the counter. | ||||
|         /// </summary> | ||||
|         /// <exception cref="ObjectDisposedException"><see cref="Counter{T}"/> instance was disposed</exception> | ||||
|         /// <remarks> | ||||
|         /// This can refer to freed memory if the owning <see cref="EntryTable{TEntry}"/> is disposed. | ||||
|         /// </remarks> | ||||
|         public ref T Value | ||||
|         { | ||||
|             get | ||||
|             { | ||||
|                 if (_disposed) | ||||
|                 { | ||||
|                     throw new ObjectDisposedException(null); | ||||
|                 } | ||||
|  | ||||
|                 return ref _countTable.GetValue(_index); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Releases all resources used by the <see cref="Counter{T}"/> instance. | ||||
|         /// </summary> | ||||
|         public void Dispose() | ||||
|         { | ||||
|             Dispose(true); | ||||
|             GC.SuppressFinalize(this); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Releases all unmanaged and optionally managed resources used by the <see cref="Counter{T}"/> 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) | ||||
|             { | ||||
|                 try | ||||
|                 { | ||||
|                     // The index into the EntryTable is essentially an unmanaged resource since we allocate and free the | ||||
|                     // resource ourselves. | ||||
|                     _countTable.Free(_index); | ||||
|                 } | ||||
|                 catch (ObjectDisposedException) | ||||
|                 { | ||||
|                     // Can happen because _countTable may be disposed before the Counter instance. | ||||
|                 } | ||||
|  | ||||
|                 _disposed = true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Frees resources used by the <see cref="Counter{T}"/> instance. | ||||
|         /// </summary> | ||||
|         ~Counter() | ||||
|         { | ||||
|             Dispose(false); | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										196
									
								
								ARMeilleure/Common/EntryTable.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								ARMeilleure/Common/EntryTable.cs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Numerics; | ||||
| using System.Runtime.InteropServices; | ||||
|  | ||||
| namespace ARMeilleure.Common | ||||
| { | ||||
|     /// <summary> | ||||
|     /// Represents an expandable table of the type <typeparamref name="TEntry"/>, whose entries will remain at the same | ||||
|     /// address through out the table's lifetime. | ||||
|     /// </summary> | ||||
|     /// <typeparam name="TEntry">Type of the entry in the table</typeparam> | ||||
|     class EntryTable<TEntry> : IDisposable where TEntry : unmanaged | ||||
|     { | ||||
|         private bool _disposed; | ||||
|         private int _freeHint; | ||||
|         private readonly int _pageCapacity; // Number of entries per page. | ||||
|         private readonly int _pageLogCapacity; | ||||
|         private readonly Dictionary<int, IntPtr> _pages; | ||||
|         private readonly BitMap _allocated; | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Initializes a new instance of the <see cref="EntryTable{TEntry}"/> class with the desired page size in | ||||
|         /// bytes. | ||||
|         /// </summary> | ||||
|         /// <param name="pageSize">Desired page size in bytes</param> | ||||
|         /// <exception cref="ArgumentOutOfRangeException"><paramref name="pageSize"/> is less than 0</exception> | ||||
|         /// <exception cref="ArgumentException"><typeparamref name="TEntry"/>'s size is zero</exception> | ||||
|         /// <remarks> | ||||
|         /// The actual page size may be smaller or larger depending on the size of <typeparamref name="TEntry"/>. | ||||
|         /// </remarks> | ||||
|         public unsafe EntryTable(int pageSize = 4096) | ||||
|         { | ||||
|             if (pageSize < 0) | ||||
|             { | ||||
|                 throw new ArgumentOutOfRangeException(nameof(pageSize), "Page size cannot be negative."); | ||||
|             } | ||||
|  | ||||
|             if (sizeof(TEntry) == 0) | ||||
|             { | ||||
|                 throw new ArgumentException("Size of TEntry cannot be zero."); | ||||
|             } | ||||
|  | ||||
|             _allocated = new BitMap(); | ||||
|             _pages = new Dictionary<int, IntPtr>(); | ||||
|             _pageLogCapacity = BitOperations.Log2((uint)(pageSize / sizeof(TEntry))); | ||||
|             _pageCapacity = 1 << _pageLogCapacity; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Allocates an entry in the <see cref="EntryTable{TEntry}"/>. | ||||
|         /// </summary> | ||||
|         /// <returns>Index of entry allocated in the table</returns> | ||||
|         /// <exception cref="ObjectDisposedException"><see cref="EntryTable{TEntry}"/> instance was disposed</exception> | ||||
|         public int Allocate() | ||||
|         { | ||||
|             if (_disposed) | ||||
|             { | ||||
|                 throw new ObjectDisposedException(null); | ||||
|             } | ||||
|  | ||||
|             lock (_allocated) | ||||
|             { | ||||
|                 if (_allocated.IsSet(_freeHint)) | ||||
|                 { | ||||
|                     _freeHint = _allocated.FindFirstUnset(); | ||||
|                 } | ||||
|  | ||||
|                 int index = _freeHint++; | ||||
|                 var page = GetPage(index); | ||||
|  | ||||
|                 _allocated.Set(index); | ||||
|  | ||||
|                 GetValue(page, index) = default; | ||||
|  | ||||
|                 return index; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Frees the entry at the specified <paramref name="index"/>. | ||||
|         /// </summary> | ||||
|         /// <param name="index">Index of entry to free</param> | ||||
|         /// <exception cref="ObjectDisposedException"><see cref="EntryTable{TEntry}"/> instance was disposed</exception> | ||||
|         public void Free(int index) | ||||
|         { | ||||
|             if (_disposed) | ||||
|             { | ||||
|                 throw new ObjectDisposedException(null); | ||||
|             } | ||||
|  | ||||
|             lock (_allocated) | ||||
|             { | ||||
|                 if (_allocated.IsSet(index)) | ||||
|                 { | ||||
|                     _allocated.Clear(index); | ||||
|  | ||||
|                     _freeHint = index; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets a reference to the entry at the specified allocated <paramref name="index"/>. | ||||
|         /// </summary> | ||||
|         /// <param name="index">Index of the entry</param> | ||||
|         /// <returns>Reference to the entry at the specified <paramref name="index"/></returns> | ||||
|         /// <exception cref="ObjectDisposedException"><see cref="EntryTable{TEntry}"/> instance was disposed</exception> | ||||
|         /// <exception cref="ArgumentException">Entry at <paramref name="index"/> is not allocated</exception> | ||||
|         public ref TEntry GetValue(int index) | ||||
|         { | ||||
|             if (_disposed) | ||||
|             { | ||||
|                 throw new ObjectDisposedException(null); | ||||
|             } | ||||
|  | ||||
|             lock (_allocated) | ||||
|             { | ||||
|                 if (!_allocated.IsSet(index)) | ||||
|                 { | ||||
|                     throw new ArgumentException("Entry at the specified index was not allocated", nameof(index)); | ||||
|                 } | ||||
|  | ||||
|                 var page = GetPage(index); | ||||
|  | ||||
|                 return ref GetValue(page, index); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets a reference to the entry at using the specified <paramref name="index"/> from the specified | ||||
|         /// <paramref name="page"/>. | ||||
|         /// </summary> | ||||
|         /// <param name="page">Page to use</param> | ||||
|         /// <param name="index">Index to use</param> | ||||
|         /// <returns>Reference to the entry</returns> | ||||
|         private ref TEntry GetValue(Span<TEntry> page, int index) | ||||
|         { | ||||
|             return ref page[index & (_pageCapacity - 1)]; | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Gets the page for the specified <see cref="index"/>. | ||||
|         /// </summary> | ||||
|         /// <param name="index">Index to use</param> | ||||
|         /// <returns>Page for the specified <see cref="index"/></returns> | ||||
|         private unsafe Span<TEntry> GetPage(int index) | ||||
|         { | ||||
|             var pageIndex = (int)((uint)(index & ~(_pageCapacity - 1)) >> _pageLogCapacity); | ||||
|  | ||||
|             if (!_pages.TryGetValue(pageIndex, out IntPtr page)) | ||||
|             { | ||||
|                 page = Marshal.AllocHGlobal(sizeof(TEntry) * _pageCapacity); | ||||
|  | ||||
|                 _pages.Add(pageIndex, page); | ||||
|             } | ||||
|  | ||||
|             return new Span<TEntry>((void*)page, _pageCapacity); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Releases all resources used by the <see cref="EntryTable{TEntry}"/> instance. | ||||
|         /// </summary> | ||||
|         public void Dispose() | ||||
|         { | ||||
|             Dispose(true); | ||||
|             GC.SuppressFinalize(this); | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Releases all unmanaged and optionally managed resources used by the <see cref="EntryTable{TEntry}{T}"/> | ||||
|         /// 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) | ||||
|             { | ||||
|                 foreach (var page in _pages.Values) | ||||
|                 { | ||||
|                     Marshal.FreeHGlobal(page); | ||||
|                 } | ||||
|  | ||||
|                 _disposed = true; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// <summary> | ||||
|         /// Frees resources used by the <see cref="EntryTable{TEntry}"/> instance. | ||||
|         /// </summary> | ||||
|         ~EntryTable() | ||||
|         { | ||||
|             Dispose(false); | ||||
|         } | ||||
|     } | ||||
| } | ||||
		Reference in New Issue
	
	Block a user