578 lines
16 KiB
C#
578 lines
16 KiB
C#
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<T>(ref NativeHashSet<T> container, int expected)
|
|
where T : unmanaged, IEquatable<T>
|
|
{
|
|
Assert.AreEqual(expected == 0, container.IsEmpty);
|
|
Assert.AreEqual(expected, container.Count());
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_IsEmpty()
|
|
{
|
|
var container = new NativeHashSet<int>(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<int>(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<int>(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<System.InvalidOperationException>(() => { 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<int>(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<int>(16, CommonRwdAllocator.Handle);
|
|
hashMap.Dispose();
|
|
Assert.Throws<ObjectDisposedException>(
|
|
() => { hashMap.Dispose(); });
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_Collisions()
|
|
{
|
|
var container = new NativeHashSet<int>(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<int>(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<int>(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<int>.ParallelWriter Writer;
|
|
|
|
public void Execute(int index)
|
|
{
|
|
Writer.Add(index);
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_CanBeReadFromJob()
|
|
{
|
|
using (var hashSet = new NativeHashSet<int>(1, CommonRwdAllocator.Handle))
|
|
using (var result = new NativeReference<int>(CommonRwdAllocator.Handle))
|
|
{
|
|
hashSet.Add(42);
|
|
new ReadHashSetJob
|
|
{
|
|
Input = hashSet,
|
|
Output = result,
|
|
}.Run();
|
|
Assert.AreEqual(42, result.Value);
|
|
}
|
|
}
|
|
|
|
struct ReadHashSetJob : IJob
|
|
{
|
|
[ReadOnly]
|
|
public NativeHashSet<int> Input;
|
|
|
|
public NativeReference<int> 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<FixedString32Bytes>(10, Allocator.Persistent) { "Hello", ",", "World", "!" })
|
|
{
|
|
var container = new NativeHashSet<FixedString128Bytes>(50, Allocator.Temp);
|
|
var seen = new NativeArray<int>(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<int>(n, Allocator.Temp);
|
|
using (var container = new NativeHashSet<int>(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<int> Input;
|
|
|
|
[ReadOnly]
|
|
public int Num;
|
|
|
|
public void Execute()
|
|
{
|
|
var seen = new NativeArray<int>(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<int>(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<int>(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<ObjectDisposedException>(() =>
|
|
{
|
|
foreach (var item in container)
|
|
{
|
|
container.Add(10);
|
|
}
|
|
});
|
|
|
|
Assert.Throws<ObjectDisposedException>(() =>
|
|
{
|
|
foreach (var item in container)
|
|
{
|
|
container.Remove(1);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_ForEach_Throws()
|
|
{
|
|
using (var container = new NativeHashSet<int>(32, CommonRwdAllocator.Handle))
|
|
{
|
|
var iter = container.GetEnumerator();
|
|
|
|
var jobHandle = new ParallelWriteToHashSetJob
|
|
{
|
|
Writer = container.AsParallelWriter()
|
|
|
|
}.Schedule(1, 2);
|
|
|
|
Assert.Throws<ObjectDisposedException>(() =>
|
|
{
|
|
while (iter.MoveNext())
|
|
{
|
|
}
|
|
});
|
|
|
|
jobHandle.Complete();
|
|
}
|
|
}
|
|
|
|
struct ForEachIterator : IJob
|
|
{
|
|
[ReadOnly]
|
|
public NativeHashSet<int>.Enumerator Iter;
|
|
|
|
public void Execute()
|
|
{
|
|
while (Iter.MoveNext())
|
|
{
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_ForEach_Throws_Job_Iterator()
|
|
{
|
|
using (var container = new NativeHashSet<int>(32, CommonRwdAllocator.Handle))
|
|
{
|
|
var jobHandle = new ForEachIterator
|
|
{
|
|
Iter = container.GetEnumerator()
|
|
|
|
}.Schedule();
|
|
|
|
Assert.Throws<InvalidOperationException>(() => { container.Add(1); });
|
|
|
|
jobHandle.Complete();
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_EIU_ExceptWith_Empty()
|
|
{
|
|
var setA = new NativeHashSet<int>(8, CommonRwdAllocator.Handle) { };
|
|
var setB = new NativeHashSet<int>(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<int>(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 };
|
|
var setB = new NativeHashSet<int>(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<int>(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 };
|
|
var setB = new NativeHashSet<int>(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<int>(8, CommonRwdAllocator.Handle) { };
|
|
var setB = new NativeHashSet<int>(8, CommonRwdAllocator.Handle) { };
|
|
setA.IntersectWith(setB);
|
|
|
|
ExpectedCount(ref setA, 0);
|
|
|
|
setA.Dispose();
|
|
setB.Dispose();
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_EIU_IntersectWith()
|
|
{
|
|
var setA = new NativeHashSet<int>(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 };
|
|
var setB = new NativeHashSet<int>(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<int>(8, CommonRwdAllocator.Handle) { };
|
|
var setB = new NativeHashSet<int>(8, CommonRwdAllocator.Handle) { };
|
|
setA.UnionWith(setB);
|
|
|
|
ExpectedCount(ref setA, 0);
|
|
|
|
setA.Dispose();
|
|
setB.Dispose();
|
|
}
|
|
|
|
[Test]
|
|
public void NativeHashSet_EIU_UnionWith()
|
|
{
|
|
var setA = new NativeHashSet<int>(8, CommonRwdAllocator.Handle) { 0, 1, 2, 3, 4, 5 };
|
|
var setB = new NativeHashSet<int>(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<int>(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<CustomAllocatorTests.CountingAllocator>(AllocatorManager.Persistent);
|
|
ref var allocator = ref allocatorHelper.Allocator;
|
|
allocator.Initialize();
|
|
|
|
using (var container = new NativeHashSet<int>(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<int>(1, Allocator->Handle))
|
|
{
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public unsafe void NativeHashSet_BurstedCustomAllocatorTest()
|
|
{
|
|
AllocatorManager.Initialize();
|
|
var allocatorHelper = new AllocatorHelper<CustomAllocatorTests.CountingAllocator>(AllocatorManager.Persistent);
|
|
ref var allocator = ref allocatorHelper.Allocator;
|
|
allocator.Initialize();
|
|
|
|
var allocatorPtr = (CustomAllocatorTests.CountingAllocator*)UnsafeUtility.AddressOf<CustomAllocatorTests.CountingAllocator>(ref allocator);
|
|
unsafe
|
|
{
|
|
var handle = new BurstedCustomAllocatorJob {Allocator = allocatorPtr }.Schedule();
|
|
handle.Complete();
|
|
}
|
|
|
|
FastAssert.IsTrue(allocator.WasUsed);
|
|
allocator.Dispose();
|
|
allocatorHelper.Dispose();
|
|
AllocatorManager.Shutdown();
|
|
}
|
|
}
|