| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 |
- using System;
- using System.Collections.Generic;
- using System.Runtime.CompilerServices;
- namespace Ryujinx.Graphics.Vulkan
- {
- interface IRefEquatable<T>
- {
- bool Equals(ref T other);
- }
- class HashTableSlim<TKey, TValue> where TKey : IRefEquatable<TKey>
- {
- private const int TotalBuckets = 16; // Must be power of 2
- private const int TotalBucketsMask = TotalBuckets - 1;
- private struct Entry
- {
- public int Hash;
- public TKey Key;
- public TValue Value;
- }
- private struct Bucket
- {
- public int Length;
- public Entry[] Entries;
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- public readonly Span<Entry> AsSpan()
- {
- return Entries == null ? Span<Entry>.Empty : Entries.AsSpan(0, Length);
- }
- }
- private readonly Bucket[] _hashTable = new Bucket[TotalBuckets];
- public IEnumerable<TKey> Keys
- {
- get
- {
- foreach (Bucket bucket in _hashTable)
- {
- for (int i = 0; i < bucket.Length; i++)
- {
- yield return bucket.Entries[i].Key;
- }
- }
- }
- }
- public IEnumerable<TValue> Values
- {
- get
- {
- foreach (Bucket bucket in _hashTable)
- {
- for (int i = 0; i < bucket.Length; i++)
- {
- yield return bucket.Entries[i].Value;
- }
- }
- }
- }
- public void Add(ref TKey key, TValue value)
- {
- Entry entry = new()
- {
- Hash = key.GetHashCode(),
- Key = key,
- Value = value,
- };
- int hashCode = key.GetHashCode();
- int bucketIndex = hashCode & TotalBucketsMask;
- ref Bucket bucket = ref _hashTable[bucketIndex];
- if (bucket.Entries != null)
- {
- int index = bucket.Length;
- if (index >= bucket.Entries.Length)
- {
- Array.Resize(ref bucket.Entries, index + 1);
- }
- bucket.Entries[index] = entry;
- }
- else
- {
- bucket.Entries =
- [
- entry
- ];
- }
- bucket.Length++;
- }
- public bool Remove(ref TKey key)
- {
- int hashCode = key.GetHashCode();
- ref Bucket bucket = ref _hashTable[hashCode & TotalBucketsMask];
- Span<Entry> entries = bucket.AsSpan();
- for (int i = 0; i < entries.Length; i++)
- {
- ref Entry entry = ref entries[i];
- if (entry.Hash == hashCode && entry.Key.Equals(ref key))
- {
- entries[(i + 1)..].CopyTo(entries[i..]);
- bucket.Length--;
- return true;
- }
- }
- return false;
- }
- public bool TryGetValue(ref TKey key, out TValue value)
- {
- int hashCode = key.GetHashCode();
- Span<Entry> entries = _hashTable[hashCode & TotalBucketsMask].AsSpan();
- for (int i = 0; i < entries.Length; i++)
- {
- ref Entry entry = ref entries[i];
- if (entry.Hash == hashCode && entry.Key.Equals(ref key))
- {
- value = entry.Value;
- return true;
- }
- }
- value = default;
- return false;
- }
- }
- }
|