Skip to content

Commit 0818df1

Browse files
authoredNov 13, 2021
[DllImportGenerator] Treat pointer fields in structs as blittable regardless of what they point at (#61538)
1 parent aae8f51 commit 0818df1

File tree

5 files changed

+110
-19
lines changed

5 files changed

+110
-19
lines changed
 

‎src/libraries/System.Runtime.InteropServices/gen/Microsoft.Interop.SourceGeneration/TypeSymbolExtensions.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ private static bool HasOnlyBlittableFields(this ITypeSymbol type, ImmutableHashS
3131
bool fieldBlittable = field switch
3232
{
3333
{ Type: { IsReferenceType: true } } => false,
34-
{ Type: IPointerTypeSymbol ptr } => IsConsideredBlittable(ptr.PointedAtType),
34+
{ Type: IPointerTypeSymbol ptr } => true,
3535
{ Type: IFunctionPointerTypeSymbol } => true,
3636
not { Type: { SpecialType: SpecialType.None } } => IsSpecialTypeBlittable(field.Type.SpecialType),
3737
// Assume that type parameters that can be blittable are blittable.

‎src/libraries/System.Runtime.InteropServices/tests/DllImportGenerator.Tests/BlittableStructTests.cs

+78-1
Original file line numberDiff line numberDiff line change
@@ -29,12 +29,23 @@ public static partial void DoubleIntFieldsRefReturn(
2929
public static partial void DoubleIntFieldsOutReturn(
3030
IntFields input,
3131
out IntFields result);
32+
33+
[GeneratedDllImport(NativeExportsNE_Binary, EntryPoint = "blittablestructs_increment_invert_ptrfields_byref")]
34+
public static partial void IncrementInvertPointerFieldsByRef(ref PointerFields result);
35+
36+
[GeneratedDllImport(NativeExportsNE_Binary, EntryPoint = "blittablestructs_increment_invert_ptrfields_byref")]
37+
public static partial void IncrementInvertPointerFieldsByRefIn(in PointerFields result);
38+
39+
[GeneratedDllImport(NativeExportsNE_Binary, EntryPoint = "blittablestructs_increment_invert_ptrfields_refreturn")]
40+
public static partial void IncrementInvertPointerFieldsRefReturn(
41+
PointerFields input,
42+
ref PointerFields result);
3243
}
3344

3445
public class BlittableStructTests
3546
{
3647
[Fact]
37-
public void ValidateBlittableStruct()
48+
public void ValidateIntFields()
3849
{
3950
const int A = 24, B = 37, C = 59;
4051
var initial = new IntFields()
@@ -82,5 +93,71 @@ public void ValidateBlittableStruct()
8293
Assert.Equal(expected, input); // Updated even when passed with in keyword (matches built-in system)
8394
}
8495
}
96+
97+
[Fact]
98+
public unsafe void ValidatePointerFields()
99+
{
100+
int iInitial = 31;
101+
bool bInitial = false;
102+
char cInitial = 'A';
103+
104+
int iExpected = iInitial + 1;
105+
bool bExpected = !bInitial;
106+
char cExpected = (char)(cInitial + 1);
107+
108+
int i = iInitial;
109+
bool b = bInitial;
110+
char c = cInitial;
111+
var initial = new PointerFields()
112+
{
113+
i = &i,
114+
b = &b,
115+
c = &c,
116+
};
117+
118+
PointerFields input = initial;
119+
{
120+
int iResult;
121+
bool bResult;
122+
char cResult;
123+
var result = new PointerFields()
124+
{
125+
i = &iResult,
126+
b = &bResult,
127+
c = &cResult
128+
};
129+
NativeExportsNE.IncrementInvertPointerFieldsRefReturn(input, ref result);
130+
Assert.Equal(initial, input);
131+
ValidateFieldValues(result);
132+
}
133+
134+
{
135+
ResetFieldValues(input);
136+
NativeExportsNE.IncrementInvertPointerFieldsByRef(ref input);
137+
Assert.Equal(initial, input);
138+
ValidateFieldValues(input);
139+
}
140+
141+
{
142+
ResetFieldValues(input);
143+
NativeExportsNE.IncrementInvertPointerFieldsByRefIn(in input);
144+
Assert.Equal(initial, input);
145+
ValidateFieldValues(input);
146+
}
147+
148+
void ResetFieldValues(PointerFields input)
149+
{
150+
*(input.i) = iInitial;
151+
*(input.b) = bInitial;
152+
*(input.c) = cInitial;
153+
}
154+
155+
void ValidateFieldValues(PointerFields result)
156+
{
157+
Assert.Equal(iExpected, *result.i);
158+
Assert.Equal(bExpected, *result.b);
159+
Assert.Equal(cExpected, *result.c);
160+
}
161+
}
85162
}
86163
}

‎src/libraries/System.Runtime.InteropServices/tests/DllImportGenerator.UnitTests/ManualTypeMarshallingAnalyzerTests.cs

+3-17
Original file line numberDiff line numberDiff line change
@@ -1280,34 +1280,20 @@ public Native(S<T> s)
12801280
}
12811281

