Rasagar/Library/PackageCache/com.unity.collections/Unity.Collections.Tests/UnsafeUtilityTests.cs

276 lines
9.0 KiB
C#
Raw Normal View History

2024-08-26 13:07:20 -07:00
using System;
using NUnit.Framework;
using Unity.Collections;
using Unity.Collections.LowLevel.Unsafe;
using Unity.Collections.Tests;
using Unity.Mathematics;
[TestFixture]
internal class UnsafeUtilityTests : CollectionsTestCommonBase
{
#pragma warning disable 649
struct DummyVec
{
public uint A, B, C, D;
}
#pragma warning restore
private NativeArray<T> MakeTestArray<T>(params T[] data)
where T : unmanaged
{
return CollectionHelper.CreateNativeArray<T>(data, CommonRwdAllocator.Handle);
}
[Test]
public void ReinterpretUIntFloat()
{
using (var src = MakeTestArray(1.0f, 2.0f, 3.0f))
{
var dst = src.Reinterpret<float, uint>();
Assert.AreEqual(src.Length, dst.Length);
Assert.AreEqual(0x3f800000u, dst[0]);
Assert.AreEqual(0x40000000u, dst[1]);
Assert.AreEqual(0x40400000u, dst[2]);
}
}
[Test]
public void ReinterpretUInt4Float()
{
using (var src = MakeTestArray(1.0f, 2.0f, 3.0f, -1.0f))
{
var dst = src.Reinterpret<float, DummyVec>();
Assert.AreEqual(1, dst.Length);
var e = dst[0];
Assert.AreEqual(0x3f800000u, e.A);
Assert.AreEqual(0x40000000u, e.B);
Assert.AreEqual(0x40400000u, e.C);
Assert.AreEqual(0xbf800000u, e.D);
}
}
[Test]
public void ReinterpretFloatUint4()
{
var dummies = new DummyVec[]
{
new DummyVec { A = 0x3f800000u, B = 0x40000000u, C = 0x40400000u, D = 0xbf800000u },
new DummyVec { A = 0xbf800000u, B = 0xc0000000u, C = 0xc0400000u, D = 0x3f800000u },
};
using (var src = MakeTestArray(dummies))
{
var dst = src.Reinterpret<DummyVec, float>();
Assert.AreEqual(8, dst.Length);
Assert.AreEqual(1.0f, dst[0]);
Assert.AreEqual(2.0f, dst[1]);
Assert.AreEqual(3.0f, dst[2]);
Assert.AreEqual(-1.0f, dst[3]);
Assert.AreEqual(-1.0f, dst[4]);
Assert.AreEqual(-2.0f, dst[5]);
Assert.AreEqual(-3.0f, dst[6]);
Assert.AreEqual(1.0f, dst[7]);
}
}
[Test]
[TestRequiresDotsDebugOrCollectionChecks]
public void MismatchThrows1()
{
using (var src = MakeTestArray(0.0f, 1.0f, 2.0f))
{
Assert.Throws<InvalidOperationException>(() => src.Reinterpret<float, DummyVec>());
}
}
[Test]
[TestRequiresDotsDebugOrCollectionChecks]
public void MismatchThrows2()
{
using (var src = MakeTestArray(12))
{
Assert.Throws<InvalidOperationException>(() => src.Reinterpret<int, double>());
}
}
[Test]
public void AliasCanBeDisposed()
{
using (var src = MakeTestArray(12))
{
using (var dst = src.Reinterpret<int, float>())
{
}
}
}
[Test]
[TestRequiresCollectionChecks]
public void CannotUseAliasAfterSourceIsDisposed()
{
NativeArray<float> alias;
var src = MakeTestArray(12);
alias = src.Reinterpret<int, float>();
// `Free` of memory allocated by world update allocator is an no-op.
// World update allocator needs to be rewound in order to free the memory.
CommonRwdAllocator.Rewind();
Assert.Throws<ObjectDisposedException>(
() => alias[0] = 1.0f);
}
[Test]
public void MutabilityWorks()
{
using (var src = MakeTestArray(0.0f, -1.0f))
{
var alias = src.Reinterpret<float, uint>();
alias[0] = 0x3f800000;
Assert.AreEqual(1.0f, src[0]);
Assert.AreEqual(-1.0f, src[1]);
}
}
#pragma warning disable 0169 // field is never used
struct AlignOfX
{
float x;
bool y;
}
struct AlignOfY
{
float x;
bool y;
float z;
bool w;
}
struct AlignOfZ
{
float4 x;
bool y;
}
struct AlignOfW
{
float4 x;
bool y;
float4x4 z;
bool w;
}
struct BoolLong
{
bool x;
long y;
}
struct BoolPtr
{
bool x;
unsafe void* y;
}
#pragma warning restore 0169 // field is never used
[Test]
public void UnsafeUtility_AlignOf()
{
Assert.AreEqual(UnsafeUtility.SizeOf<byte>(), UnsafeUtility.AlignOf<byte>());
Assert.AreEqual(UnsafeUtility.SizeOf<short>(), UnsafeUtility.AlignOf<short>());
Assert.AreEqual(UnsafeUtility.SizeOf<ushort>(), UnsafeUtility.AlignOf<ushort>());
Assert.AreEqual(UnsafeUtility.SizeOf<int>(), UnsafeUtility.AlignOf<int>());
Assert.AreEqual(UnsafeUtility.SizeOf<uint>(), UnsafeUtility.AlignOf<uint>());
Assert.AreEqual(UnsafeUtility.SizeOf<long>(), UnsafeUtility.AlignOf<long>());
Assert.AreEqual(UnsafeUtility.SizeOf<ulong>(), UnsafeUtility.AlignOf<ulong>());
Assert.AreEqual(4, UnsafeUtility.AlignOf<float4>());
Assert.AreEqual(4, UnsafeUtility.AlignOf<AlignOfX>());
Assert.AreEqual(4, UnsafeUtility.AlignOf<AlignOfY>());
Assert.AreEqual(4, UnsafeUtility.AlignOf<AlignOfZ>());
Assert.AreEqual(4, UnsafeUtility.AlignOf<AlignOfW>());
Assert.AreEqual(8, UnsafeUtility.AlignOf<BoolLong>());
Assert.AreEqual(UnsafeUtility.SizeOf<IntPtr>(), UnsafeUtility.AlignOf<BoolPtr>());
}
[Test]
public unsafe void UnsafeUtility_MemSwap()
{
using (var array0 = MakeTestArray(0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678))
using (var array1 = MakeTestArray(0x21436587, 0x21436587, 0x21436587, 0x21436587, 0x21436587, 0x21436587))
{
UnsafeUtilityExtensions.MemSwap(NativeArrayUnsafeUtility.GetUnsafePtr(array0), NativeArrayUnsafeUtility.GetUnsafePtr(array1), array0.Length*UnsafeUtility.SizeOf<int>());
foreach (var b in array0) { Assert.AreEqual(0x21436587, b); }
foreach (var b in array1) { Assert.AreEqual(0x12345678, b); }
}
}
[Test]
[TestRequiresCollectionChecks]
public unsafe void UnsafeUtility_MemSwap_DoesThrow_Overlapped()
{
using (var array0 = MakeTestArray(0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678))
{
var mem = (byte*)NativeArrayUnsafeUtility.GetUnsafePtr(array0);
var len = array0.Length * UnsafeUtility.SizeOf<int>();
Assert.DoesNotThrow(() => { UnsafeUtilityExtensions.MemSwap(mem + 10, mem, 10); });
Assert.Throws<InvalidOperationException>(() => { UnsafeUtilityExtensions.MemSwap(mem + 10, mem, len - 10); });
Assert.DoesNotThrow(() => { UnsafeUtilityExtensions.MemSwap(mem, mem + 10, 10); });
Assert.Throws<InvalidOperationException>(() => { UnsafeUtilityExtensions.MemSwap(mem, mem + 10, len - 10); });
}
}
[Test]
[TestRequiresDotsDebugOrCollectionChecks]
public unsafe void UnsafeUtility_ReadArrayElementBoundsChecked_Works()
{
using (var array0 = MakeTestArray(0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678))
{
var mem = (byte*)NativeArrayUnsafeUtility.GetUnsafePtr(array0);
var len = array0.Length;
Assert.DoesNotThrow(() => { UnsafeUtilityExtensions.ReadArrayElementBoundsChecked<int>(mem, 5, len); });
Assert.Throws<IndexOutOfRangeException>(() => { UnsafeUtilityExtensions.ReadArrayElementBoundsChecked<int>(mem, 6, len); });
Assert.Throws<IndexOutOfRangeException>(() => { UnsafeUtilityExtensions.ReadArrayElementBoundsChecked<int>(mem, -1, len); });
}
}
[Test]
[TestRequiresDotsDebugOrCollectionChecks]
public unsafe void UnsafeUtility_WriteArrayElementBoundsChecked_Works()
{
using (var array0 = MakeTestArray(0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678, 0x12345678))
{
var mem = (byte*)NativeArrayUnsafeUtility.GetUnsafePtr(array0);
var len = array0.Length;
Assert.DoesNotThrow(() => { UnsafeUtilityExtensions.WriteArrayElementBoundsChecked(mem, 5, -98765432, len); });
Assert.Throws<IndexOutOfRangeException>(() => { UnsafeUtilityExtensions.WriteArrayElementBoundsChecked(mem, 6, -98765432, len); });
Assert.Throws<IndexOutOfRangeException>(() => { UnsafeUtilityExtensions.WriteArrayElementBoundsChecked(mem, -1, -98765432, len); });
}
}
[Test]
public unsafe void UnsafeUtility_AsRefAddressOfIn_Works()
{
DummyVec thing = default;
void* thingInPtr = UnsafeUtilityExtensions.AddressOf(in thing);
ref DummyVec thingRef = ref UnsafeUtilityExtensions.AsRef(in thing);
void* thingInRefPtr = UnsafeUtility.AddressOf(ref thingRef);
void* thingRefPtr = UnsafeUtility.AddressOf(ref thing);
void* thingPtr = &thing;
Assert.AreEqual((IntPtr) thingPtr, (IntPtr) thingInPtr);
Assert.AreEqual((IntPtr) thingPtr, (IntPtr) thingRefPtr);
Assert.AreEqual((IntPtr) thingPtr, (IntPtr) thingInRefPtr);
}
}