using System; using System.Reflection; using System.Runtime.CompilerServices; [assembly: AssemblyFileVersion("4.0.0.0")] [assembly: AssemblyInformationalVersion("4.0.0.0")] [assembly: AssemblyTitle("System.Runtime.CompilerServices.Unsafe")] [assembly: AssemblyDescription("System.Runtime.CompilerServices.Unsafe")] [assembly: AssemblyMetadata(".NETFrameworkAssembly", "")] [assembly: AssemblyMetadata("Serviceable", "True")] [assembly: AssemblyCopyright("© Microsoft Corporation. All rights reserved.")] [assembly: AssemblyCompany("Microsoft Corporation")] [assembly: AssemblyProduct("Microsoft® .NET Framework")] [assembly: CLSCompliant(false)] internal sealed class ExtraUnsafeTests { public unsafe static void PinWithTypeMismatch(ref uint managedPtr) { fixed (ushort* ptr = &Unsafe.As(ref managedPtr)) { } } public unsafe static uint* RefToPointerWithoutPinning(ref uint managedPtr) { return (uint*)Unsafe.AsPointer(ref managedPtr); } public static ref ulong RefAssignTypeMismatch(ref uint a, ref uint b) { ref ushort reference = ref Unsafe.As(ref a); if (a != 0) { reference = ref Unsafe.As(ref b); } Console.WriteLine(reference); return ref Unsafe.As(ref reference); } public unsafe static byte[] Issue1292(int val, byte[] arr) { //The blocks IL_0019 are reachable both inside and outside the pinned region starting at IL_0013. ILSpy has duplicated these blocks in order to place them both within and outside the `fixed` statement. byte[] array; if ((array = arr) != null && array.Length != 0) { fixed (byte* ptr = &array[0]) { *(int*)ptr = val; } } else { /*pinned*/ref byte reference = ref *(byte*)null; *(int*)Unsafe.AsPointer(ref reference) = val; } return arr; } public unsafe void pin_ptr_test(int[] a, int[] b) { //The blocks IL_0016 are reachable both inside and outside the pinned region starting at IL_0007. ILSpy has duplicated these blocks in order to place them both within and outside the `fixed` statement. ref int reference; fixed (int* ptr = &a[0]) { if (*ptr <= 0) { ptr[4 * 0] = 1; return; } reference = ref *ptr; } fixed (int* ptr = &b[reference]) { ptr[4 * 0] = 1; } } } namespace System.Runtime.CompilerServices { public static class Unsafe { [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static T Read(void* source) { return *(T*)source; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static T ReadUnaligned(void* source) { return *(T*)source; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static T ReadUnaligned(ref byte source) { return *(T*)(&source); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void Write(void* destination, T value) { *(T*)destination = value; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void WriteUnaligned(void* destination, T value) { *(T*)destination = value; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void WriteUnaligned(ref byte destination, T value) { *(T*)(&destination) = value; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void Copy(void* destination, ref T source) { *(T*)destination = source; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void Copy(ref T destination, void* source) { destination = *(T*)source; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void* AsPointer(ref T value) { return Unsafe.AsPointer(ref value); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static int SizeOf() { return sizeof(T); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void CopyBlock(void* destination, void* source, uint byteCount) { // IL cpblk instruction Unsafe.CopyBlock(destination, source, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void CopyBlock(ref byte destination, ref byte source, uint byteCount) { // IL cpblk instruction Unsafe.CopyBlock(ref destination, ref source, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void CopyBlockUnaligned(void* destination, void* source, uint byteCount) { // IL cpblk instruction Unsafe.CopyBlockUnaligned(destination, source, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void CopyBlockUnaligned(ref byte destination, ref byte source, uint byteCount) { // IL cpblk instruction Unsafe.CopyBlockUnaligned(ref destination, ref source, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void InitBlock(void* startAddress, byte value, uint byteCount) { // IL initblk instruction Unsafe.InitBlock(startAddress, value, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void InitBlock(ref byte startAddress, byte value, uint byteCount) { // IL initblk instruction Unsafe.InitBlock(ref startAddress, value, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void InitBlockUnaligned(void* startAddress, byte value, uint byteCount) { // IL initblk instruction Unsafe.InitBlockUnaligned(startAddress, value, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static void InitBlockUnaligned(ref byte startAddress, byte value, uint byteCount) { // IL initblk instruction Unsafe.InitBlockUnaligned(ref startAddress, value, byteCount); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static T As(object o) where T : class { return (T)o; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static ref T AsRef(void* source) { return ref *(T*)source; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T AsRef(in T source) { return ref source; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref TTo As(ref TFrom source) { return ref Unsafe.As(ref source); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T Unbox(object box) where T : struct { return ref (T)box; } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T Add(ref T source, int elementOffset) { return ref Unsafe.Add(ref source, elementOffset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void* Add(void* source, int elementOffset) { return (byte*)source + (nint)elementOffset * (nint)sizeof(T); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T Add(ref T source, IntPtr elementOffset) { return ref Unsafe.Add(ref source, elementOffset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T AddByteOffset(ref T source, IntPtr byteOffset) { return ref Unsafe.AddByteOffset(ref source, byteOffset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T Subtract(ref T source, int elementOffset) { return ref Unsafe.Subtract(ref source, elementOffset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public unsafe static void* Subtract(void* source, int elementOffset) { return (byte*)source - (nint)elementOffset * (nint)sizeof(T); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T Subtract(ref T source, IntPtr elementOffset) { return ref Unsafe.Subtract(ref source, elementOffset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static ref T SubtractByteOffset(ref T source, IntPtr byteOffset) { return ref Unsafe.SubtractByteOffset(ref source, byteOffset); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static IntPtr ByteOffset(ref T origin, ref T target) { return Unsafe.ByteOffset(target: ref target, origin: ref origin); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static bool AreSame(ref T left, ref T right) { return (ref left) == (ref right); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static bool IsAddressGreaterThan(ref T left, ref T right) { return (ref left) > (ref right); } [MethodImpl(MethodImplOptions.AggressiveInlining)] public static bool IsAddressLessThan(ref T left, ref T right) { return (ref left) < (ref right); } } }