12821282
[ConditionalFact]
1283-
public async Task ValueTypeContainingPointerBlittableType_DoesNotReportDiagnostic()
1283+
public async Task ValueTypeContainingPointers_DoesNotReportDiagnostic()
12841284
{
12851285
var source = @"
12861286
using System.Runtime.InteropServices;
12871287
12881288
[BlittableType]
12891289
unsafe struct S
12901290
{
1291-
private int* ptr;
1291+
private int* intPtr;
1292+
private bool* boolPtr;
12921293
}";
12931294
await VerifyCS.VerifyAnalyzerAsync(source);
12941295
}
12951296

1296-
[ConditionalFact]
1297-
public async Task ValueTypeContainingPointerToNonBlittableType_ReportsDiagnostic()
1298-
{
1299-
var source = @"
1300-
using System.Runtime.InteropServices;
1301-
1302-
[{|#0:BlittableType|}]
1303-
unsafe struct S
1304-
{
1305-
private bool* ptr;
1306-
}";
1307-
await VerifyCS.VerifyAnalyzerAsync(source,
1308-
VerifyCS.Diagnostic(BlittableTypeMustBeBlittableRule).WithLocation(0).WithArguments("S"));
1309-
}
1310-
13111297
[ConditionalFact]
13121298
public async Task BlittableValueTypeContainingPointerToSelf_DoesNotReportDiagnostic()
13131299
{

‎src/libraries/System.Runtime.InteropServices/tests/TestAssets/NativeExports/BlittableStructs.cs

+20
Original file line numberDiff line numberDiff line change
@@ -40,5 +40,25 @@ public static void DoubleIntFieldsRefReturn(
4040
result->b = input.b * 2;
4141
result->c = input.c * 2;
4242
}
43+
44+
[UnmanagedCallersOnly(EntryPoint = "blittablestructs_increment_invert_ptrfields_byref")]
45+
[DNNE.C99DeclCode("struct ptr_fields { int* i; int* b; uint16_t* c; };")]
46+
public static void IncrementInvertPointerFieldsByRef(
47+
[DNNE.C99Type("struct ptr_fields*")] PointerFields* result)
48+
{
49+
*(result->i) += 1;
50+
*(result->b) = !*(result->b);
51+
*(result->c) += (char)1;
52+
}
53+
54+
[UnmanagedCallersOnly(EntryPoint = "blittablestructs_increment_invert_ptrfields_refreturn")]
55+
public static void IncrementInvertPointerFieldsRefReturn(
56+
[DNNE.C99Type("struct ptr_fields")] PointerFields input,
57+
[DNNE.C99Type("struct ptr_fields*")] PointerFields* result)
58+
{
59+
*(result->i) = *(input.i) + 1;
60+
*(result->b) = !(*input.b);
61+
*(result->c) = (char)(*(input.c) + 1);
62+
}
4363
}
4464
}

‎src/libraries/System.Runtime.InteropServices/tests/TestAssets/SharedTypes/Blittable.cs

+8
Original file line numberDiff line numberDiff line change
@@ -13,4 +13,12 @@ public struct IntFields
1313
public int b;
1414
public int c;
1515
}
16+
17+
[BlittableType]
18+
public unsafe struct PointerFields
19+
{
20+
public int* i;
21+
public bool* b;
22+
public char* c;
23+
}
1624
}

0 commit comments

Comments
 (0)
Please sign in to comment.