File: DynamicallyAccessedMembersTypeHierarchy.cs
Web Access
Project: src\src\tools\illink\src\ILLink.RoslynAnalyzer\ILLink.RoslynAnalyzer.csproj (ILLink.RoslynAnalyzer)
// Copyright (c) .NET Foundation and contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
 
using System;
using System.Diagnostics.CodeAnalysis;
using Microsoft.CodeAnalysis;
using ILLink.Shared;
using ILLink.Shared.TrimAnalysis;
using ILLink.RoslynAnalyzer.TrimAnalysis;
 
namespace ILLink.RoslynAnalyzer
{
	sealed class DynamicallyAccessedMembersTypeHierarchy
	{
		public static void ApplyDynamicallyAccessedMembersToTypeHierarchy (Location typeLocation, INamedTypeSymbol type, Action<Diagnostic> reportDiagnostic)
		{
			var annotation = FlowAnnotations.GetTypeAnnotation (type);
 
			// We need to apply annotations to this type, and its base/interface types (recursively)
			// But the annotations on base/interfaces may already be applied so we don't need to apply those
			// again (and should avoid doing so as it would produce extra warnings).
			var reflectionAccessAnalyzer = new ReflectionAccessAnalyzer (reportDiagnostic, type);
			if (type.BaseType is INamedTypeSymbol baseType) {
				var baseAnnotation = FlowAnnotations.GetTypeAnnotation (baseType);
				var annotationToApplyToBase = Annotations.GetMissingMemberTypes (annotation, baseAnnotation);
 
				// Apply any annotations that didn't exist on the base type to the base type.
				// This may produce redundant warnings when the annotation is DAMT.All or DAMT.PublicConstructors and the base already has a
				// subset of those annotations.
				reflectionAccessAnalyzer.GetReflectionAccessDiagnostics (typeLocation, baseType, annotationToApplyToBase, declaredOnly: false);
			}
 
			// Most of the DynamicallyAccessedMemberTypes don't select members on interfaces. We only need to apply
			// annotations to interfaces separately if dealing with DAMT.All or DAMT.Interfaces.
			if (annotation.HasFlag (DynamicallyAccessedMemberTypes.Interfaces))
			{
				var annotationToApplyToInterfaces = annotation == DynamicallyAccessedMemberTypes.All ? annotation : DynamicallyAccessedMemberTypes.Interfaces;
				foreach (var iface in type.AllInterfaces) {
					if (FlowAnnotations.GetTypeAnnotation (iface).HasFlag (annotationToApplyToInterfaces))
						continue;
 
					// Apply All or Interfaces to the interface type.
					// DAMT.All may produce redundant warnings from implementing types, when the interface type already had some annotations.
					reflectionAccessAnalyzer.GetReflectionAccessDiagnostics (typeLocation, iface, annotationToApplyToInterfaces, declaredOnly: false);
				}
			}
 
			// The annotations this type inherited from its base types or interfaces should not produce
			// warnings on the respective base/interface members, since those are already covered by applying
			// the annotations to those types. So we only need to handle the members directly declared on this type.
			reflectionAccessAnalyzer.GetReflectionAccessDiagnostics (typeLocation, type, annotation, declaredOnly: true);
		}
	}
}