我试图有一个明确的结构布局,我将字节数组重新解释为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");
}
}
}
答案 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中的错误,并通过以下请求请求解决了。