From eb123a7f77a9e453b2a67e99525598ece83bcabe Mon Sep 17 00:00:00 2001 From: SingleAccretion <62474226+SingleAccretion@users.noreply.github.com> Date: Wed, 6 Jul 2022 01:32:06 +0300 Subject: [PATCH] Do not spill "mis-sized" struct args passed on stack (#71399) * Enable a test on Unix x64 * Handle mis-sized structs in LA PUTARG_STK codegen * Handle mis-sized structs in LA PUTARG_SPLIT codegen * MisSizedStructs_ArmSplit -> MisSizedStructs_ArmArch * Add tests * Do not spill mis-sized stack args All backends support them. --- src/coreclr/jit/codegenloongarch64.cpp | 73 ++-- src/coreclr/jit/morph.cpp | 2 +- .../StructABI/MisSizedStructs_ArmArch.cs | 331 ++++++++++++++++++ ....csproj => MisSizedStructs_ArmArch.csproj} | 0 .../StructABI/MisSizedStructs_ArmSplit.cs | 124 ------- 5 files changed, 376 insertions(+), 154 deletions(-) create mode 100644 src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmArch.cs rename src/tests/JIT/Directed/StructABI/{MisSizedStructs_ArmSplit.csproj => MisSizedStructs_ArmArch.csproj} (100%) delete mode 100644 src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmSplit.cs diff --git a/src/coreclr/jit/codegenloongarch64.cpp b/src/coreclr/jit/codegenloongarch64.cpp index 1443e43e7ea97..a1423cd28e298 100644 --- a/src/coreclr/jit/codegenloongarch64.cpp +++ b/src/coreclr/jit/codegenloongarch64.cpp @@ -6015,8 +6015,9 @@ void CodeGen::genPutArgStk(GenTreePutArgStk* treeNode) while (remainingSize > 0) { - var_types type; + nextIndex = structOffset / TARGET_POINTER_SIZE; + var_types type; if (remainingSize >= TARGET_POINTER_SIZE) { type = layout->GetGCPtrType(nextIndex); @@ -6026,20 +6027,21 @@ void CodeGen::genPutArgStk(GenTreePutArgStk* treeNode) // the left over size is smaller than a pointer and thus can never be a GC type assert(!layout->IsGCPtr(nextIndex)); - if (remainingSize == 1) + if (remainingSize >= 4) { - type = TYP_UBYTE; + type = TYP_INT; } - else if (remainingSize == 2) + else if (remainingSize >= 2) { type = TYP_USHORT; } else { - assert(remainingSize == 4); - type = TYP_UINT; + assert(remainingSize == 1); + type = TYP_UBYTE; } } + const emitAttr attr = emitTypeSize(type); const unsigned moveSize = genTypeSize(type); assert(EA_SIZE_IN_BYTES(attr) == moveSize); @@ -6066,7 +6068,6 @@ void CodeGen::genPutArgStk(GenTreePutArgStk* treeNode) assert(argOffsetOut <= argOffsetMax); // We can't write beyond the outgoing arg area structOffset += moveSize; - nextIndex++; } } } @@ -6225,11 +6226,7 @@ void CodeGen::genPutArgSplit(GenTreePutArgSplit* treeNode) } else // addrNode is used { - assert(addrNode != nullptr); - // TODO-Cleanup: `Lowering::NewPutArg` marks only `LCL_VAR_ADDR` as contained nowadays, - // Generate code to load the address that we need into a register - genConsumeAddress(addrNode); - addrReg = addrNode->GetRegNum(); + addrReg = genConsumeReg(addrNode); // If addrReg equal to baseReg, we use the last target register as alternative baseReg. // Because the candidate mask for the internal baseReg does not include any of the target register, @@ -6243,21 +6240,40 @@ void CodeGen::genPutArgSplit(GenTreePutArgSplit* treeNode) ClassLayout* layout = source->AsObj()->GetLayout(); // Put on stack first - unsigned nextIndex = treeNode->gtNumRegs; - unsigned structOffset = nextIndex * TARGET_POINTER_SIZE; - int remainingSize = treeNode->GetStackByteSize(); + unsigned structOffset = treeNode->gtNumRegs * TARGET_POINTER_SIZE; + unsigned remainingSize = layout->GetSize() - structOffset; unsigned argOffsetOut = treeNode->getArgOffset(); - // remainingSize is always multiple of TARGET_POINTER_SIZE - assert(remainingSize % TARGET_POINTER_SIZE == 0); + assert((remainingSize > 0) && (roundUp(remainingSize, TARGET_POINTER_SIZE) == treeNode->GetStackByteSize())); while (remainingSize > 0) { - var_types type = layout->GetGCPtrType(nextIndex); + var_types type; + if (remainingSize >= TARGET_POINTER_SIZE) + { + type = layout->GetGCPtrType(structOffset / TARGET_POINTER_SIZE); + } + else if (remainingSize >= 4) + { + type = TYP_INT; + } + else if (remainingSize >= 2) + { + type = TYP_USHORT; + } + else + { + assert(remainingSize == 1); + type = TYP_UBYTE; + } + emitAttr attr = emitActualTypeSize(type); + unsigned moveSize = genTypeSize(type); + + instruction loadIns = ins_Load(type); if (varNode != nullptr) { - // Load from our varNumImp source - emit->emitIns_R_S(INS_ld_d, emitTypeSize(type), baseReg, srcVarNum, structOffset); + // Load from our local source + emit->emitIns_R_S(loadIns, attr, baseReg, srcVarNum, structOffset); } else { @@ -6265,17 +6281,16 @@ void CodeGen::genPutArgSplit(GenTreePutArgSplit* treeNode) assert(baseReg != addrReg); // Load from our address expression source - emit->emitIns_R_R_I(INS_ld_d, emitTypeSize(type), baseReg, addrReg, structOffset); + emit->emitIns_R_R_I(loadIns, attr, baseReg, addrReg, structOffset); } - // Emit str instruction to store the register into the outgoing argument area - emit->emitIns_S_R(INS_st_d, emitTypeSize(type), baseReg, varNumOut, argOffsetOut); + // Emit the instruction to store the register into the outgoing argument area + emit->emitIns_S_R(ins_Store(type), attr, baseReg, varNumOut, argOffsetOut); + argOffsetOut += moveSize; + assert(argOffsetOut <= argOffsetMax); - argOffsetOut += TARGET_POINTER_SIZE; // We stored 4-bytes of the struct - assert(argOffsetOut <= argOffsetMax); // We can't write beyond the outgoing arg area - remainingSize -= TARGET_POINTER_SIZE; // We loaded 4-bytes of the struct - structOffset += TARGET_POINTER_SIZE; - nextIndex += 1; + remainingSize -= moveSize; + structOffset += moveSize; } // We set up the registers in order, so that we assign the last target register `baseReg` is no longer in use, @@ -6288,7 +6303,7 @@ void CodeGen::genPutArgSplit(GenTreePutArgSplit* treeNode) if (varNode != nullptr) { - // Load from our varNumImp source + // Load from our local source emit->emitIns_R_S(ins_Load(type), emitTypeSize(type), targetReg, srcVarNum, structOffset); } else diff --git a/src/coreclr/jit/morph.cpp b/src/coreclr/jit/morph.cpp index 1c3394e7d68b3..808c768f65a49 100644 --- a/src/coreclr/jit/morph.cpp +++ b/src/coreclr/jit/morph.cpp @@ -1022,7 +1022,7 @@ void CallArgs::ArgsComplete(Compiler* comp, GenTreeCall* call) // TODO-Arm: This optimization is not implemented for ARM32 // so we skip this for ARM32 until it is ported to use RyuJIT backend // - if (argx->OperGet() == GT_OBJ) + if (argx->OperIs(GT_OBJ) && (arg.AbiInfo.GetRegNum() != REG_STK)) { GenTreeObj* argObj = argx->AsObj(); unsigned structSize = argObj->GetLayout()->GetSize(); diff --git a/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmArch.cs b/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmArch.cs new file mode 100644 index 0000000000000..6cd2367b32728 --- /dev/null +++ b/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmArch.cs @@ -0,0 +1,331 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Runtime.InteropServices; +using System.Runtime.CompilerServices; + +public unsafe class MisSizedStructs_ArmSplit +{ + public const byte ByteValue = 0xC1; + + public static int Main() + { + if (ProblemWithOutOfBoundsLoads(out int result)) + { + return result; + } + + return 100; + } + + // Test that we do not load of bounds for split arguments on ARM. + // + static bool ProblemWithOutOfBoundsLoads(out int result) + { + result = 100; + + if (!OperatingSystem.IsLinux()) + { + return false; + } + + const int PROT_NONE = 0x0; + const int PROT_READ = 0x1; + const int PROT_WRITE = 0x2; + const int MAP_PRIVATE = 0x02; + const int MAP_ANONYMOUS = 0x20; + const int PAGE_SIZE = 0x1000; + + byte* pages = (byte*)mmap(null, 2 * PAGE_SIZE, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); + + if (pages == (byte*)-1) + { + Console.WriteLine("Failed to allocate two pages, errno is {0}, giving up on the test", Marshal.GetLastSystemError()); + return false; + } + + if (mprotect(pages + PAGE_SIZE, PAGE_SIZE, PROT_NONE) != 0) + { + Console.WriteLine("Failed to protect the second page, errno is {0}, giving up on the test", Marshal.GetLastSystemError()); + munmap(pages, 2 * PAGE_SIZE); + return false; + } + + pages[PAGE_SIZE - 1] = ByteValue; + + // Split args on ARM. + // + if (CallForSplitStructWithSixteenBytes_Arm(0, *(StructWithSixteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithSixteenBytes))) != ByteValue) + { + result = 216; + return true; + } + if (CallForSplitStructWithSeventeenBytes_Arm(0, *(StructWithSeventeenBytes*)(pages + PAGE_SIZE - sizeof(StructWithSeventeenBytes))) != ByteValue) + { + result = 217; + return true; + } + if (CallForSplitStructWithEighteenBytes_Arm(0, *(StructWithEighteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithEighteenBytes))) != ByteValue) + { + result = 218; + return true; + } + if (CallForSplitStructWithNineteenBytes_Arm(0, *(StructWithNineteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithNineteenBytes))) != ByteValue) + { + result = 219; + return true; + } + + // Stack args on ARM64. + // + if (CallForStkStructWithOneByte_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithOneByte*)(pages + PAGE_SIZE - sizeof(StructWithOneByte))) != ByteValue) + { + result = 301; + return true; + } + if (CallForStkStructWithTwoBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithTwoBytes*)(pages + PAGE_SIZE - sizeof(StructWithTwoBytes))) != ByteValue) + { + result = 302; + return true; + } + if (CallForStkStructWithThreeBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithThreeBytes*)(pages + PAGE_SIZE - sizeof(StructWithThreeBytes))) != ByteValue) + { + result = 303; + return true; + } + if (CallForStkStructWithFourBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithFourBytes*)(pages + PAGE_SIZE - sizeof(StructWithFourBytes))) != ByteValue) + { + result = 304; + return true; + } + if (CallForStkStructWithFiveBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithFiveBytes*)(pages + PAGE_SIZE - sizeof(StructWithFiveBytes))) != ByteValue) + { + result = 305; + return true; + } + if (CallForStkStructWithSixBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithSixBytes*)(pages + PAGE_SIZE - sizeof(StructWithSixBytes))) != ByteValue) + { + result = 306; + return true; + } + if (CallForStkStructWithSevenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithSevenBytes*)(pages + PAGE_SIZE - sizeof(StructWithSevenBytes))) != ByteValue) + { + result = 307; + return true; + } + if (CallForStkStructWithEightBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithEightBytes*)(pages + PAGE_SIZE - sizeof(StructWithEightBytes))) != ByteValue) + { + result = 308; + return true; + } + if (CallForStkStructWithNineBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithNineBytes*)(pages + PAGE_SIZE - sizeof(StructWithNineBytes))) != ByteValue) + { + result = 309; + return true; + } + if (CallForStkStructWithTenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithTenBytes*)(pages + PAGE_SIZE - sizeof(StructWithTenBytes))) != ByteValue) + { + result = 310; + return true; + } + if (CallForStkStructWithElevenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithElevenBytes*)(pages + PAGE_SIZE - sizeof(StructWithElevenBytes))) != ByteValue) + { + result = 311; + return true; + } + if (CallForStkStructWithTwelveBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithTwelveBytes*)(pages + PAGE_SIZE - sizeof(StructWithTwelveBytes))) != ByteValue) + { + result = 312; + return true; + } + if (CallForStkStructWithThirteenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithThirteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithThirteenBytes))) != ByteValue) + { + result = 313; + return true; + } + if (CallForStkStructWithFourteenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithFourteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithFourteenBytes))) != ByteValue) + { + result = 314; + return true; + } + if (CallForStkStructWithFifteenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithFifteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithFifteenBytes))) != ByteValue) + { + result = 315; + return true; + } + if (CallForStkStructWithSixteenBytes_Arm64(0, 0, 0, 0, 0, 0, 0, *(StructWithSixteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithSixteenBytes))) != ByteValue) + { + result = 316; + return true; + } + + munmap(pages, 2 * PAGE_SIZE); + + return false; + } + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForSplitStructWithSixteenBytes_Arm(long arg0, StructWithSixteenBytes splitArg) => splitArg.Bytes[15]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForSplitStructWithSeventeenBytes_Arm(long arg0, StructWithSeventeenBytes splitArg) => splitArg.Bytes[16]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForSplitStructWithEighteenBytes_Arm(long arg0, StructWithEighteenBytes splitArg) => splitArg.Bytes[17]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForSplitStructWithNineteenBytes_Arm(long arg0, StructWithNineteenBytes splitArg) => splitArg.Bytes[18]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithOneByte_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithOneByte stkArg) => stkArg.Bytes[0]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithTwoBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithTwoBytes stkArg) => stkArg.Bytes[1]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithThreeBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithThreeBytes stkArg) => stkArg.Bytes[2]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithFourBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithFourBytes stkArg) => stkArg.Bytes[3]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithFiveBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithFiveBytes stkArg) => stkArg.Bytes[4]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithSixBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithSixBytes stkArg) => stkArg.Bytes[5]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithSevenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithSevenBytes stkArg) => stkArg.Bytes[6]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithEightBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithEightBytes stkArg) => stkArg.Bytes[7]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithNineBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithNineBytes stkArg) => stkArg.Bytes[8]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithTenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithTenBytes stkArg) => stkArg.Bytes[9]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithElevenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithElevenBytes stkArg) => stkArg.Bytes[10]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithTwelveBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithTwelveBytes stkArg) => stkArg.Bytes[11]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithThirteenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithThirteenBytes stkArg) => stkArg.Bytes[12]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithFourteenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithFourteenBytes stkArg) => stkArg.Bytes[13]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithFifteenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithFifteenBytes stkArg) => stkArg.Bytes[14]; + + [MethodImpl(MethodImplOptions.NoInlining)] + private static byte CallForStkStructWithSixteenBytes_Arm64(int arg0, int arg1, int arg2, int arg3, int arg4, int arg5, int arg6, StructWithSixteenBytes stkArg) => stkArg.Bytes[15]; + + [DllImport("libc")] + private static extern void* mmap(void* addr, nuint length, int prot, int flags, int fd, nuint offset); + + [DllImport("libc")] + private static extern int mprotect(void* addr, nuint len, int prot); + + [DllImport("libc")] + private static extern int munmap(void* addr, nuint length); + + struct StructWithOneByte + { + public fixed byte Bytes[1]; + } + + struct StructWithTwoBytes + { + public fixed byte Bytes[2]; + } + + struct StructWithThreeBytes + { + public fixed byte Bytes[3]; + } + + struct StructWithFourBytes + { + public fixed byte Bytes[4]; + } + + struct StructWithFiveBytes + { + public fixed byte Bytes[5]; + } + + struct StructWithSixBytes + { + public fixed byte Bytes[6]; + } + + struct StructWithSevenBytes + { + public fixed byte Bytes[7]; + } + + struct StructWithEightBytes + { + public fixed byte Bytes[8]; + } + + struct StructWithNineBytes + { + public fixed byte Bytes[9]; + } + + struct StructWithTenBytes + { + public fixed byte Bytes[10]; + } + + struct StructWithElevenBytes + { + public fixed byte Bytes[11]; + } + + struct StructWithTwelveBytes + { + public fixed byte Bytes[12]; + } + + struct StructWithThirteenBytes + { + public fixed byte Bytes[13]; + } + + struct StructWithFourteenBytes + { + public fixed byte Bytes[14]; + } + + struct StructWithFifteenBytes + { + public fixed byte Bytes[15]; + } + + struct StructWithSixteenBytes + { + public fixed byte Bytes[16]; + } + + struct StructWithSeventeenBytes + { + public fixed byte Bytes[17]; + } + + struct StructWithEighteenBytes + { + public fixed byte Bytes[18]; + } + + struct StructWithNineteenBytes + { + public fixed byte Bytes[19]; + } +} diff --git a/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmSplit.csproj b/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmArch.csproj similarity index 100% rename from src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmSplit.csproj rename to src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmArch.csproj diff --git a/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmSplit.cs b/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmSplit.cs deleted file mode 100644 index 2f83aa0685b7d..0000000000000 --- a/src/tests/JIT/Directed/StructABI/MisSizedStructs_ArmSplit.cs +++ /dev/null @@ -1,124 +0,0 @@ -// Licensed to the .NET Foundation under one or more agreements. -// The .NET Foundation licenses this file to you under the MIT license. - -using System; -using System.Runtime.InteropServices; -using System.Runtime.CompilerServices; - -public unsafe class MisSizedStructs_ArmSplit -{ - public const byte ByteValue = 0xC1; - - public static int Main() - { - if (ProblemWithOutOfBoundsLoads(out int result)) - { - return result; - } - - return 100; - } - - // Test that we do not load of bounds for split arguments on ARM. - // - static bool ProblemWithOutOfBoundsLoads(out int result) - { - result = 100; - - // TODO: enable for x64 once https://github.com/dotnet/runtime/issues/65937 has been fixed. - if (!OperatingSystem.IsLinux() || (RuntimeInformation.ProcessArchitecture == Architecture.X64)) - { - return false; - } - - const int PROT_NONE = 0x0; - const int PROT_READ = 0x1; - const int PROT_WRITE = 0x2; - const int MAP_PRIVATE = 0x02; - const int MAP_ANONYMOUS = 0x20; - const int PAGE_SIZE = 0x1000; - - byte* pages = (byte*)mmap(null, 2 * PAGE_SIZE, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); - - if (pages == (byte*)-1) - { - Console.WriteLine("Failed to allocate two pages, errno is {0}, giving up on the test", Marshal.GetLastSystemError()); - return false; - } - - if (mprotect(pages + PAGE_SIZE, PAGE_SIZE, PROT_NONE) != 0) - { - Console.WriteLine("Failed to protect the second page, errno is {0}, giving up on the test", Marshal.GetLastSystemError()); - munmap(pages, 2 * PAGE_SIZE); - return false; - } - - pages[PAGE_SIZE - 1] = ByteValue; - - if (CallForSplitStructWithSixteenBytes(0, *(StructWithSixteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithSixteenBytes))) != ByteValue) - { - result = 200; - return true; - } - if (CallForSplitStructWithSeventeenBytes(0, *(StructWithSeventeenBytes*)(pages + PAGE_SIZE - sizeof(StructWithSeventeenBytes))) != ByteValue) - { - result = 201; - return true; - } - if (CallForSplitStructWithEighteenBytes(0, *(StructWithEighteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithEighteenBytes))) != ByteValue) - { - result = 202; - return true; - } - if (CallForSplitStructWithNineteenBytes(0, *(StructWithNineteenBytes*)(pages + PAGE_SIZE - sizeof(StructWithNineteenBytes))) != ByteValue) - { - result = 203; - return true; - } - - munmap(pages, 2 * PAGE_SIZE); - - return false; - } - - [MethodImpl(MethodImplOptions.NoInlining)] - private static byte CallForSplitStructWithSixteenBytes(long arg0, StructWithSixteenBytes splitArg) => splitArg.Bytes[15]; - - [MethodImpl(MethodImplOptions.NoInlining)] - private static byte CallForSplitStructWithSeventeenBytes(long arg0, StructWithSeventeenBytes splitArg) => splitArg.Bytes[16]; - - [MethodImpl(MethodImplOptions.NoInlining)] - private static byte CallForSplitStructWithEighteenBytes(long arg0, StructWithEighteenBytes splitArg) => splitArg.Bytes[17]; - - [MethodImpl(MethodImplOptions.NoInlining)] - private static byte CallForSplitStructWithNineteenBytes(long arg0, StructWithNineteenBytes splitArg) => splitArg.Bytes[18]; - - [DllImport("libc")] - private static extern void* mmap(void* addr, nuint length, int prot, int flags, int fd, nuint offset); - - [DllImport("libc")] - private static extern int mprotect(void* addr, nuint len, int prot); - - [DllImport("libc")] - private static extern int munmap(void* addr, nuint length); - - struct StructWithSixteenBytes - { - public fixed byte Bytes[16]; - } - - struct StructWithSeventeenBytes - { - public fixed byte Bytes[17]; - } - - struct StructWithEighteenBytes - { - public fixed byte Bytes[18]; - } - - struct StructWithNineteenBytes - { - public fixed byte Bytes[19]; - } -}