| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261 |
- using ARMeilleure.Diagnostics.EventSources;
- using System;
- using System.Collections.Generic;
- using System.Runtime.InteropServices;
- namespace ARMeilleure.Common
- {
- /// <summary>
- /// Represents a table of guest address to a value.
- /// </summary>
- /// <typeparam name="TEntry">Type of the value</typeparam>
- unsafe class AddressTable<TEntry> : IDisposable where TEntry : unmanaged
- {
- /// <summary>
- /// Represents a level in an <see cref="AddressTable{TEntry}"/>.
- /// </summary>
- public readonly struct Level
- {
- /// <summary>
- /// Gets the index of the <see cref="Level"/> in the guest address.
- /// </summary>
- public int Index { get; }
- /// <summary>
- /// Gets the length of the <see cref="Level"/> in the guest address.
- /// </summary>
- public int Length { get; }
- /// <summary>
- /// Gets the mask which masks the bits used by the <see cref="Level"/>.
- /// </summary>
- public ulong Mask => ((1ul << Length) - 1) << Index;
- /// <summary>
- /// Initializes a new instance of the <see cref="Level"/> structure with the specified
- /// <paramref name="index"/> and <paramref name="length"/>.
- /// </summary>
- /// <param name="index">Index of the <see cref="Level"/></param>
- /// <param name="length">Length of the <see cref="Level"/></param>
- public Level(int index, int length)
- {
- (Index, Length) = (index, length);
- }
- /// <summary>
- /// Gets the value of the <see cref="Level"/> from the specified guest <paramref name="address"/>.
- /// </summary>
- /// <param name="address">Guest address</param>
- /// <returns>Value of the <see cref="Level"/> from the specified guest <paramref name="address"/></returns>
- public int GetValue(ulong address)
- {
- return (int)((address & Mask) >> Index);
- }
- }
- private bool _disposed;
- private TEntry** _table;
- private readonly List<IntPtr> _pages;
- /// <summary>
- /// Gets the bits used by the <see cref="Levels"/> of the <see cref="AddressTable{TEntry}"/> instance.
- /// </summary>
- public ulong Mask { get; }
- /// <summary>
- /// Gets the <see cref="Level"/>s used by the <see cref="AddressTable{TEntry}"/> instance.
- /// </summary>
- public Level[] Levels { get; }
- /// <summary>
- /// Gets or sets the default fill value of newly created leaf pages.
- /// </summary>
- public TEntry Fill { get; set; }
- /// <summary>
- /// Gets the base address of the <see cref="EntryTable{TEntry}"/>.
- /// </summary>
- /// <exception cref="ObjectDisposedException"><see cref="EntryTable{TEntry}"/> instance was disposed</exception>
- public IntPtr Base
- {
- get
- {
- if (_disposed)
- {
- throw new ObjectDisposedException(null);
- }
- lock (_pages)
- {
- return (IntPtr)GetRootPage();
- }
- }
- }
- /// <summary>
- /// Constructs a new instance of the <see cref="AddressTable{TEntry}"/> class with the specified list of
- /// <see cref="Level"/>.
- /// </summary>
- /// <exception cref="ArgumentNullException"><paramref name="levels"/> is null</exception>
- /// <exception cref="ArgumentException">Length of <paramref name="levels"/> is less than 2</exception>
- public AddressTable(Level[] levels)
- {
- if (levels == null)
- {
- throw new ArgumentNullException(nameof(levels));
- }
- if (levels.Length < 2)
- {
- throw new ArgumentException("Table must be at least 2 levels deep.", nameof(levels));
- }
- _pages = new List<IntPtr>(capacity: 16);
- Levels = levels;
- Mask = 0;
- foreach (var level in Levels)
- {
- Mask |= level.Mask;
- }
- }
- /// <summary>
- /// Determines if the specified <paramref name="address"/> is in the range of the
- /// <see cref="AddressTable{TEntry}"/>.
- /// </summary>
- /// <param name="address">Guest address</param>
- /// <returns><see langword="true"/> if is valid; otherwise <see langword="false"/></returns>
- public bool IsValid(ulong address)
- {
- return (address & ~Mask) == 0;
- }
- /// <summary>
- /// Gets a reference to the value at the specified guest <paramref name="address"/>.
- /// </summary>
- /// <param name="address">Guest address</param>
- /// <returns>Reference to the value at the specified guest <paramref name="address"/></returns>
- /// <exception cref="ObjectDisposedException"><see cref="EntryTable{TEntry}"/> instance was disposed</exception>
- /// <exception cref="ArgumentException"><paramref name="address"/> is not mapped</exception>
- public ref TEntry GetValue(ulong address)
- {
- if (_disposed)
- {
- throw new ObjectDisposedException(null);
- }
- if (!IsValid(address))
- {
- throw new ArgumentException($"Address 0x{address:X} is not mapped onto the table.", nameof(address));
- }
- lock (_pages)
- {
- return ref GetPage(address)[Levels[^1].GetValue(address)];
- }
- }
- /// <summary>
- /// Gets the leaf page for the specified guest <paramref name="address"/>.
- /// </summary>
- /// <param name="address">Guest address</param>
- /// <returns>Leaf page for the specified guest <paramref name="address"/></returns>
- private TEntry* GetPage(ulong address)
- {
- TEntry** page = GetRootPage();
- for (int i = 0; i < Levels.Length - 1; i++)
- {
- ref Level level = ref Levels[i];
- ref TEntry* nextPage = ref page[level.GetValue(address)];
- if (nextPage == null)
- {
- ref Level nextLevel = ref Levels[i + 1];
- nextPage = i == Levels.Length - 2 ?
- (TEntry*)Allocate(1 << nextLevel.Length, Fill, leaf: true) :
- (TEntry*)Allocate(1 << nextLevel.Length, IntPtr.Zero, leaf: false);
- }
- page = (TEntry**)nextPage;
- }
- return (TEntry*)page;
- }
- /// <summary>
- /// Lazily initialize and get the root page of the <see cref="AddressTable{TEntry}"/>.
- /// </summary>
- /// <returns>Root page of the <see cref="AddressTable{TEntry}"/></returns>
- private TEntry** GetRootPage()
- {
- if (_table == null)
- {
- _table = (TEntry**)Allocate(1 << Levels[0].Length, fill: IntPtr.Zero, leaf: false);
- }
- return _table;
- }
- /// <summary>
- /// Allocates a block of memory of the specified type and length.
- /// </summary>
- /// <typeparam name="T">Type of elements</typeparam>
- /// <param name="length">Number of elements</param>
- /// <param name="fill">Fill value</param>
- /// <param name="leaf"><see langword="true"/> if leaf; otherwise <see langword=""="false"/></param>
- /// <returns>Allocated block</returns>
- private IntPtr Allocate<T>(int length, T fill, bool leaf) where T : unmanaged
- {
- var size = sizeof(T) * length;
- var page = (IntPtr)NativeAllocator.Instance.Allocate((uint)size);
- var span = new Span<T>((void*)page, length);
- span.Fill(fill);
- _pages.Add(page);
- AddressTableEventSource.Log.Allocated(size, leaf);
- return page;
- }
- /// <summary>
- /// Releases all resources used by the <see cref="AddressTable{TEntry}"/> instance.
- /// </summary>
- public void Dispose()
- {
- Dispose(true);
- GC.SuppressFinalize(this);
- }
- /// <summary>
- /// Releases all unmanaged and optionally managed resources used by the <see cref="AddressTable{TEntry}"/>
- /// 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)
- {
- Marshal.FreeHGlobal(page);
- }
- _disposed = true;
- }
- }
- /// <summary>
- /// Frees resources used by the <see cref="AddressTable{TEntry}"/> instance.
- /// </summary>
- ~AddressTable()
- {
- Dispose(false);
- }
- }
- }
|