using System; using System.Runtime.InteropServices; using NUnit.Framework; using Unity.Burst; using Unity.Collections; using Unity.Collections.Tests; using Unity.Jobs; using UnityEngine; using UnityEngine.TestTools; using Unity.Collections.LowLevel.Unsafe; #if !UNITY_PORTABLE_TEST_RUNNER using System.Text.RegularExpressions; #endif using Assert = FastAssert; internal class NativeHashMapTests : CollectionsTestFixture { #pragma warning disable 0649 // always default value #if !UNITY_PORTABLE_TEST_RUNNER struct NonBlittableStruct : IEquatable { object o; public bool Equals(NonBlittableStruct other) { return Equals(o, other.o); } public override bool Equals(object obj) { if (ReferenceEquals(null, obj)) return false; return obj is NonBlittableStruct other && Equals(other); } public override int GetHashCode() { return (o != null ? o.GetHashCode() : 0); } } #endif #pragma warning restore 0649 #if !UNITY_PORTABLE_TEST_RUNNER [Test, DotsRuntimeIgnore] public void NativeHashMap_Non_Blittable_Throws() { #pragma warning disable 0219 // assigned but its value is never used Assert.Throws(() => { var hashMap = new NativeHashMap(16, Allocator.Temp); }); Assert.Throws(() => { var hashMap = new NativeHashMap(16, Allocator.Temp); }); #pragma warning restore 0219 } #endif static void ExpectedCount(ref NativeHashMap container, int expected) where TKey : struct, IEquatable where TValue : struct { Assert.AreEqual(expected == 0, container.IsEmpty); Assert.AreEqual(expected, container.Count()); } [Test] public void NativeHashMap_TryAdd_TryGetValue_Clear() { var hashMap = new NativeHashMap(16, Allocator.Temp); ExpectedCount(ref hashMap, 0); int iSquared; // Make sure GetValue fails if hash map is empty Assert.IsFalse(hashMap.TryGetValue(0, out iSquared), "TryGetValue on empty hash map did not fail"); // Make sure inserting values work for (int i = 0; i < 16; ++i) Assert.IsTrue(hashMap.TryAdd(i, i * i), "Failed to add value"); ExpectedCount(ref hashMap, 16); // Make sure inserting duplicate keys fails for (int i = 0; i < 16; ++i) Assert.IsFalse(hashMap.TryAdd(i, i), "Adding duplicate keys did not fail"); ExpectedCount(ref hashMap, 16); // Make sure reading the inserted values work for (int i = 0; i < 16; ++i) { Assert.IsTrue(hashMap.TryGetValue(i, out iSquared), "Failed get value from hash table"); Assert.AreEqual(iSquared, i * i, "Got the wrong value from the hash table"); } // Make sure clearing removes all keys hashMap.Clear(); ExpectedCount(ref hashMap, 0); for (int i = 0; i < 16; ++i) Assert.IsFalse(hashMap.TryGetValue(i, out iSquared), "Got value from hash table after clearing"); hashMap.Dispose(); } [Test] public void NativeHashMap_Full_HashMap_Throws() { var hashMap = new NativeHashMap(16, Allocator.Temp); // Fill the hash map for (int i = 0; i < 16; ++i) Assert.IsTrue(hashMap.TryAdd(i, i), "Failed to add value"); // Make sure overallocating throws and exception if using the Concurrent version - normal hash map would grow var cHashMap = hashMap.AsParallelWriter(); Assert.Throws(() => { cHashMap.TryAdd(100, 100); }); hashMap.Dispose(); } [Test] public void NativeHashMap_Double_Deallocate_Throws() { var hashMap = new NativeHashMap(16, CommonRwdAllocator.Handle); hashMap.Dispose(); Assert.Throws( () => { hashMap.Dispose(); }); } [Test] public void NativeHashMap_Key_Collisions() { var hashMap = new NativeHashMap(16, Allocator.Temp); int iSquared; // Make sure GetValue fails if hash map is empty Assert.IsFalse(hashMap.TryGetValue(0, out iSquared), "TryGetValue on empty hash map did not fail"); // Make sure inserting values work for (int i = 0; i < 8; ++i) Assert.IsTrue(hashMap.TryAdd(i, i * i), "Failed to add value"); // The bucket size is capacity * 2, adding that number should result in hash collisions for (int i = 0; i < 8; ++i) Assert.IsTrue(hashMap.TryAdd(i + 32, i), "Failed to add value with potential hash collision"); // Make sure reading the inserted values work for (int i = 0; i < 8; ++i) { Assert.IsTrue(hashMap.TryGetValue(i, out iSquared), "Failed get value from hash table"); Assert.AreEqual(iSquared, i * i, "Got the wrong value from the hash table"); } for (int i = 0; i < 8; ++i) { Assert.IsTrue(hashMap.TryGetValue(i + 32, out iSquared), "Failed get value from hash table"); Assert.AreEqual(iSquared, i, "Got the wrong value from the hash table"); } hashMap.Dispose(); } [StructLayout(LayoutKind.Explicit)] unsafe struct LargeKey : IEquatable { [FieldOffset(0)] public int* Ptr; [FieldOffset(300)] int x; public bool Equals(LargeKey rhs) { return Ptr == rhs.Ptr; } public override int GetHashCode() { return (int)Ptr; } } [BurstCompile] struct HashMapTryAddAtomic : IJobParallelFor { [ReadOnly] public NativeArray keys; [WriteOnly] public NativeHashMap.ParallelWriter hashMap; public void Execute(int index) { hashMap.TryAdd(keys[index & (keys.Length - 1)], false); } } #if !(UNITY_DOTSRUNTIME && UNITY_WEBGL) // https://unity3d.atlassian.net/browse/DOTSR-2039 [Test] public unsafe void NativeHashMap_Key_Collisions_FromJobs() { // Assert.True(false); var keys = CollectionHelper.CreateNativeArray(4, CommonRwdAllocator.Handle); for (var i = 0; i < keys.Length; i++) { keys[i] = new LargeKey { Ptr = (int*)(((ulong)i) << 32) }; } for (var spin = 0; spin < 1024; spin++) { var hashMap = new NativeHashMap(32, CommonRwdAllocator.Handle); var jobHandle = new HashMapTryAddAtomic { keys = keys, hashMap = hashMap.AsParallelWriter(), } .Schedule(8, 1); jobHandle.Complete(); Assert.AreEqual(keys.Length, hashMap.Count()); for (var i = 0; i < keys.Length; ++i) { var key = new LargeKey { Ptr = (int*)(((ulong)i) << 32) }; Assert.IsTrue(hashMap.ContainsKey(key)); } hashMap.Dispose(); } keys.Dispose(); } #endif [Test] public void NativeHashMap_HashMapSupportsAutomaticCapacityChange() { var hashMap = new NativeHashMap(1, Allocator.Temp); int iSquared; // Make sure inserting values work and grows the capacity for (int i = 0; i < 8; ++i) Assert.IsTrue(hashMap.TryAdd(i, i * i), "Failed to add value"); Assert.IsTrue(hashMap.Capacity >= 8, "Capacity was not updated correctly"); // Make sure reading the inserted values work for (int i = 0; i < 8; ++i) { Assert.IsTrue(hashMap.TryGetValue(i, out iSquared), "Failed get value from hash table"); Assert.AreEqual(iSquared, i * i, "Got the wrong value from the hash table"); } hashMap.Dispose(); } [Test] public void NativeHashMap_HashMapSameKey() { using (var hashMap = new NativeHashMap(0, Allocator.Persistent)) { Assert.DoesNotThrow(() => hashMap.Add(0, 0)); Assert.Throws(() => hashMap.Add(0, 0)); } using (var hashMap = new NativeHashMap(0, Allocator.Persistent)) { Assert.IsTrue(hashMap.TryAdd(0, 0)); Assert.IsFalse(hashMap.TryAdd(0, 0)); } } [Test] public void NativeHashMap_IsEmpty() { var container = new NativeHashMap(0, Allocator.Persistent); Assert.IsTrue(container.IsEmpty); container.TryAdd(0, 0); Assert.IsFalse(container.IsEmpty); Assert.AreEqual(1, container.Capacity); ExpectedCount(ref container, 1); container.Remove(0); Assert.IsTrue(container.IsEmpty); container.TryAdd(0, 0); container.Clear(); Assert.IsTrue(container.IsEmpty); container.Dispose(); } [Test] public void NativeHashMap_HashMapEmptyCapacity() { var hashMap = new NativeHashMap(0, Allocator.Persistent); hashMap.TryAdd(0, 0); Assert.AreEqual(1, hashMap.Capacity); ExpectedCount(ref hashMap, 1); hashMap.Dispose(); } [Test] public void NativeHashMap_Remove() { var hashMap = new NativeHashMap(8, Allocator.Temp); int iSquared; // Make sure inserting values work for (int i = 0; i < 8; ++i) Assert.IsTrue(hashMap.TryAdd(i, i * i), "Failed to add value"); Assert.AreEqual(8, hashMap.Capacity, "HashMap grew larger than expected"); // Make sure reading the inserted values work for (int i = 0; i < 8; ++i) { Assert.IsTrue(hashMap.TryGetValue(i, out iSquared), "Failed get value from hash table"); Assert.AreEqual(iSquared, i * i, "Got the wrong value from the hash table"); } for (int rm = 0; rm < 8; ++rm) { Assert.IsTrue(hashMap.Remove(rm)); Assert.IsFalse(hashMap.TryGetValue(rm, out iSquared), "Failed to remove value from hash table"); for (int i = rm + 1; i < 8; ++i) { Assert.IsTrue(hashMap.TryGetValue(i, out iSquared), "Failed get value from hash table"); Assert.AreEqual(iSquared, i * i, "Got the wrong value from the hash table"); } } // Make sure entries were freed for (int i = 0; i < 8; ++i) Assert.IsTrue(hashMap.TryAdd(i, i * i), "Failed to add value"); Assert.AreEqual(8, hashMap.Capacity, "HashMap grew larger than expected"); hashMap.Dispose(); } [Test] public void NativeHashMap_RemoveOnEmptyMap_DoesNotThrow() { var hashMap = new NativeHashMap(0, Allocator.Temp); Assert.DoesNotThrow(() => hashMap.Remove(0)); Assert.DoesNotThrow(() => hashMap.Remove(-425196)); hashMap.Dispose(); } [Test] public void NativeHashMap_TryAddScalability() { var hashMap = new NativeHashMap(1, Allocator.Persistent); for (int i = 0; i != 1000 * 100; i++) { hashMap.TryAdd(i, i); } int value; Assert.IsFalse(hashMap.TryGetValue(-1, out value)); Assert.IsFalse(hashMap.TryGetValue(1000 * 1000, out value)); for (int i = 0; i != 1000 * 100; i++) { Assert.IsTrue(hashMap.TryGetValue(i, out value)); Assert.AreEqual(i, value); } hashMap.Dispose(); } [Test] public void NativeHashMap_GetKeysEmpty() { var hashMap = new NativeHashMap(1, Allocator.Temp); var keys = hashMap.GetKeyArray(Allocator.Temp); hashMap.Dispose(); Assert.AreEqual(0, keys.Length); keys.Dispose(); } [Test] public void NativeHashMap_GetKeys() { var hashMap = new NativeHashMap(1, Allocator.Temp); for (int i = 0; i < 30; ++i) { hashMap.TryAdd(i, 2 * i); } var keys = hashMap.GetKeyArray(Allocator.Temp); hashMap.Dispose(); Assert.AreEqual(30, keys.Length); keys.Sort(); for (int i = 0; i < 30; ++i) { Assert.AreEqual(i, keys[i]); } keys.Dispose(); } [Test] public void NativeHashMap_GetValues() { var hashMap = new NativeHashMap(1, Allocator.Temp); for (int i = 0; i < 30; ++i) { hashMap.TryAdd(i, 2 * i); } var values = hashMap.GetValueArray(Allocator.Temp); hashMap.Dispose(); Assert.AreEqual(30, values.Length); values.Sort(); for (int i = 0; i < 30; ++i) { Assert.AreEqual(2 * i, values[i]); } values.Dispose(); } [Test] public void NativeHashMap_GetKeysAndValues() { var hashMap = new NativeHashMap(1, Allocator.Temp); for (int i = 0; i < 30; ++i) { hashMap.TryAdd(i, 2 * i); } var keysValues = hashMap.GetKeyValueArrays(Allocator.Temp); hashMap.Dispose(); Assert.AreEqual(30, keysValues.Keys.Length); Assert.AreEqual(30, keysValues.Values.Length); // ensure keys and matching values are aligned for (int i = 0; i < 30; ++i) { Assert.AreEqual(2 * keysValues.Keys[i], keysValues.Values[i]); } keysValues.Keys.Sort(); for (int i = 0; i < 30; ++i) { Assert.AreEqual(i, keysValues.Keys[i]); } keysValues.Values.Sort(); for (int i = 0; i < 30; ++i) { Assert.AreEqual(2 * i, keysValues.Values[i]); } keysValues.Dispose(); } public struct TestEntityGuid : IEquatable, IComparable { public ulong a; public ulong b; public bool Equals(TestEntityGuid other) { return a == other.a && b == other.b; } public override int GetHashCode() { unchecked { return (a.GetHashCode() * 397) ^ b.GetHashCode(); } } public int CompareTo(TestEntityGuid other) { var aComparison = a.CompareTo(other.a); if (aComparison != 0) return aComparison; return b.CompareTo(other.b); } } [Test] public void NativeHashMap_GetKeysGuid() { var hashMap = new NativeHashMap(1, Allocator.Temp); for (int i = 0; i < 30; ++i) { var didAdd = hashMap.TryAdd(new TestEntityGuid() { a = (ulong)i * 5, b = 3 * (ulong)i }, 2 * i); Assert.IsTrue(didAdd); } // Validate Hashtable has all the expected values ExpectedCount(ref hashMap, 30); for (int i = 0; i < 30; ++i) { int output; var exists = hashMap.TryGetValue(new TestEntityGuid() { a = (ulong)i * 5, b = 3 * (ulong)i }, out output); Assert.IsTrue(exists); Assert.AreEqual(2 * i, output); } // Validate keys array var keys = hashMap.GetKeyArray(Allocator.Temp); Assert.AreEqual(30, keys.Length); keys.Sort(); for (int i = 0; i < 30; ++i) { Assert.AreEqual(new TestEntityGuid() { a = (ulong)i * 5, b = 3 * (ulong)i }, keys[i]); } hashMap.Dispose(); keys.Dispose(); } [Test] public void NativeHashMap_IndexerWorks() { var hashMap = new NativeHashMap(1, Allocator.Temp); hashMap[5] = 7; Assert.AreEqual(7, hashMap[5]); hashMap[5] = 9; Assert.AreEqual(9, hashMap[5]); hashMap.Dispose(); } [Test] public void NativeHashMap_ContainsKeyHashMap() { var hashMap = new NativeHashMap(1, Allocator.Temp); hashMap[5] = 7; Assert.IsTrue(hashMap.ContainsKey(5)); Assert.IsFalse(hashMap.ContainsKey(6)); hashMap.Dispose(); } #if !UNITY_DOTSRUNTIME // DOTS-Runtime has an assertion in the C++ layer, that can't be caught in C# [Test] public void NativeHashMap_NativeKeyValueArrays_DisposeJob() { var container = new NativeHashMap(1, Allocator.Persistent); Assert.True(container.IsCreated); Assert.DoesNotThrow(() => { container[0] = 0; }); Assert.DoesNotThrow(() => { container[1] = 1; }); Assert.DoesNotThrow(() => { container[2] = 2; }); Assert.DoesNotThrow(() => { container[3] = 3; }); var kv = container.GetKeyValueArrays(Allocator.Persistent); var disposeJob = container.Dispose(default); Assert.False(container.IsCreated); Assert.Throws( () => { container[0] = 2; }); kv.Dispose(disposeJob); disposeJob.Complete(); } #endif // These tests require: // - JobsDebugger support for static safety IDs (added in 2020.1) // - Asserting throws #if !UNITY_DOTSRUNTIME [Test, DotsRuntimeIgnore] public void NativeHashMap_UseAfterFree_UsesCustomOwnerTypeName() { var container = new NativeHashMap(10, CommonRwdAllocator.Handle); container[0] = 123; container.Dispose(); NUnit.Framework.Assert.That(() => container[0], Throws.Exception.TypeOf() .With.Message.Contains($"The {container.GetType()} has been deallocated")); } [BurstCompile(CompileSynchronously = true)] struct NativeHashMap_CreateAndUseAfterFreeBurst : IJob { public void Execute() { var container = new NativeHashMap(10, Allocator.Temp); container[0] = 17; container.Dispose(); container[1] = 42; } } [Test, DotsRuntimeIgnore] public void NativeHashMap_CreateAndUseAfterFreeInBurstJob_UsesCustomOwnerTypeName() { // Make sure this isn't the first container of this type ever created, so that valid static safety data exists var container = new NativeHashMap(10, CommonRwdAllocator.Handle); container.Dispose(); var job = new NativeHashMap_CreateAndUseAfterFreeBurst { }; // Two things: // 1. This exception is logged, not thrown; thus, we use LogAssert to detect it. // 2. Calling write operation after container.Dispose() emits an unintuitive error message. For now, all this test cares about is whether it contains the // expected type name. job.Run(); LogAssert.Expect(LogType.Exception, new Regex($"InvalidOperationException: The {Regex.Escape(container.GetType().ToString())} has been declared as \\[ReadOnly\\] in the job, but you are writing to it")); } #endif [Test] public void NativeHashMap_ForEach_FixedStringInHashMap() { using (var stringList = new NativeList(10, Allocator.Persistent) { "Hello", ",", "World", "!" }) { var seen = new NativeArray(stringList.Length, Allocator.Temp); var container = new NativeHashMap(50, Allocator.Temp); foreach (var str in stringList) { container.Add(str, 0); } foreach (var pair in container) { int index = stringList.IndexOf(pair.Key); Assert.AreEqual(stringList[index], pair.Key.ToString()); seen[index] = seen[index] + 1; } for (int i = 0; i < stringList.Length; i++) { Assert.AreEqual(1, seen[i], $"Incorrect value count {stringList[i]}"); } } } [Test] public void NativeHashMap_EnumeratorDoesNotReturnRemovedElementsTest() { NativeHashMap container = new NativeHashMap(5, Allocator.Temp); for (int i = 0; i < 5; i++) { container.Add(i, i); } int elementToRemove = 2; container.Remove(elementToRemove); using (var enumerator = container.GetEnumerator()) { while (enumerator.MoveNext()) { Assert.AreNotEqual(elementToRemove, enumerator.Current.Key); } } container.Dispose(); } [Test] public void NativeHashMap_EnumeratorInfiniteIterationTest() { NativeHashMap container = new NativeHashMap(5, Allocator.Temp); for (int i = 0; i < 5; i++) { container.Add(i, i); } for (int i = 0; i < 2; i++) { container.Remove(i); } var expected = container.Count(); int count = 0; using (var enumerator = container.GetEnumerator()) { while (enumerator.MoveNext()) { if (count++ > expected) { break; } } } Assert.AreEqual(expected, count); container.Dispose(); } [Test] public void NativeHashMap_ForEach([Values(10, 1000)]int n) { var seen = new NativeArray(n, Allocator.Temp); using (var container = new NativeHashMap(32, CommonRwdAllocator.Handle)) { for (int i = 0; i < n; i++) { container.Add(i, i * 37); } var count = 0; foreach (var kv in container) { int value; Assert.True(container.TryGetValue(kv.Key, out value)); Assert.AreEqual(value, kv.Value); Assert.AreEqual(kv.Key * 37, kv.Value); seen[kv.Key] = seen[kv.Key] + 1; ++count; } Assert.AreEqual(container.Count(), count); for (int i = 0; i < n; i++) { Assert.AreEqual(1, seen[i], $"Incorrect key count {i}"); } } } struct NativeHashMap_ForEach_Job : IJob { [ReadOnly] public NativeHashMap Input; [ReadOnly] public int Num; public void Execute() { var seen = new NativeArray(Num, Allocator.Temp); var count = 0; foreach (var kv in Input) { int value; Assert.True(Input.TryGetValue(kv.Key, out value)); Assert.AreEqual(value, kv.Value); Assert.AreEqual(kv.Key * 37, kv.Value); seen[kv.Key] = seen[kv.Key] + 1; ++count; } Assert.AreEqual(Input.Count(), count); for (int i = 0; i < Num; i++) { Assert.AreEqual(1, seen[i], $"Incorrect key count {i}"); } seen.Dispose(); } } [Test] public void NativeHashMap_ForEach_From_Job([Values(10, 1000)] int n) { var seen = new NativeArray(n, Allocator.Temp); using (var container = new NativeHashMap(32, CommonRwdAllocator.Handle)) { for (int i = 0; i < n; i++) { container.Add(i, i * 37); } new NativeHashMap_ForEach_Job { Input = container, Num = n, }.Run(); } } [Test] public void NativeHashMap_ForEach_Throws_When_Modified() { using (var container = new NativeHashMap(32, CommonRwdAllocator.Handle)) { container.Add(0, 012); container.Add(1, 123); container.Add(2, 234); container.Add(3, 345); container.Add(4, 456); container.Add(5, 567); container.Add(6, 678); container.Add(7, 789); container.Add(8, 890); container.Add(9, 901); Assert.Throws(() => { foreach (var kv in container) { container.Add(10, 10); } }); Assert.Throws(() => { foreach (var kv in container) { container.Remove(1); } }); } } struct NativeHashMap_ForEachIterator : IJob { [ReadOnly] public NativeHashMap.Enumerator Iter; public void Execute() { while (Iter.MoveNext()) { } } } [Test] public void NativeHashMap_ForEach_Throws_Job_Iterator() { using (var container = new NativeHashMap(32, CommonRwdAllocator.Handle)) { var jobHandle = new NativeHashMap_ForEachIterator { Iter = container.GetEnumerator() }.Schedule(); Assert.Throws(() => { container.Add(1, 1); }); jobHandle.Complete(); } } struct ParallelWriteToHashMapJob : IJobParallelFor { [WriteOnly] public NativeHashMap.ParallelWriter Writer; public void Execute(int index) { Writer.TryAdd(index, 0); } } [Test] public void NativeHashMap_ForEach_Throws() { using (var container = new NativeHashMap(32, CommonRwdAllocator.Handle)) { var iter = container.GetEnumerator(); var jobHandle = new ParallelWriteToHashMapJob { Writer = container.AsParallelWriter() }.Schedule(1, 2); Assert.Throws(() => { while (iter.MoveNext()) { } }); jobHandle.Complete(); } } [Test] public unsafe void NativeHashMap_GetUnsafeBucketData() { using (var container = new NativeHashMap(32, CommonRwdAllocator.Handle)) { container.Add(0, 012); container.Add(1, 123); container.Add(2, 234); container.Add(3, 345); container.Add(4, 456); container.Add(5, 567); container.Add(6, 678); container.Add(7, 789); container.Add(8, 890); container.Add(9, 901); var bucketData = container.GetUnsafeBucketData(); var buckets = (int*)bucketData.buckets; var nextPtrs = (int*)bucketData.next; var keys = bucketData.keys; var values = bucketData.values; var other = new NativeHashMap(32, CommonRwdAllocator.Handle); for (int i = 0, count = container.Count(); i < count; i++) { int entryIndex = buckets[i]; while (entryIndex != -1) { var bdKey = UnsafeUtility.ReadArrayElement(keys, entryIndex); var bdValue = UnsafeUtility.ArrayElementAsRef(values, entryIndex); other.Add(bdKey, bdValue); entryIndex = nextPtrs[entryIndex]; } } Assert.AreEqual(container.Count(), other.Count()); var kvArray = container.GetKeyValueArrays(CommonRwdAllocator.Handle); for (int i = 0, count = kvArray.Length; i < count; i++) { int value; Assert.True(other.TryGetValue(kvArray.Keys[i], out value)); Assert.AreEqual(value, kvArray.Values[i]); } kvArray.Dispose(); other.Dispose(); } } [Test] public void NativeHashMap_CustomAllocatorTest() { AllocatorManager.Initialize(); var allocatorHelper = new AllocatorHelper(AllocatorManager.Persistent); ref var allocator = ref allocatorHelper.Allocator; allocator.Initialize(); using (var container = new NativeHashMap(1, allocator.Handle)) { } Assert.IsTrue(allocator.WasUsed); allocator.Dispose(); allocatorHelper.Dispose(); AllocatorManager.Shutdown(); } [BurstCompile] struct BurstedCustomAllocatorJob : IJob { [NativeDisableUnsafePtrRestriction] public unsafe CustomAllocatorTests.CountingAllocator* Allocator; public void Execute() { unsafe { using (var container = new NativeHashMap(1, Allocator->Handle)) { } } } } [Test] public unsafe void NativeHashMap_BurstedCustomAllocatorTest() { AllocatorManager.Initialize(); var allocatorHelper = new AllocatorHelper(AllocatorManager.Persistent); ref var allocator = ref allocatorHelper.Allocator; allocator.Initialize(); var allocatorPtr = (CustomAllocatorTests.CountingAllocator*)UnsafeUtility.AddressOf(ref allocator); unsafe { var handle = new BurstedCustomAllocatorJob {Allocator = allocatorPtr}.Schedule(); handle.Complete(); } Assert.IsTrue(allocator.WasUsed); allocator.Dispose(); allocatorHelper.Dispose(); AllocatorManager.Shutdown(); } }