|
// 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.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
using Mono.Cecil;
using Mono.Cecil.Cil;
namespace Mono.Linker
{
public static class MethodBodyScanner
{
public static bool IsWorthConvertingToThrow (MethodIL body)
{
// Some bodies are cheaper size wise to leave alone than to convert to a throw
Instruction? previousMeaningful = null;
int meaningfulCount = 0;
foreach (var ins in body.Instructions) {
// Handle ignoring noops because (1) it's a valid case to ignore
// and (2) When running the tests on .net core roslyn tosses in no ops
// and that leads to a difference in test results between mcs and .net framework csc.
if (ins.OpCode.Code == Code.Nop)
continue;
meaningfulCount++;
if (meaningfulCount == 1 && ins.OpCode.Code == Code.Ret)
return false;
if (meaningfulCount == 2 && ins.OpCode.Code == Code.Ret && previousMeaningful != null) {
if (previousMeaningful.OpCode.StackBehaviourPop == StackBehaviour.Pop0) {
switch (previousMeaningful.OpCode.StackBehaviourPush) {
case StackBehaviour.Pushi:
case StackBehaviour.Pushi8:
case StackBehaviour.Pushr4:
case StackBehaviour.Pushr8:
return false;
}
switch (previousMeaningful.OpCode.Code) {
case Code.Ldnull:
return false;
}
}
}
if (meaningfulCount >= 2)
return true;
previousMeaningful = ins;
}
return true;
}
}
readonly struct InterfacesOnStackScanner
{
readonly LinkContext context;
public InterfacesOnStackScanner (LinkContext context)
{
this.context = context;
}
public IEnumerable<(InterfaceImplementation, TypeDefinition)>? GetReferencedInterfaces (MethodIL methodIL)
{
var possibleStackTypes = AllPossibleStackTypes (methodIL);
if (possibleStackTypes.Count == 0)
return null;
var interfaceTypes = possibleStackTypes.Where (t => t.IsInterface).ToArray ();
if (interfaceTypes.Length == 0)
return null;
var interfaceImplementations = new HashSet<(InterfaceImplementation, TypeDefinition)> ();
// If a type could be on the stack in the body and an interface it implements could be on the stack on the body
// then we need to mark that interface implementation. When this occurs it is not safe to remove the interface implementation from the type
// even if the type is never instantiated
foreach (var type in possibleStackTypes) {
// We only sweep interfaces on classes so that's why we only care about classes
if (!type.IsClass)
continue;
TypeDefinition? currentType = type;
while (currentType?.BaseType != null) // Checking BaseType != null to skip System.Object
{
AddMatchingInterfaces (interfaceImplementations, currentType, interfaceTypes);
currentType = context.TryResolve (currentType.BaseType);
}
}
return interfaceImplementations;
}
HashSet<TypeDefinition> AllPossibleStackTypes (MethodIL methodIL)
{
var types = new HashSet<TypeDefinition> ();
foreach (VariableDefinition var in methodIL.Variables)
AddIfResolved (types, var.VariableType);
foreach (var param in methodIL.Method.GetParameters ())
AddIfResolved (types, param.ParameterType);
foreach (ExceptionHandler eh in methodIL.ExceptionHandlers) {
if (eh.HandlerType == ExceptionHandlerType.Catch) {
AddIfResolved (types, eh.CatchType);
}
}
foreach (Instruction instruction in methodIL.Instructions) {
if (instruction.Operand is FieldReference fieldReference) {
if (context.TryResolve (fieldReference)?.FieldType is TypeReference fieldType)
AddIfResolved (types, fieldType);
} else if (instruction.Operand is MethodReference methodReference) {
if (methodReference is GenericInstanceMethod genericInstanceMethod)
AddFromGenericInstance (types, genericInstanceMethod);
if (methodReference.DeclaringType is GenericInstanceType genericInstanceType)
AddFromGenericInstance (types, genericInstanceType);
var resolvedMethod = context.TryResolve (methodReference);
if (resolvedMethod != null) {
if (resolvedMethod.HasMetadataParameters ()) {
foreach (var param in resolvedMethod.GetParameters ())
AddIfResolved (types, param.ParameterType);
}
AddFromGenericParameterProvider (types, resolvedMethod);
AddFromGenericParameterProvider (types, resolvedMethod.DeclaringType);
AddIfResolved (types, resolvedMethod.ReturnType);
}
}
}
return types;
}
void AddMatchingInterfaces (HashSet<(InterfaceImplementation, TypeDefinition)> results, TypeDefinition type, TypeDefinition[] interfaceTypes)
{
if (!type.HasInterfaces)
return;
foreach (var interfaceType in interfaceTypes) {
if (HasInterface (type, interfaceType, out InterfaceImplementation? implementation))
results.Add ((implementation, type));
}
}
bool HasInterface (TypeDefinition type, TypeDefinition interfaceType, [NotNullWhen (true)] out InterfaceImplementation? implementation)
{
implementation = null;
if (!type.HasInterfaces)
return false;
foreach (var iface in type.Interfaces) {
if (context.TryResolve (iface.InterfaceType) == interfaceType) {
implementation = iface;
return true;
}
}
return false;
}
void AddFromGenericInstance (HashSet<TypeDefinition> set, IGenericInstance instance)
{
if (!instance.HasGenericArguments)
return;
foreach (var genericArgument in instance.GenericArguments)
AddIfResolved (set, genericArgument);
}
void AddFromGenericParameterProvider (HashSet<TypeDefinition> set, IGenericParameterProvider provider)
{
if (!provider.HasGenericParameters)
return;
foreach (var genericParameter in provider.GenericParameters) {
foreach (var constraint in genericParameter.Constraints)
AddIfResolved (set, constraint.ConstraintType);
}
}
void AddIfResolved (HashSet<TypeDefinition> set, TypeReference item)
{
var resolved = context.TryResolve (item);
if (resolved == null)
return;
set.Add (resolved);
}
}
}
|