| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188 |
- using System;
- using System.Collections.Generic;
- using System.Numerics;
- 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(NativeAllocator.Instance);
- _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()
- {
- ObjectDisposedException.ThrowIf(_disposed, this);
- 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)
- {
- ObjectDisposedException.ThrowIf(_disposed, this);
- 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)
- {
- ObjectDisposedException.ThrowIf(_disposed, this);
- 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 = (IntPtr)NativeAllocator.Instance.Allocate((uint)sizeof(TEntry) * (uint)_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}"/>
- /// instance.
- /// </summary>
- /// <param name="disposing"><see langword="true"/> to dispose managed resources also; otherwise just unmanaged resouces</param>
- protected unsafe virtual void Dispose(bool disposing)
- {
- if (!_disposed)
- {
- _allocated.Dispose();
- foreach (var page in _pages.Values)
- {
- NativeAllocator.Instance.Free((void*)page);
- }
- _disposed = true;
- }
- }
- /// <summary>
- /// Frees resources used by the <see cref="EntryTable{TEntry}"/> instance.
- /// </summary>
- ~EntryTable()
- {
- Dispose(false);
- }
- }
- }
|