Guid

时间:2018-06-13 17:48:30

标签: c# struct marshalling .net-framework-version

我试图有一个明确的结构布局,我将字节数组重新解释为guids。对于.net 4.7和.net 4.7.2,在32位运行时工作正常。但是当你以64位运行它时,它可以在.NET 4.7中运行,但似乎无法在.NET 4.7.2中运行。这段代码无效或者这可能是.NET中的错误吗?

[更新]:测试了一些其他配置。适用于4.6.2和4.7,但不适用于4.7.1和4.7.2。

[更新2]:看看反汇编。在64位中,分配结果为无操作。所以我认为很明显它是.net中的一个错误:

32位反汇编:

TestStruct t = new TestStruct();
03010480  lea         edi,[ebp-58h]
03010483  xorps       xmm0,xmm0
03010486  movq        mmword ptr [edi],xmm0
0301048A  movq        mmword ptr [edi+8],xmm0
0301048F  movq        mmword ptr [edi+10h],xmm0
03010494  movq        mmword ptr [edi+18h],xmm0

t.Guid1 = Guid.NewGuid();
03010499  lea         eax,[ebp-58h]
0301049C  mov         dword ptr [ebp-64h],eax
0301049F  lea         ecx,[ebp-74h]
030104A2  call        723628F0
030104A7  mov         edi,dword ptr [ebp-64h]
030104AA  lea         esi,[ebp-74h]
030104AD  movq        xmm0,mmword ptr [esi]
030104B1  movq        mmword ptr [edi],xmm0
030104B5  movq        xmm0,mmword ptr [esi+8]
030104BA  movq        mmword ptr [edi+8],xmm0

t.Guid2 = t.Guid1;
030104BF  lea         edi,[ebp-58h]
030104C2  add         edi,10h
030104C5  lea         esi,[ebp-58h]
030104C8  movq        xmm0,mmword ptr [esi]
030104CC  movq        mmword ptr [edi],xmm0
030104D0  movq        xmm0,mmword ptr [esi+8]
030104D5  movq        mmword ptr [edi+8],xmm0

64位反汇编:

TestStruct t = new TestStruct();
00007FFB054604B2  lea         rcx,[rbp+78h]
00007FFB054604B6  vxorpd      xmm0,xmm0,xmm0
00007FFB054604BB  vmovdqu     xmmword ptr [rcx],xmm0
00007FFB054604C0  vmovdqu     xmmword ptr [rcx+10h],xmm0

t.Guid1 = Guid.NewGuid();
00007FFB054604C6  lea         rcx,[rbp+78h]
00007FFB054604CA  mov         qword ptr [rbp+68h],rcx
00007FFB054604CE  lea         rcx,[rbp+58h]
00007FFB054604D2  call        00007FFB639BE8C0
00007FFB054604D7  mov         rax,qword ptr [rbp+68h]
00007FFB054604DB  vmovdqu     xmm0,xmmword ptr [rbp+58h]
00007FFB054604E1  vmovdqu     xmmword ptr [rax],xmm0

t.Guid2 = t.Guid1;
00007FFB054604E6  nop

重现的代码:

[StructLayout(LayoutKind.Explicit, Size = SIZE)]
internal unsafe struct TestStruct
{
    public const int SIZE = 32;

    [FieldOffset(0)]
    private fixed byte _data[SIZE];

    [FieldOffset(0), MarshalAs(UnmanagedType.Struct, SizeConst = 16)]
    public Guid Guid1;

    [FieldOffset(16), MarshalAs(UnmanagedType.Struct, SizeConst = 16)]
    public Guid Guid2;
}

internal class Program
{
    private static void Main()
    {
        TestStruct t = new TestStruct();
        t.Guid1 = Guid.NewGuid();
        t.Guid2 = t.Guid1;

        if (t.Guid1 != t.Guid2)
        {
            throw new InvalidOperationException("Guids aren't equal");
        }
    }
}

2 个答案:

答案 0 :(得分:0)

我认为问题可能与FieldOffset有关。 FieldOffset(16)正在替换字节数组。 如果您封装字段并使用属性,则不应该有问题。

[StructLayout(LayoutKind.Explicit, Size = SIZE)]
internal unsafe struct TestStruct
{
    public const int SIZE = 32;

    [FieldOffset(0)]
    public fixed byte _data[SIZE];

    [FieldOffset(0), MarshalAs(UnmanagedType.Struct, SizeConst = 16)]
    private Guid guid1;

    [FieldOffset(16), MarshalAs(UnmanagedType.Struct, SizeConst = 16)]
    private Guid guid2;

    public Guid Guid1 { get => guid1; set => guid1 = value; }
    public Guid Guid2 { get => guid2; set => guid2 = value; }
}

答案 1 :(得分:0)

结果证明这是RyuJIT中的错误,并通过以下请求请求解决了。

https://github.com/dotnet/coreclr/pull/17971