using System; using NUnit.Framework; using Unity.Burst; using Unity.Collections; using Unity.Collections.LowLevel.Unsafe; using Unity.Collections.NotBurstCompatible; using Unity.Jobs; using Unity.Collections.Tests; using UnityEngine; using UnityEngine.TestTools; #if !UNITY_PORTABLE_TEST_RUNNER using System.Text.RegularExpressions; #endif internal class NativeHashSetTests: CollectionsTestFixture { static void ExpectedCount(ref NativeHashSet container, int expected) where T : unmanaged, IEquatable { Assert.AreEqual(expected == 0, container.IsEmpty); Assert.AreEqual(expected, container.Count()); } [Test] public void NativeHashSet_IsEmpty() { var container = new NativeHashSet(0, Allocator.Persistent); Assert.IsTrue(container.IsEmpty); Assert.IsTrue(container.Add(0)); Assert.IsFalse(container.IsEmpty); Assert.AreEqual(1, container.Capacity); ExpectedCount(ref container, 1); container.Remove(0); Assert.IsTrue(container.IsEmpty); Assert.IsTrue(container.Add(0)); container.Clear(); Assert.IsTrue(container.IsEmpty); container.Dispose(); } [Test] public void UnsafeHashSet_Capacity() { var container = new NativeHashSet(0, Allocator.Persistent); Assert.IsTrue(container.IsEmpty); Assert.AreEqual(0, container.Capacity); container.Capacity = 10; Assert.AreEqual(10, container.Capacity); container.Dispose(); } #if !UNITY_DOTSRUNTIME // DOTS-Runtime has an assertion in the C++ layer, that can't be caught in C# [Test] public void NativeHashSet_Full_Throws() { var container = new NativeHashSet(16, Allocator.Temp); ExpectedCount(ref container, 0); for (int i = 0, capacity = container.Capacity; i < capacity; ++i) { Assert.DoesNotThrow(() => { container.Add(i); }); } ExpectedCount(ref container, container.Capacity); // Make sure overallocating throws and exception if using the Concurrent version - normal hash map would grow var writer = container.AsParallelWriter(); Assert.Throws(() => { writer.Add(100); }); ExpectedCount(ref container, container.Capacity); container.Clear(); ExpectedCount(ref container, 0); container.Dispose(); } #endif [Test] public void NativeHashSet_RemoveOnEmptyMap_DoesNotThrow() { var container = new NativeHashSet(0, Allocator.Temp); Assert.DoesNotThrow(() => container.Remove(0)); Assert.DoesNotThrow(() => container.Remove(-425196)); container.Dispose(); } [Test] public void NativeHashSet_Double_Deallocate_Throws() { var hashMap = new NativeHashSet(16, CommonRwdAllocator.Handle); hashMap.Dispose(); Assert.Throws( () => { hashMap.Dispose(); }); } [Test] public void NativeHashSet_Collisions() { var container = new NativeHashSet(16, Allocator.Temp); Assert.IsFalse(container.Contains(0), "Contains on empty hash map did not fail"); ExpectedCount(ref container, 0); // Make sure inserting values work for (int i = 0; i < 8; ++i) { Assert.IsTrue(container.Add(i), "Failed to add value"); } ExpectedCount(ref container, 8); // The bucket size is capacity * 2, adding that number should result in hash collisions for (int i = 0; i < 8; ++i) { Assert.IsTrue(container.Add(i + 32), "Failed to add value with potential hash collision"); } // Make sure reading the inserted values work for (int i = 0; i < 8; ++i) { Assert.IsTrue(container.Contains(i), "Failed get value from hash set"); } for (int i = 0; i < 8; ++i) { Assert.IsTrue(container.Contains(i + 32), "Failed get value from hash set"); } container.Dispose(); } [Test] public void NativeHashSet_SameElement() { using (var container = new NativeHashSet(0, Allocator.Persistent)) { Assert.IsTrue(container.Add(0)); Assert.IsFalse(container.Add(0)); } } [Test] public void NativeHashSet_ParallelWriter_CanBeUsedInJob() { const int count = 32; using (var hashSet = new NativeHashSet(count, CommonRwdAllocator.Handle)) { new ParallelWriteToHashSetJob { Writer = hashSet.AsParallelWriter() }.Schedule(count, 2).Complete(); var result = hashSet.ToNativeArray(Allocator.Temp); result.Sort(); for (int i = 0; i < count; i++) Assert.AreEqual(i, result[i]); } } struct ParallelWriteToHashSetJob : IJobParallelFor { [WriteOnly] public NativeHashSet.ParallelWriter Writer; public void Execute(int index) { Writer.Add(index); } } [Test] public void NativeHashSet_CanBeReadFromJob() { using (var hashSet = new NativeHashSet(1, CommonRwdAllocator.Handle)) using (var result = new NativeReference(CommonRwdAllocator.Handle)) { hashSet.Add(42); new ReadHashSetJob { Input = hashSet, Output = result, }.Run(); Assert.AreEqual(42, result.Value); } } struct ReadHashSetJob : IJob { [ReadOnly] public NativeHashSet Input; public NativeReference Output; public void Execute() { Output.Value = Input.ToNativeArray(Allocator.Temp)[0]; foreach (var value in Input) { Assert.AreEqual(42, value); } } } [Test] public void NativeHashSet_ForEach_FixedStringInHashMap() { using (var stringList = new NativeList(10, Allocator.Persistent) { "Hello", ",", "World", "!" }) { var container = new NativeHashSet(50, Allocator.Temp); var seen = new NativeArray(stringList.Length, Allocator.Temp); foreach (var str in stringList) { container.Add(str); } foreach (var value in container) { int index = stringList.IndexOf(value); Assert.AreEqual(stringList[index], value.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 NativeHashSet_ForEach([Values(10, 1000)]int n) { var seen = new NativeArray(n, Allocator.Temp); using (var container = new NativeHashSet(32, CommonRwdAllocator.Handle)) { for (int i = 0; i < n; i++) { container.Add(i); } var count = 0; foreach (var item in container) { Assert.True(container.Contains(item)); seen[item] = seen[item] + 1; ++count; } Assert.AreEqual(container.Count(), count); for (int i = 0; i < n; i++) { Assert.AreEqual(1, seen[i], $"Incorrect item count {i}"); } } } struct NativeHashSet_ForEach_Job : IJob { [ReadOnly] public NativeHashSet Input; [ReadOnly] public int Num; public void Execute() { var seen = new NativeArray(Num, Allocator.Temp); var count = 0; foreach (var item in Input) { Assert.True(Input.Contains(item)); seen[item] = seen[item] + 1; ++count; } Assert.AreEqual(Input.Count(), count); for (int i = 0; i < Num; i++) { Assert.AreEqual(1, seen[i], $"Incorrect item count {i}"); } seen.Dispose(); } } [Test] public void NativeHashSet_ForEach_From_Job([Values(10, 1000)] int n) { using (var container = new NativeHashSet(32, CommonRwdAllocator.Handle)) { for (int i = 0; i < n; i++) { container.Add(i); } new NativeHashSet_ForEach_Job { Input = container, Num = n, }.Run(); } } [Test] public void NativeHashSet_ForEach_Throws_When_Modified() { using (var container = new NativeHashSet(32, CommonRwdAllocator.Handle)) { container.Add(0); container.Add(1); container.Add(2); container.Add(3); container.Add(4); container.Add(5); container.Add(6); container.Add(7); container.Add(8); container.Add(9); Assert.Throws(() => { foreach (var item in container) { container.Add(10); } }); Assert.Throws(() => { foreach (var item in container) { container.Remove(1); } }); } } [Test] public void NativeHashSet_ForEach_Throws() { using (var container = new NativeHashSet(32, CommonRwdAllocator.Handle)) { var iter = container.GetEnumerator(); var jobHandle = new ParallelWriteToHashSetJob { Writer = container.AsParallelWriter() }.Schedule(1, 2); Assert.Throws(() => { while (iter.MoveNext()) { } }); jobHandle.Complete(); } } struct ForEachIterator : IJob { [ReadOnly] public NativeHashSet.Enumerator Iter; public void Execute() { while (Iter.MoveNext()) { } } } [Test] public void NativeHashSet_ForEach_Throws_Job_Iterator() { using (var container = new NativeHashSet(32, CommonRwdAllocator.Handle)) { var jobHandle = new ForEachIterator { Iter = container.GetEnumerator() }.Schedule(); Assert.Throws(() => { container.Add(1); }); jobHandle.Complete(); } } [Test] public void NativeHashSet_EIU_ExceptWith_Empty() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { }; setA.ExceptWith(setB); ExpectedCount(ref setA, 0); setA.Dispose(); setB.Dispose(); } [Test] public void NativeHashSet_EIU_ExceptWith_AxB() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { 3, 4, 5, 6, 7, 8 }; setA.ExceptWith(setB); ExpectedCount(ref setA, 3); Assert.True(setA.Contains(0)); Assert.True(setA.Contains(1)); Assert.True(setA.Contains(2)); setA.Dispose(); setB.Dispose(); } [Test] public void NativeHashSet_EIU_ExceptWith_BxA() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { 3, 4, 5, 6, 7, 8 }; setB.ExceptWith(setA); ExpectedCount(ref setB, 3); Assert.True(setB.Contains(6)); Assert.True(setB.Contains(7)); Assert.True(setB.Contains(8)); setA.Dispose(); setB.Dispose(); } [Test] public void NativeHashSet_EIU_IntersectWith_Empty() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { }; setA.IntersectWith(setB); ExpectedCount(ref setA, 0); setA.Dispose(); setB.Dispose(); } [Test] public void NativeHashSet_EIU_IntersectWith() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { 3, 4, 5, 6, 7, 8 }; setA.IntersectWith(setB); ExpectedCount(ref setA, 3); Assert.True(setA.Contains(3)); Assert.True(setA.Contains(4)); Assert.True(setA.Contains(5)); setA.Dispose(); setB.Dispose(); } [Test] public void NativeHashSet_EIU_UnionWith_Empty() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { }; setA.UnionWith(setB); ExpectedCount(ref setA, 0); setA.Dispose(); setB.Dispose(); } [Test] public void NativeHashSet_EIU_UnionWith() { var setA = new NativeHashSet(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 }; var setB = new NativeHashSet(8, CommonRwdAllocator.Handle) { 3, 4, 5, 6, 7, 8 }; setA.UnionWith(setB); ExpectedCount(ref setA, 9); Assert.True(setA.Contains(0)); Assert.True(setA.Contains(1)); Assert.True(setA.Contains(2)); Assert.True(setA.Contains(3)); Assert.True(setA.Contains(4)); Assert.True(setA.Contains(5)); Assert.True(setA.Contains(6)); Assert.True(setA.Contains(7)); Assert.True(setA.Contains(8)); setA.Dispose(); setB.Dispose(); } #if !NET_DOTS // Array.Sort is not supported [Test] public void NativeHashSet_ToArray() { using (var set = new NativeHashSet(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 }) { var array = set.ToArray(); Array.Sort(array); for (int i = 0, num = set.Count(); i < num; i++) { Assert.AreEqual(array[i], i); } } } #endif [Test] public void NativeHashSet_CustomAllocatorTest() { AllocatorManager.Initialize(); var allocatorHelper = new AllocatorHelper(AllocatorManager.Persistent); ref var allocator = ref allocatorHelper.Allocator; allocator.Initialize(); using (var container = new NativeHashSet(1, allocator.Handle)) { } FastAssert.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 NativeHashSet(1, Allocator->Handle)) { } } } } [Test] public unsafe void NativeHashSet_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(); } FastAssert.IsTrue(allocator.WasUsed); allocator.Dispose(); allocatorHelper.Dispose(); AllocatorManager.Shutdown(); } }