From b24427db10c2ae7d8cb530a7ab35f14def0d1c59 Mon Sep 17 00:00:00 2001 From: Jeremy Koritzinsky Date: Fri, 14 May 2021 15:01:58 -0700 Subject: [PATCH 1/4] Support implicit blittability for structs declared in and not exposed outside of the current compilation. --- .../CodeSnippets.cs | 59 +++++++++++++-- .../CompileFails.cs | 32 ++++++++ .../DllImportGenerator.UnitTests/Compiles.cs | 21 +++++- .../DllImportGenerator/TypePositionInfo.cs | 17 ++++- .../TypeSymbolExtensions.cs | 73 ++++++++++++++----- 5 files changed, 173 insertions(+), 29 deletions(-) diff --git a/DllImportGenerator/DllImportGenerator.UnitTests/CodeSnippets.cs b/DllImportGenerator/DllImportGenerator.UnitTests/CodeSnippets.cs index eb4cc83858fe..364dfd0110b6 100644 --- a/DllImportGenerator/DllImportGenerator.UnitTests/CodeSnippets.cs +++ b/DllImportGenerator/DllImportGenerator.UnitTests/CodeSnippets.cs @@ -499,14 +499,6 @@ struct MyStruct private int i; private short s; }"; - public static string GenericBlittableStructParametersAndModifiers = BasicParametersAndModifiers("MyStruct") + @" -#pragma warning disable CS0169 -[BlittableType] -struct MyStruct -{ - private T t; - private short s; -}"; public static string ArrayParametersAndModifiers(string elementType) => $@" using System.Runtime.InteropServices; @@ -1028,5 +1020,56 @@ public static partial int Method2( #endif public static int Foo() => throw null; }}"; + + public static string MaybeBlittableGenericTypeParametersAndModifiers(string typeArgument) => BasicParametersAndModifiers($"Generic<{typeArgument}>") + @" +[BlittableType] +struct Generic +{ +#pragma warning disable CS0649 + public T field; +} +"; + + public static string MaybeBlittableGenericTypeParametersAndModifiers() => + MaybeBlittableGenericTypeParametersAndModifiers(typeof(T).ToString()); + + public static string ImplicitlyBlittableStructParametersAndModifiers(string visibility = "") => BasicParametersAndModifiers("ImplicitBlittable") + $@" +#pragma warning disable CS0649 +#pragma warning disable CS0169 +{visibility} struct ImplicitBlittable +{{ + int i; +}}"; + + public static string ImplicitlyBlittableGenericTypeParametersAndModifiers(string typeArgument, string visibility = "") => BasicParametersAndModifiers($"Generic<{typeArgument}>") + $@" +{visibility} struct Generic +{{ +#pragma warning disable CS0649 +#pragma warning disable CS0169 + public T field; +}} +"; + + public static string ImplicitlyBlittableGenericTypeParametersAndModifiers(string visibility = "") => + ImplicitlyBlittableGenericTypeParametersAndModifiers(typeof(T).ToString(), visibility); + + public static string RecursiveImplicitlyBlittableStruct => BasicParametersAndModifiers("RecursiveStruct") + @" +struct RecursiveStruct +{ + RecursiveStruct s; + int i; +}"; + public static string MutuallyRecursiveImplicitlyBlittableStruct => BasicParametersAndModifiers("RecursiveStruct1") + @" +struct RecursiveStruct1 +{ + RecursiveStruct2 s; + int i; +} + +struct RecursiveStruct2 +{ + RecursiveStruct1 s; + int i; +}"; } } diff --git a/DllImportGenerator/DllImportGenerator.UnitTests/CompileFails.cs b/DllImportGenerator/DllImportGenerator.UnitTests/CompileFails.cs index 89622d1cdf23..19736372a253 100644 --- a/DllImportGenerator/DllImportGenerator.UnitTests/CompileFails.cs +++ b/DllImportGenerator/DllImportGenerator.UnitTests/CompileFails.cs @@ -95,6 +95,15 @@ public static IEnumerable CodeSnippetsToCompile() // Abstract SafeHandle type by reference yield return new object[] { CodeSnippets.BasicParameterWithByRefModifier("ref", "System.Runtime.InteropServices.SafeHandle"), 1, 0 }; + + // Non-blittable instantiation of generic type + yield return new object[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers(), 5, 0 }; + + // No marshalling annotations + + yield return new object[] { CodeSnippets.ImplicitlyBlittableStructParametersAndModifiers("public"), 5, 0 }; + yield return new object[] { CodeSnippets.ImplicitlyBlittableGenericTypeParametersAndModifiers(), 5, 0 }; + yield return new object[] { CodeSnippets.ImplicitlyBlittableGenericTypeParametersAndModifiers("public"), 5, 0 }; } [Theory] @@ -113,5 +122,28 @@ public async Task ValidateSnippets(string source, int expectedGeneratorErrors, i int compilerErrors = newComp.GetDiagnostics().Count(d => d.Severity == DiagnosticSeverity.Error); Assert.Equal(expectedCompilerErrors, compilerErrors); } + + public static IEnumerable CodeSnippetsToCompile_InvalidCode() + { + yield return new object[] { CodeSnippets.RecursiveImplicitlyBlittableStruct, 5, 1 }; + yield return new object[] { CodeSnippets.MutuallyRecursiveImplicitlyBlittableStruct, 5, 2 }; + } + + [Theory] + [MemberData(nameof(CodeSnippetsToCompile_InvalidCode))] + public async Task ValidateSnippets_InvalidCodeGracefulFailure(string source, int expectedGeneratorErrors, int expectedCompilerErrors) + { + // Do not validate that the compilation has no errors that the generator will not fix. + Compilation comp = await TestUtils.CreateCompilation(source); + + var newComp = TestUtils.RunGenerators(comp, out var generatorDiags, new Microsoft.Interop.DllImportGenerator()); + + // Verify the compilation failed with errors. + int generatorErrors = generatorDiags.Count(d => d.Severity == DiagnosticSeverity.Error); + Assert.Equal(expectedGeneratorErrors, generatorErrors); + + int compilerErrors = newComp.GetDiagnostics().Count(d => d.Severity == DiagnosticSeverity.Error); + Assert.Equal(expectedCompilerErrors, compilerErrors); + } } } diff --git a/DllImportGenerator/DllImportGenerator.UnitTests/Compiles.cs b/DllImportGenerator/DllImportGenerator.UnitTests/Compiles.cs index e5c3ccf8f187..66384762f22b 100644 --- a/DllImportGenerator/DllImportGenerator.UnitTests/Compiles.cs +++ b/DllImportGenerator/DllImportGenerator.UnitTests/Compiles.cs @@ -131,7 +131,6 @@ public static IEnumerable CodeSnippetsToCompile() // Structs yield return new[] { CodeSnippets.BlittableStructParametersAndModifiers }; - yield return new[] { CodeSnippets.GenericBlittableStructParametersAndModifiers }; // SafeHandle yield return new[] { CodeSnippets.BasicParametersAndModifiers("Microsoft.Win32.SafeHandles.SafeFileHandle") }; @@ -179,6 +178,26 @@ public static IEnumerable CodeSnippetsToCompile() yield return new[] { CodeSnippets.CustomStructMarshallingNativeTypePinnable }; yield return new[] { CodeSnippets.CustomStructMarshallingMarshalUsingParametersAndModifiers }; yield return new[] { CodeSnippets.ArrayMarshallingWithCustomStructElement }; + + // Generics + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.MaybeBlittableGenericTypeParametersAndModifiers() }; + + // Implicit blittable types + yield return new[] { CodeSnippets.ImplicitlyBlittableStructParametersAndModifiers() }; + yield return new[] { CodeSnippets.ImplicitlyBlittableStructParametersAndModifiers("internal") }; + yield return new[] { CodeSnippets.ImplicitlyBlittableGenericTypeParametersAndModifiers() }; + yield return new[] { CodeSnippets.ImplicitlyBlittableGenericTypeParametersAndModifiers("internal") }; } [Theory] diff --git a/DllImportGenerator/DllImportGenerator/TypePositionInfo.cs b/DllImportGenerator/DllImportGenerator/TypePositionInfo.cs index 0a5d4c921b22..5e53e4c800c3 100644 --- a/DllImportGenerator/DllImportGenerator/TypePositionInfo.cs +++ b/DllImportGenerator/DllImportGenerator/TypePositionInfo.cs @@ -153,7 +153,12 @@ private static MarshallingInfo GetMarshallingInfo(ITypeSymbol type, IEnumerable< if (SymbolEqualityComparer.Default.Equals(compilation.GetTypeByMetadataName(TypeNames.BlittableTypeAttribute), attributeClass)) { - return new BlittableTypeAttributeInfo(); + // If type is generic, then we need to re-evaluate that it is blittable at usage time. + if (type is INamedTypeSymbol { IsGenericType: false } || type.HasOnlyBlittableFields()) + { + return new BlittableTypeAttributeInfo(); + } + break; } else if (SymbolEqualityComparer.Default.Equals(compilation.GetTypeByMetadataName(TypeNames.NativeMarshallingAttribute), attributeClass)) { @@ -335,6 +340,16 @@ static bool TryCreateTypeBasedMarshallingInfo(ITypeSymbol type, DefaultMarshalli marshallingInfo = new ArrayMarshallingInfo(GetMarshallingInfo(elementType, Array.Empty(), defaultInfo, compilation, diagnostics, scopeSymbol)); return true; } + + if (type is INamedTypeSymbol { IsValueType: true } valueType + && !valueType.IsExposedOutsideOfCurrentCompilation() + && valueType.IsConsideredBlittable()) + { + // Allow implicit [BlittableType] on internal value types. + marshallingInfo = new BlittableTypeAttributeInfo(); + return true; + } + marshallingInfo = NoMarshallingInfo.Instance; return false; } diff --git a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs index 745f861f6101..1db9eb33336c 100644 --- a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs +++ b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs @@ -13,26 +13,34 @@ namespace Microsoft.Interop { static class TypeSymbolExtensions { - public static bool HasOnlyBlittableFields(this ITypeSymbol type) +#pragma warning disable RS1024 // Compare symbols correctly. We are using the correct comparer, but the analyzer doesn't see it correctly. + public static bool HasOnlyBlittableFields(this ITypeSymbol type) => HasOnlyBlittableFields(type, new HashSet(SymbolEqualityComparer.Default)); +#pragma warning restore RS1024 // Compare symbols correctly + + private static bool HasOnlyBlittableFields(this ITypeSymbol type, HashSet seenTypes) { + if (seenTypes.Contains(type)) + { + // A recursive struct type isn't blittable. + // It's also illegal in C#, but I believe that source generators run + // before that is detected, so we check here to avoid a stack overflow. + return false; + } + seenTypes.Add(type); foreach (var field in type.GetMembers().OfType()) { bool? fieldBlittable = field switch { - { IsStatic : true } => null, - { Type : { IsReferenceType : true }} => false, - { Type : IPointerTypeSymbol ptr } => IsConsideredBlittable(ptr.PointedAtType), - { Type : IFunctionPointerTypeSymbol } => true, - not { Type : { SpecialType : SpecialType.None }} => IsSpecialTypeBlittable(field.Type.SpecialType), + { IsStatic: true } => null, + { Type: { IsReferenceType: true } } => false, + { Type: IPointerTypeSymbol ptr } => IsConsideredBlittable(ptr.PointedAtType), + { Type: IFunctionPointerTypeSymbol } => true, + not { Type: { SpecialType: SpecialType.None } } => IsSpecialTypeBlittable(field.Type.SpecialType), // Assume that type parameters that can be blittable are blittable. // We'll re-evaluate blittability for generic fields of generic types at instantation time. - { Type : ITypeParameterSymbol } => true, - { Type : { IsValueType : false }} => false, - // A recursive struct type isn't blittable. - // It's also illegal in C#, but I believe that source generators run - // before that is detected, so we check here to avoid a stack overflow. - // [TODO]: Handle mutual recursion. - _ => !SymbolEqualityComparer.Default.Equals(field.Type, type) && IsConsideredBlittable(field.Type) + { Type: ITypeParameterSymbol } => true, + { Type: { IsValueType: false } } => false, + _ => IsConsideredBlittable(field.Type, seenTypes) }; if (fieldBlittable is false) @@ -62,7 +70,11 @@ or SpecialType.System_IntPtr _ => false }; - public static bool IsConsideredBlittable(this ITypeSymbol type) +#pragma warning disable RS1024 // Compare symbols correctly. We are using the correct comparer, but the analyzer doesn't see it correctly. + public static bool IsConsideredBlittable(this ITypeSymbol type) => IsConsideredBlittable(type, new HashSet(SymbolEqualityComparer.Default)); +#pragma warning restore RS1024 // Compare symbols correctly + + private static bool IsConsideredBlittable(this ITypeSymbol type, HashSet seenTypes) { if (type.SpecialType != SpecialType.None) { @@ -79,14 +91,14 @@ public static bool IsConsideredBlittable(this ITypeSymbol type) return false; } - if (type is IPointerTypeSymbol { PointedAtType: ITypeSymbol pointedAtType }) + if (type is IPointerTypeSymbol) { - return pointedAtType.IsConsideredBlittable(); + return true; } if (type is INamedTypeSymbol { TypeKind: TypeKind.Enum, EnumUnderlyingType: ITypeSymbol underlyingType }) { - return underlyingType!.IsConsideredBlittable(); + return underlyingType!.IsConsideredBlittable(seenTypes); } bool hasNativeMarshallingAttribute = false; @@ -107,7 +119,7 @@ public static bool IsConsideredBlittable(this ITypeSymbol type) // since we are guaranteed that if a type has generic fields, // they will be present in the contract assembly to ensure // that recursive structs can be identified at build time. - return generic.HasOnlyBlittableFields(); + return generic.HasOnlyBlittableFields(seenTypes); } return true; } @@ -126,7 +138,16 @@ public static bool IsConsideredBlittable(this ITypeSymbol type) // The struct type has generated marshalling via a source generator. // We can't guarantee that we can see the results of the struct source generator, // so we re-calculate if the type is blittable here. - return type.HasOnlyBlittableFields(); + return type.HasOnlyBlittableFields(seenTypes); + } + + if (type is INamedTypeSymbol namedType + && namedType.DeclaringSyntaxReferences.Length != 0 + && !namedType.IsExposedOutsideOfCurrentCompilation()) + { + // If a type is declared in the current compilation and not exposed outside of it, + // we will allow it to be considered blittable if its fields are considered blittable. + return type.HasOnlyBlittableFields(seenTypes); } return false; } @@ -165,5 +186,19 @@ or SpecialType.System_IntPtr _ => false }; } + + public static bool IsExposedOutsideOfCurrentCompilation(this INamedTypeSymbol type) + { + for (; type is not null; type = type.ContainingType) + { + Accessibility accessibility = type.DeclaredAccessibility; + + if (accessibility is Accessibility.Internal or Accessibility.ProtectedAndInternal or Accessibility.Private or Accessibility.Friend or Accessibility.ProtectedAndFriend) + { + return false; + } + } + return true; + } } } From 505a2a23fc883d9fd2c87555f5ce316227c33661 Mon Sep 17 00:00:00 2001 From: Jeremy Koritzinsky Date: Fri, 14 May 2021 16:17:48 -0700 Subject: [PATCH 2/4] Remove from hashtable as we return from HasOnlyBlittableFields to make sure we're handling our recursion check correctly. --- .../TypeSymbolExtensions.cs | 38 ++++++++++--------- 1 file changed, 21 insertions(+), 17 deletions(-) diff --git a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs index 1db9eb33336c..4d854c904463 100644 --- a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs +++ b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs @@ -29,26 +29,30 @@ private static bool HasOnlyBlittableFields(this ITypeSymbol type, HashSet()) { - bool? fieldBlittable = field switch + if (!field.IsStatic) { - { IsStatic: true } => null, - { Type: { IsReferenceType: true } } => false, - { Type: IPointerTypeSymbol ptr } => IsConsideredBlittable(ptr.PointedAtType), - { Type: IFunctionPointerTypeSymbol } => true, - not { Type: { SpecialType: SpecialType.None } } => IsSpecialTypeBlittable(field.Type.SpecialType), - // Assume that type parameters that can be blittable are blittable. - // We'll re-evaluate blittability for generic fields of generic types at instantation time. - { Type: ITypeParameterSymbol } => true, - { Type: { IsValueType: false } } => false, - _ => IsConsideredBlittable(field.Type, seenTypes) - }; - - if (fieldBlittable is false) - { - return false; + bool fieldBlittable = field switch + { + { Type: { IsReferenceType: true } } => false, + { Type: IPointerTypeSymbol ptr } => IsConsideredBlittable(ptr.PointedAtType), + { Type: IFunctionPointerTypeSymbol } => true, + not { Type: { SpecialType: SpecialType.None } } => IsSpecialTypeBlittable(field.Type.SpecialType), + // Assume that type parameters that can be blittable are blittable. + // We'll re-evaluate blittability for generic fields of generic types at instantation time. + { Type: ITypeParameterSymbol } => true, + { Type: { IsValueType: false } } => false, + _ => IsConsideredBlittable(field.Type, seenTypes) + }; + + if (!fieldBlittable) + { + seenTypes.Remove(type); + return false; + } } } + seenTypes.Remove(type); return true; } @@ -98,7 +102,7 @@ private static bool IsConsideredBlittable(this ITypeSymbol type, HashSet Date: Fri, 14 May 2021 16:19:46 -0700 Subject: [PATCH 3/4] Simplify function pointer and pointer cases. --- .../DllImportGenerator/TypeSymbolExtensions.cs | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs index 4d854c904463..4d4b105f55ed 100644 --- a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs +++ b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs @@ -85,7 +85,7 @@ private static bool IsConsideredBlittable(this ITypeSymbol type, HashSet Date: Tue, 18 May 2021 11:48:59 -0700 Subject: [PATCH 4/4] Update analyzers package. --- DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs | 4 ---- eng/Versions.props | 2 +- 2 files changed, 1 insertion(+), 5 deletions(-) diff --git a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs index 4d4b105f55ed..4495704b3dcc 100644 --- a/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs +++ b/DllImportGenerator/DllImportGenerator/TypeSymbolExtensions.cs @@ -13,9 +13,7 @@ namespace Microsoft.Interop { static class TypeSymbolExtensions { -#pragma warning disable RS1024 // Compare symbols correctly. We are using the correct comparer, but the analyzer doesn't see it correctly. public static bool HasOnlyBlittableFields(this ITypeSymbol type) => HasOnlyBlittableFields(type, new HashSet(SymbolEqualityComparer.Default)); -#pragma warning restore RS1024 // Compare symbols correctly private static bool HasOnlyBlittableFields(this ITypeSymbol type, HashSet seenTypes) { @@ -74,9 +72,7 @@ or SpecialType.System_IntPtr _ => false }; -#pragma warning disable RS1024 // Compare symbols correctly. We are using the correct comparer, but the analyzer doesn't see it correctly. public static bool IsConsideredBlittable(this ITypeSymbol type) => IsConsideredBlittable(type, new HashSet(SymbolEqualityComparer.Default)); -#pragma warning restore RS1024 // Compare symbols correctly private static bool IsConsideredBlittable(this ITypeSymbol type, HashSet seenTypes) { diff --git a/eng/Versions.props b/eng/Versions.props index 1409aeb7aa16..c1cc543589d8 100644 --- a/eng/Versions.props +++ b/eng/Versions.props @@ -21,7 +21,7 @@ 3.10.0-3.21229.26 1.0.1-beta1.20478.1 - 3.3.2 + 3.3.3-beta1.21268.3 2.0.0-beta1.21118.1 1.8.0