WuhuIslandTesting/Library/PackageCache/com.unity.collections@1.2.4/Unity.Collections.Tests/NativeListTests.cs

514 lines
15 KiB
C#
Raw Normal View History

2025-01-07 02:06:59 +01:00
using NUnit.Framework;
using System;
using Unity.Burst;
using Unity.Collections;
using Unity.Collections.NotBurstCompatible;
using Unity.Collections.LowLevel.Unsafe;
using Unity.Collections.Tests;
using Unity.Jobs;
using UnityEngine;
using UnityEngine.TestTools;
#if !UNITY_PORTABLE_TEST_RUNNER
using System.Text.RegularExpressions;
#endif
internal class NativeListTests : CollectionsTestFixture
{
static void ExpectedLength<T>(ref NativeList<T> container, int expected)
where T : unmanaged
{
Assert.AreEqual(expected == 0, container.IsEmpty);
Assert.AreEqual(expected, container.Length);
}
[Test]
public void NullListThrow()
{
var list = new NativeList<int>();
Assert.Throws<NullReferenceException>(() => list[0] = 5);
Assert.Throws<ObjectDisposedException>(
() => list.Add(1));
}
[Test]
public void NativeList_Allocate_Deallocate_Read_Write()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Add(1);
list.Add(2);
ExpectedLength(ref list, 2);
Assert.AreEqual(1, list[0]);
Assert.AreEqual(2, list[1]);
list.Dispose();
}
[Test]
public void NativeArrayFromNativeList()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Add(42);
list.Add(2);
NativeArray<int> array = list;
Assert.AreEqual(2, array.Length);
Assert.AreEqual(42, array[0]);
Assert.AreEqual(2, array[1]);
list.Dispose();
}
[Test]
public void NativeArrayFromNativeListInvalidatesOnAdd()
{
var list = new NativeList<int>(Allocator.Persistent);
// This test checks that adding an element without reallocation invalidates the native array
// (length changes)
list.Capacity = 2;
list.Add(42);
NativeArray<int> array = list;
list.Add(1000);
ExpectedLength(ref list, 2);
Assert.Throws<ObjectDisposedException>(
() => { array[0] = 1; });
list.Dispose();
}
[Test]
public void NativeArrayFromNativeListInvalidatesOnCapacityChange()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Add(42);
NativeArray<int> array = list;
ExpectedLength(ref list, 1);
list.Capacity = 10;
//Assert.AreEqual(1, array.Length); - temporarily commenting out updated assert checks to ensure editor version promotion succeeds
Assert.Throws<ObjectDisposedException>(
() => { array[0] = 1; });
list.Dispose();
}
[Test]
public void NativeArrayFromNativeListInvalidatesOnDispose()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Add(42);
NativeArray<int> array = list;
list.Dispose();
Assert.Throws<ObjectDisposedException>(
() => { array[0] = 1; });
Assert.Throws<ObjectDisposedException>(
() => { list[0] = 1; });
}
[Test]
public void NativeArrayFromNativeListMayDeallocate()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Add(42);
NativeArray<int> array = list;
Assert.DoesNotThrow(() => { array.Dispose(); });
list.Dispose();
}
[Test]
public void CopiedNativeListIsKeptInSync()
{
var list = new NativeList<int>(Allocator.Persistent);
var listCpy = list;
list.Add(42);
Assert.AreEqual(42, listCpy[0]);
Assert.AreEqual(42, list[0]);
Assert.AreEqual(1, listCpy.Length);
ExpectedLength(ref list, 1);
list.Dispose();
}
[Test]
public void NativeList_CopyFrom_Managed()
{
var list = new NativeList<float>(4, Allocator.Persistent);
var ar = new float[] { 0, 1, 2, 3, 4, 5, 6, 7 };
list.CopyFromNBC(ar);
ExpectedLength(ref list, 8);
for (int i = 0; i < list.Length; ++i)
{
Assert.AreEqual(i, list[i]);
}
list.Dispose();
}
[Test]
public void NativeList_CopyFrom_Unmanaged()
{
var list = new NativeList<float>(4, Allocator.Persistent);
var ar = new NativeArray<float>(new float[] { 0, 1, 2, 3, 4, 5, 6, 7 }, Allocator.Persistent);
list.CopyFrom(ar);
ExpectedLength(ref list, 8);
for (int i = 0; i < list.Length; ++i)
{
Assert.AreEqual(i, list[i]);
}
ar.Dispose();
list.Dispose();
}
[BurstCompile(CompileSynchronously = true)]
struct TempListInJob : IJob
{
public NativeArray<int> Output;
public void Execute()
{
var list = new NativeList<int>(Allocator.Temp);
list.Add(17);
Output[0] = list[0];
list.Dispose();
}
}
[Test]
[Ignore("Unstable on CI, DOTS-1965")]
public void TempListInBurstJob()
{
var job = new TempListInJob() { Output = CollectionHelper.CreateNativeArray<int>(1, CommonRwdAllocator.Handle) };
job.Schedule().Complete();
Assert.AreEqual(17, job.Output[0]);
job.Output.Dispose();
}
#if ENABLE_UNITY_COLLECTIONS_CHECKS
[Test]
public void SetCapacityLessThanLength()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Resize(10, NativeArrayOptions.UninitializedMemory);
Assert.Throws<ArgumentOutOfRangeException>(() => { list.Capacity = 5; });
list.Dispose();
}
[Test]
public void DisposingNativeListDerivedArrayDoesNotThrow()
{
var list = new NativeList<int>(Allocator.Persistent);
list.Add(1);
NativeArray<int> array = list;
Assert.DoesNotThrow(() => { array.Dispose(); });
list.Dispose();
}
#endif
[Test]
public void NativeList_DisposeJob()
{
var container = new NativeList<int>(Allocator.Persistent);
Assert.True(container.IsCreated);
Assert.DoesNotThrow(() => { container.Add(0); });
Assert.DoesNotThrow(() => { container.Contains(0); });
var disposeJob = container.Dispose(default);
Assert.False(container.IsCreated);
Assert.Throws<ObjectDisposedException>(
() => { container.Contains(0); });
disposeJob.Complete();
}
[Test]
public void ForEachWorks()
{
var container = new NativeList<int>(Allocator.Persistent);
container.Add(10);
container.Add(20);
int sum = 0;
int count = 0;
GCAllocRecorder.ValidateNoGCAllocs(() =>
{
sum = 0;
count = 0;
foreach (var p in container)
{
sum += p;
count++;
}
});
Assert.AreEqual(30, sum);
Assert.AreEqual(2, count);
container.Dispose();
}
struct NativeQueueAddJob : IJob
{
NativeQueue<int> queue;
public NativeQueueAddJob(NativeQueue<int> queue) { this.queue = queue; }
public void Execute()
{
queue.Enqueue(1);
}
}
[Test]
public void NativeQueue_DisposeJobWithMissingDependencyThrows()
{
var queue = new NativeQueue<int>(Allocator.Persistent);
var deps = new NativeQueueAddJob(queue).Schedule();
Assert.Throws<InvalidOperationException>(() => { queue.Dispose(default); });
deps.Complete();
queue.Dispose();
}
[Test]
public void NativeQueue_DisposeJobCantBeScheduled()
{
var queue = new NativeQueue<int>(Allocator.Persistent);
var deps = queue.Dispose(default);
Assert.Throws<InvalidOperationException>(() => { new NativeQueueAddJob(queue).Schedule(deps); });
deps.Complete();
}
// These tests require:
// - JobsDebugger support for static safety IDs (added in 2020.1)
// - Asserting throws
#if !UNITY_DOTSRUNTIME
[Test,DotsRuntimeIgnore]
public void NativeList_UseAfterFree_UsesCustomOwnerTypeName()
{
var list = new NativeList<int>(10, CommonRwdAllocator.Handle);
list.Add(17);
list.Dispose();
Assert.That(() => list[0],
Throws.Exception.TypeOf<ObjectDisposedException>()
.With.Message.Contains($"The {list.GetType()} has been deallocated"));
}
[Test,DotsRuntimeIgnore]
public void AtomicSafetyHandle_AllocatorTemp_UniqueStaticSafetyIds()
{
// All collections that use Allocator.Temp share the same core AtomicSafetyHandle.
// This test verifies that containers can proceed to assign unique static safety IDs to each
// AtomicSafetyHandle value, which will not be shared by other containers using Allocator.Temp.
var listInt = new NativeList<int>(10, Allocator.Temp);
var listFloat = new NativeList<float>(10, Allocator.Temp);
listInt.Add(17);
listInt.Dispose();
Assert.That(() => listInt[0],
Throws.Exception.TypeOf<ObjectDisposedException>()
.With.Message.Contains($"The {listInt.GetType()} has been deallocated"));
listFloat.Add(1.0f);
listFloat.Dispose();
Assert.That(() => listFloat[0],
Throws.Exception.TypeOf<ObjectDisposedException>()
.With.Message.Contains($"The {listFloat.GetType()} has been deallocated"));
}
[BurstCompile(CompileSynchronously = true)]
struct NativeListCreateAndUseAfterFreeBurst : IJob
{
public void Execute()
{
var list = new NativeList<int>(10, Allocator.Temp);
list.Add(17);
list.Dispose();
list.Add(42);
}
}
[Test,DotsRuntimeIgnore]
public void NativeList_CreateAndUseAfterFreeInBurstJob_UsesCustomOwnerTypeName()
{
// Make sure this isn't the first container of this type ever created, so that valid static safety data exists
var list = new NativeList<int>(10, CommonRwdAllocator.Handle);
list.Dispose();
var job = new NativeListCreateAndUseAfterFreeBurst
{
};
// 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(list.GetType().ToString())} has been declared as \\[ReadOnly\\] in the job, but you are writing to it"));
}
#endif
[Test]
public unsafe void NativeList_IndexOf()
{
using (var list = new NativeList<int>(10, Allocator.Persistent) { 123, 789 })
{
bool r0 = false, r1 = false, r2 = false;
GCAllocRecorder.ValidateNoGCAllocs(() =>
{
r0 = -1 != list.IndexOf(456);
r1 = list.Contains(123);
r2 = list.Contains(789);
});
Assert.False(r0);
Assert.True(r1);
Assert.True(r2);
}
}
[Test]
public void NativeList_InsertRangeWithBeginEnd()
{
var list = new NativeList<byte>(3, Allocator.Persistent);
list.Add(0);
list.Add(3);
list.Add(4);
Assert.Throws<ArgumentOutOfRangeException>(() => list.InsertRangeWithBeginEnd(-1, 8));
Assert.Throws<ArgumentOutOfRangeException>(() => list.InsertRangeWithBeginEnd(0, 8));
Assert.Throws<ArgumentException>(() => list.InsertRangeWithBeginEnd(3, 1));
Assert.DoesNotThrow(() => list.InsertRangeWithBeginEnd(1, 3));
list[1] = 1;
list[2] = 2;
for (var i = 0; i < 5; ++i)
{
Assert.AreEqual(i, list[i]);
}
list.Dispose();
}
[Test]
public void NativeList_CustomAllocatorTest()
{
AllocatorManager.Initialize();
var allocatorHelper = new AllocatorHelper<CustomAllocatorTests.CountingAllocator>(AllocatorManager.Persistent);
ref var allocator = ref allocatorHelper.Allocator;
allocator.Initialize();
using (var container = new NativeList<byte>(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 NativeList<byte>(1, Allocator->Handle))
{
}
}
}
}
[Test]
public unsafe void NativeList_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();
}
Assert.IsTrue(allocator.WasUsed);
allocator.Dispose();
allocatorHelper.Dispose();
AllocatorManager.Shutdown();
}
[Test]
public unsafe void NativeList_SetCapacity()
{
using (var list = new NativeList<int>(1, Allocator.Persistent))
{
list.Add(1);
Assert.DoesNotThrow(() => list.SetCapacity(128));
list.Add(1);
Assert.AreEqual(2, list.Length);
Assert.Throws<ArgumentOutOfRangeException>(() => list.SetCapacity(1));
list.RemoveAtSwapBack(0);
Assert.AreEqual(1, list.Length);
Assert.DoesNotThrow(() => list.SetCapacity(1));
list.TrimExcess();
Assert.AreEqual(1, list.Capacity);
}
}
[Test]
public unsafe void NativeList_TrimExcess()
{
using (var list = new NativeList<int>(32, Allocator.Persistent))
{
list.Add(1);
list.TrimExcess();
Assert.AreEqual(1, list.Length);
Assert.AreEqual(1, list.Capacity);
list.RemoveAtSwapBack(0);
Assert.AreEqual(list.Length, 0);
list.TrimExcess();
Assert.AreEqual(list.Capacity, 0);
list.Add(1);
Assert.AreEqual(list.Length, 1);
Assert.AreNotEqual(list.Capacity, 0);
list.Clear();
}
}
}