|
| 1 | +using System; |
| 2 | +using System.Security; |
| 3 | + |
| 4 | +namespace Unity.Storage |
| 5 | +{ |
| 6 | + [SecuritySafeCritical] |
| 7 | + public class QuickSet<TValue> |
| 8 | + { |
| 9 | + #region Fields |
| 10 | + |
| 11 | + private int _prime; |
| 12 | + private int[] Buckets; |
| 13 | + private Entry[] Entries; |
| 14 | + public int Count { get; private set; } |
| 15 | + |
| 16 | + #endregion |
| 17 | + |
| 18 | + |
| 19 | + #region Constructors |
| 20 | + |
| 21 | + public QuickSet() |
| 22 | + { |
| 23 | + var size = Primes[_prime]; |
| 24 | + Buckets = new int[size]; |
| 25 | + Entries = new Entry[size]; |
| 26 | + |
| 27 | +#if !NET40 |
| 28 | + unsafe |
| 29 | + { |
| 30 | + fixed (int* bucketsPtr = Buckets) |
| 31 | + { |
| 32 | + int* ptr = bucketsPtr; |
| 33 | + var end = bucketsPtr + Buckets.Length; |
| 34 | + while (ptr < end) *ptr++ = -1; |
| 35 | + } |
| 36 | + } |
| 37 | +#else |
| 38 | + for(int i = 0; i < Buckets.Length; i++) |
| 39 | + Buckets[i] = -1; |
| 40 | +#endif |
| 41 | + } |
| 42 | + |
| 43 | + #endregion |
| 44 | + |
| 45 | + |
| 46 | + #region Public Methods |
| 47 | + |
| 48 | + public bool Add(int hashCode, TValue value) |
| 49 | + { |
| 50 | + var collisions = 0; |
| 51 | + var targetBucket = (hashCode & UnityContainer.HashMask) % Buckets.Length; |
| 52 | + |
| 53 | + // Check for the existing |
| 54 | + for (var i = Buckets[targetBucket]; i >= 0; i = Entries[i].Next) |
| 55 | + { |
| 56 | + ref var candidate = ref Entries[i]; |
| 57 | + if (candidate.HashCode != hashCode || !Equals(candidate.Value, value)) |
| 58 | + { |
| 59 | + collisions++; |
| 60 | + continue; |
| 61 | + } |
| 62 | + |
| 63 | + // Already exists |
| 64 | + return false; |
| 65 | + } |
| 66 | + |
| 67 | + // Expand if required |
| 68 | + if (Count >= Entries.Length || 3 < collisions) |
| 69 | + { |
| 70 | + Expand(); |
| 71 | + targetBucket = (hashCode & UnityContainer.HashMask) % Buckets.Length; |
| 72 | + } |
| 73 | + |
| 74 | + // Add registration |
| 75 | + ref var entry = ref Entries[Count]; |
| 76 | + entry.HashCode = hashCode; |
| 77 | + entry.Value = value; |
| 78 | + entry.Next = Buckets[targetBucket]; |
| 79 | + Buckets[targetBucket] = Count++; |
| 80 | + |
| 81 | + return true; |
| 82 | + } |
| 83 | + |
| 84 | + #endregion |
| 85 | + |
| 86 | + |
| 87 | + #region Entry Type |
| 88 | + |
| 89 | + private struct Entry |
| 90 | + { |
| 91 | + public int HashCode; |
| 92 | + public TValue Value; |
| 93 | + public int Next; |
| 94 | + } |
| 95 | + |
| 96 | + #endregion |
| 97 | + |
| 98 | + |
| 99 | + #region Implementation |
| 100 | + |
| 101 | + private void Expand() |
| 102 | + { |
| 103 | + var entries = Entries; |
| 104 | + |
| 105 | + _prime += 1; |
| 106 | + |
| 107 | + var size = Primes[_prime]; |
| 108 | + Buckets = new int[size]; |
| 109 | + Entries = new Entry[size]; |
| 110 | + |
| 111 | +#if !NET40 |
| 112 | + unsafe |
| 113 | + { |
| 114 | + fixed (int* bucketsPtr = Buckets) |
| 115 | + { |
| 116 | + int* ptr = bucketsPtr; |
| 117 | + var end = bucketsPtr + Buckets.Length; |
| 118 | + while (ptr < end) *ptr++ = -1; |
| 119 | + } |
| 120 | + } |
| 121 | +#else |
| 122 | + for(int i = 0; i < Buckets.Length; i++) |
| 123 | + Buckets[i] = -1; |
| 124 | +#endif |
| 125 | + Array.Copy(entries, 0, Entries, 0, Count); |
| 126 | + for (var i = 0; i < Count; i++) |
| 127 | + { |
| 128 | + var hashCode = Entries[i].HashCode & UnityContainer.HashMask; |
| 129 | + if (hashCode < 0) continue; |
| 130 | + |
| 131 | + var bucket = hashCode % Buckets.Length; |
| 132 | + Entries[i].Next = Buckets[bucket]; |
| 133 | + Buckets[bucket] = i; |
| 134 | + } |
| 135 | + } |
| 136 | + |
| 137 | + public static readonly int[] Primes = { |
| 138 | + 11, 37, 71, 107, 131, 163, 197, 239, 293, 353, 431, 521, 631, 761, 919, 1103, 1327, 1597, |
| 139 | + 1931, 2333, 2801, 3371, 4049, 4861, 5839, 7013, 8419, 10103, 12143, 14591, 17519, 21023, |
| 140 | + 25229, 30293, 36353, 43627, 52361, 62851, 75431, 90523, 108631, 130363, 156437, 187751, |
| 141 | + 225307, 270371, 324449, 389357, 467237, 560689, 672827, 807403, 968897, 1162687, 1395263, |
| 142 | + 1674319, 2009191, 2411033, 2893249, 3471899, 4166287, 4999559, 5999471, 7199369}; |
| 143 | + |
| 144 | + #endregion |
| 145 | + } |
| 146 | +} |
0 commit comments