From b72ec6c5acfa5249d4e50d6831916d96e993c842 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Petryka?= <35800402+MichalPetryka@users.noreply.github.com> Date: Thu, 29 Feb 2024 20:33:58 +0100 Subject: [PATCH] Fix asserts and GC holes with `Unsafe.Subtract/ByteOffset` (#99019) --- src/coreclr/jit/morph.cpp | 6 ++--- .../UnsafeTests.cs | 27 ++++++++++++++++++- 2 files changed, 29 insertions(+), 4 deletions(-) diff --git a/src/coreclr/jit/morph.cpp b/src/coreclr/jit/morph.cpp index 2d7fb7c310265..81394493fa8b0 100644 --- a/src/coreclr/jit/morph.cpp +++ b/src/coreclr/jit/morph.cpp @@ -9220,11 +9220,11 @@ GenTree* Compiler::fgMorphSmpOp(GenTree* tree, MorphAddrContext* mac, bool* optA // TODO #4104: there are a lot of other places where // this condition is not checked before transformations. - if (fgGlobalMorph) + noway_assert(op2); + if (fgGlobalMorph && !op2->TypeIs(TYP_BYREF)) { /* Check for "op1 - cns2" , we change it to "op1 + (-cns2)" */ - noway_assert(op2); if (op2->IsCnsIntOrI() && !op2->IsIconHandle()) { // Negate the constant and change the node to be "+", @@ -9242,7 +9242,7 @@ GenTree* Compiler::fgMorphSmpOp(GenTree* tree, MorphAddrContext* mac, bool* optA noway_assert(op1); if (op1->IsCnsIntOrI()) { - noway_assert(varTypeIsIntOrI(tree)); + noway_assert(varTypeIsIntegralOrI(tree)); // The type of the new GT_NEG node cannot just be op2->TypeGet(). // Otherwise we may sign-extend incorrectly in cases where the GT_NEG diff --git a/src/libraries/System.Runtime/tests/System.Runtime.CompilerServices.Unsafe.Tests/UnsafeTests.cs b/src/libraries/System.Runtime/tests/System.Runtime.CompilerServices.Unsafe.Tests/UnsafeTests.cs index 13488161781df..6eb527f923255 100644 --- a/src/libraries/System.Runtime/tests/System.Runtime.CompilerServices.Unsafe.Tests/UnsafeTests.cs +++ b/src/libraries/System.Runtime/tests/System.Runtime.CompilerServices.Unsafe.Tests/UnsafeTests.cs @@ -462,6 +462,26 @@ public static void ByteOffsetStackByte4() Assert.Equal(new IntPtr(-3), Unsafe.ByteOffset(ref byte4.B3, ref byte4.B0)); } + private static unsafe class StaticReadonlyHolder + { + public static readonly void* Pointer = (void*)RuntimeHelpers.AllocateTypeAssociatedMemory(typeof(StaticReadonlyHolder), 1); + } + + [Fact] + public static unsafe void ByteOffsetConstantRef() + { + // https://github.com/dotnet/runtime/pull/99019 + [MethodImpl(MethodImplOptions.NoInlining)] + static nint NullTest(ref byte origin) => Unsafe.ByteOffset(ref origin, ref Unsafe.NullRef()); + Assert.Equal(0, NullTest(ref Unsafe.NullRef())); + + [MethodImpl(MethodImplOptions.AggressiveInlining)] + static ref byte GetStatic(ref byte x) => ref x; + [MethodImpl(MethodImplOptions.NoInlining)] + static nint StaticReadonlyTest(ref byte x) => Unsafe.ByteOffset(ref GetStatic(ref Unsafe.AsRef(StaticReadonlyHolder.Pointer)), ref x); + Assert.Equal(0, StaticReadonlyTest(ref Unsafe.AsRef(StaticReadonlyHolder.Pointer))); + } + [Fact] public static unsafe void AsRef() { @@ -597,7 +617,7 @@ public static void RefAddNuintByteOffset() } [Fact] - public static void RefSubtract() + public static unsafe void RefSubtract() { string[] a = new string[] { "abc", "def", "ghi", "jkl" }; @@ -609,6 +629,11 @@ public static void RefSubtract() ref string r3 = ref Unsafe.Subtract(ref r2, 3); Assert.Equal("abc", r3); + + // https://github.com/dotnet/runtime/pull/99019 + [MethodImpl(MethodImplOptions.NoInlining)] + static ref byte NullTest(nuint offset) => ref Unsafe.Subtract(ref Unsafe.NullRef(), offset); + Assert.True(Unsafe.IsNullRef(ref NullTest(0))); } [Fact]