// Copyright (c) 2017 Siegfried Pammer
//
// Permission is hereby granted, free of charge, to any person obtaining a copy of this
// software and associated documentation files (the "Software"), to deal in the Software
// without restriction, including without limitation the rights to use, copy, modify, merge,
// publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons
// to whom the Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all copies or
// substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
// INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
// FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using ICSharpCode.Decompiler.CSharp.Resolver;
using ICSharpCode.Decompiler.Semantics;
using ICSharpCode.Decompiler.TypeSystem;
using ICSharpCode.Decompiler.TypeSystem.Implementation;
using ICSharpCode.Decompiler.Util;
namespace ICSharpCode.Decompiler.IL.Transforms
{
///
/// Converts LINQ Expression Trees to ILFunctions/ILAst instructions.
///
/// We build a tree of Func{ILInstruction}s, which are only executed, if the whole transform succeeds.
///
public class TransformExpressionTrees : IStatementTransform
{
///
/// Returns true if the instruction matches the pattern for Expression.Lambda calls.
///
static bool MightBeExpressionTree(ILInstruction inst, ILInstruction stmt)
{
if (!(inst is CallInstruction call
&& call.Method.FullNameIs("System.Linq.Expressions.Expression", "Lambda")
&& call.Arguments.Count == 2))
return false;
if (!(IsEmptyParameterList(call.Arguments[1]) || (call.Arguments[1] is Block block && block.Kind == BlockKind.ArrayInitializer)))
return false;
//if (!ILInlining.CanUninline(call, stmt))
// return false;
return true;
}
static bool IsEmptyParameterList(ILInstruction inst)
{
if (inst is CallInstruction emptyCall && emptyCall.Method.FullNameIs("System.Array", "Empty") && emptyCall.Arguments.Count == 0)
return true;
if (inst.MatchNewArr(out var type) && type.FullName == "System.Linq.Expressions.ParameterExpression")
return true;
if (inst.MatchNewArr(out type) && type.FullName == "System.Linq.Expressions.Expression")
return true;
return false;
}
bool MatchParameterVariableAssignment(ILInstruction expr, out ILVariable parameterReferenceVar, out IType type, out string name)
{
// stloc(v, call(Expression::Parameter, call(Type::GetTypeFromHandle, ldtoken(...)), ldstr(...)))
type = null;
name = null;
if (!expr.MatchStLoc(out parameterReferenceVar, out var init))
return false;
if (!parameterReferenceVar.IsSingleDefinition)
return false;
if (!(parameterReferenceVar.Kind == VariableKind.Local || parameterReferenceVar.Kind == VariableKind.StackSlot))
return false;
if (parameterReferenceVar.Type == null || parameterReferenceVar.Type.FullName != "System.Linq.Expressions.ParameterExpression")
return false;
if (!(init is CallInstruction initCall && initCall.Arguments.Count == 2))
return false;
if (!(initCall.Method.FullNameIs("System.Linq.Expressions.Expression", "Parameter")))
return false;
CallInstruction typeArg = initCall.Arguments[0] as CallInstruction;
if (typeArg == null || typeArg.Arguments.Count != 1)
return false;
if (!typeArg.Method.FullNameIs("System.Type", "GetTypeFromHandle"))
return false;
return typeArg.Arguments[0].MatchLdTypeToken(out type) && initCall.Arguments[1].MatchLdStr(out name);
}
StatementTransformContext context;
Dictionary parameters;
Dictionary parameterMapping;
List instructionsToRemove;
Stack lambdaStack;
CSharpConversions conversions;
CSharpResolver resolver;
public void Run(Block block, int pos, StatementTransformContext context)
{
if (!context.Settings.ExpressionTrees)
return;
this.context = context;
this.conversions = CSharpConversions.Get(context.TypeSystem);
this.resolver = new CSharpResolver(context.TypeSystem);
this.parameters = new Dictionary();
this.parameterMapping = new Dictionary();
this.instructionsToRemove = new List();
this.lambdaStack = new Stack();
for (int i = pos; i < block.Instructions.Count; i++)
{
if (MatchParameterVariableAssignment(block.Instructions[i], out var v, out var type, out var name))
{
parameters.Add(v, (type, name));
continue;
}
if (TryConvertExpressionTree(block.Instructions[i], block.Instructions[i]))
{
foreach (var inst in instructionsToRemove)
block.Instructions.Remove(inst);
instructionsToRemove.Clear();
}
break;
}
}
bool TryConvertExpressionTree(ILInstruction instruction, ILInstruction statement)
{
if (MightBeExpressionTree(instruction, statement))
{
var (lambda, type) = ConvertLambda((CallInstruction)instruction);
if (lambda != null)
{
context.Step("Convert Expression Tree", instruction);
var newLambda = (ILFunction)lambda();
SetExpressionTreeFlag(newLambda, (CallInstruction)instruction);
instruction.ReplaceWith(newLambda);
return true;
}
return false;
}
if (instruction is Block block && block.Kind == BlockKind.ControlFlow)
return false; // don't look into nested blocks
foreach (var child in instruction.Children)
{
if (TryConvertExpressionTree(child, statement))
return true;
}
return false;
}
///
/// Converts a Expression.Lambda call into an ILFunction.
/// If the conversion fails, null is returned.
///
(Func, IType) ConvertLambda(CallInstruction instruction)
{
if (instruction.Method.Name != "Lambda" || instruction.Arguments.Count != 2 || instruction.Method.ReturnType.FullName != "System.Linq.Expressions.Expression" || instruction.Method.ReturnType.TypeArguments.Count != 1)
return (null, SpecialType.UnknownType);
var parameterList = new List();
var parameterVariablesList = new List();
if (!ReadParameters(instruction.Arguments[1], parameterList, parameterVariablesList, new SimpleTypeResolveContext(context.Function.Method)))
return (null, SpecialType.UnknownType);
var container = new BlockContainer();
container.AddILRange(instruction);
var functionType = instruction.Method.ReturnType.TypeArguments[0];
var returnType = functionType.GetDelegateInvokeMethod()?.ReturnType ?? SpecialType.UnknownType;
var function = new ILFunction(returnType, parameterList, context.Function.GenericContext, container, ILFunctionKind.ExpressionTree);
function.DelegateType = functionType;
function.Kind = IsExpressionTree(functionType) ? ILFunctionKind.ExpressionTree : ILFunctionKind.Delegate;
function.Variables.AddRange(parameterVariablesList);
function.AddILRange(instruction);
lambdaStack.Push(function);
var (bodyInstruction, type) = ConvertInstruction(instruction.Arguments[0]);
lambdaStack.Pop();
if (bodyInstruction == null)
return (null, SpecialType.UnknownType);
return (BuildFunction, function.DelegateType);
ILFunction BuildFunction()
{
lambdaStack.Push(function);
var convertedBody = bodyInstruction();
lambdaStack.Pop();
container.ExpectedResultType = convertedBody.ResultType;
container.Blocks.Add(new Block() { Instructions = { new Leave(container, convertedBody) } });
// Replace all other usages of the parameter variable
foreach (var mapping in parameterMapping)
{
foreach (var load in mapping.Key.LoadInstructions.ToArray())
{
if (load.IsDescendantOf(instruction))
continue;
load.ReplaceWith(new LdLoc(mapping.Value));
}
}
return function;
}
}
(Func, IType) ConvertQuote(CallInstruction invocation)
{
if (invocation.Arguments.Count != 1)
return (null, SpecialType.UnknownType);
var argument = invocation.Arguments.Single();
if (argument is ILFunction function)
{
return (() => function, function.DelegateType);
}
else
{
var (converted, type) = ConvertInstruction(argument);
if (converted == null)
return (converted, type);
return (BuildQuote, type);
ILInstruction BuildQuote()
{
var f = converted();
if (f is ILFunction lambda && argument is CallInstruction call)
{
SetExpressionTreeFlag(lambda, call);
}
return f;
}
}
}
void SetExpressionTreeFlag(ILFunction lambda, CallInstruction call)
{
lambda.Kind = IsExpressionTree(call.Method.ReturnType) ? ILFunctionKind.ExpressionTree : ILFunctionKind.Delegate;
lambda.DelegateType = call.Method.ReturnType;
}
bool ReadParameters(ILInstruction initializer, IList parameters, IList parameterVariables, ITypeResolveContext resolveContext)
{
switch (initializer)
{
case Block initializerBlock:
if (initializerBlock.Kind != BlockKind.ArrayInitializer)
return false;
int i = 0;
foreach (var inst in initializerBlock.Instructions.OfType())
{
if (i >= this.parameters.Count)
return false;
if (!inst.Value.MatchLdLoc(out var v))
return false;
if (!this.parameters.TryGetValue(v, out var value))
return false;
// Add parameter variable only once to mapping.
if (!this.parameterMapping.ContainsKey(v))
{
var param = new ILVariable(VariableKind.Parameter, value.Item1, i) { Name = value.Item2 };
parameterMapping.Add(v, param);
parameterVariables.Add(param);
parameters.Add(new DefaultParameter(value.Item1, value.Item2));
instructionsToRemove.Add((ILInstruction)v.StoreInstructions[0]);
}
i++;
}
return true;
default:
return IsEmptyParameterList(initializer);
}
}
(Func, IType) ConvertInstruction(ILInstruction instruction, IType typeHint = null)
{
var (inst, type) = Convert();
if (inst == null)
return (null, type);
ILInstruction DoConvert()
{
var result = inst();
Debug.Assert(type != null, "IType must be non-null!");
Debug.Assert(result.ResultType == type.GetStackType(), "StackTypes must match!");
if (typeHint != null)
{
if (result.ResultType != typeHint.GetStackType())
{
return new Conv(result, typeHint.GetStackType().ToPrimitiveType(), false, typeHint.GetSign());
}
}
return result;
}
return (DoConvert, typeHint ?? type);
(Func, IType) Convert()
{
switch (instruction)
{
case CallInstruction invocation:
if (invocation.Method.DeclaringType.FullName != "System.Linq.Expressions.Expression")
return (null, SpecialType.UnknownType);
switch (invocation.Method.Name)
{
case "Add":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Add, false);
case "AddChecked":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Add, true);
case "And":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.BitAnd);
case "AndAlso":
return ConvertLogicOperator(invocation, true);
case "ArrayAccess":
case "ArrayIndex":
return ConvertArrayIndex(invocation);
case "ArrayLength":
return ConvertArrayLength(invocation);
case "Call":
return ConvertCall(invocation);
case "Coalesce":
return ConvertCoalesce(invocation);
case "Condition":
return ConvertCondition(invocation);
case "Constant":
return ConvertConstant(invocation);
case "Convert":
return ConvertCast(invocation, false);
case "ConvertChecked":
return ConvertCast(invocation, true);
case "Divide":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Div);
case "Equal":
return ConvertComparison(invocation, ComparisonKind.Equality);
case "ExclusiveOr":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.BitXor);
case "Field":
return ConvertField(invocation, typeHint);
case "GreaterThan":
return ConvertComparison(invocation, ComparisonKind.GreaterThan);
case "GreaterThanOrEqual":
return ConvertComparison(invocation, ComparisonKind.GreaterThanOrEqual);
case "Invoke":
return ConvertInvoke(invocation);
case "Lambda":
return ConvertLambda(invocation);
case "LeftShift":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.ShiftLeft);
case "LessThan":
return ConvertComparison(invocation, ComparisonKind.LessThan);
case "LessThanOrEqual":
return ConvertComparison(invocation, ComparisonKind.LessThanOrEqual);
case "ListInit":
return ConvertListInit(invocation);
case "MemberInit":
return ConvertMemberInit(invocation);
case "Modulo":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Rem);
case "Multiply":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Mul, false);
case "MultiplyChecked":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Mul, true);
case "Negate":
return ConvertUnaryNumericOperator(invocation, BinaryNumericOperator.Sub, false);
case "NegateChecked":
return ConvertUnaryNumericOperator(invocation, BinaryNumericOperator.Sub, true);
case "New":
return ConvertNewObject(invocation);
case "NewArrayBounds":
return ConvertNewArrayBounds(invocation);
case "NewArrayInit":
return ConvertNewArrayInit(invocation);
case "Not":
return ConvertNotOperator(invocation);
case "NotEqual":
return ConvertComparison(invocation, ComparisonKind.Inequality);
case "OnesComplement":
return ConvertNotOperator(invocation);
case "Or":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.BitOr);
case "OrElse":
return ConvertLogicOperator(invocation, false);
case "Property":
return ConvertProperty(invocation);
case "Quote":
return ConvertQuote(invocation);
case "RightShift":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.ShiftRight);
case "Subtract":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Sub, false);
case "SubtractChecked":
return ConvertBinaryNumericOperator(invocation, BinaryNumericOperator.Sub, true);
case "TypeAs":
return ConvertTypeAs(invocation);
case "TypeIs":
return ConvertTypeIs(invocation);
}
return (null, SpecialType.UnknownType);
case ILFunction function:
ILFunction ApplyChangesToILFunction()
{
if (function.Kind == ILFunctionKind.ExpressionTree)
{
function.DelegateType = UnwrapExpressionTree(function.DelegateType);
function.Kind = ILFunctionKind.Delegate;
}
return function;
}
return (ApplyChangesToILFunction, function.DelegateType);
case LdLoc ldloc:
if (IsExpressionTreeParameter(ldloc.Variable))
{
// Replace an already mapped parameter with the actual ILVariable,
// we generated earlier.
if (parameterMapping.TryGetValue(ldloc.Variable, out var v))
{
if (typeHint.SkipModifiers() is ByReferenceType && !v.Type.IsByRefLike)
return (() => new LdLoca(v), typeHint);
return (() => new LdLoc(v), v.Type);
}
// This is a parameter variable from an outer scope.
// We can't replace these variables just yet, because the transform works backwards.
// We simply return the same instruction again, but return the actual expected type,
// so our transform can continue normally.
// Later, we will replace all references to unmapped variables,
// with references to mapped parameters.
if (ldloc.Variable.IsSingleDefinition && ldloc.Variable.StoreInstructions[0] is ILInstruction instr)
{
if (MatchParameterVariableAssignment(instr, out _, out var t, out _))
return (() => new ExpressionTreeCast(t, ldloc, false), t);
}
}
return (null, SpecialType.UnknownType);
default:
return (null, SpecialType.UnknownType);
}
}
}
bool IsExpressionTree(IType delegateType) => delegateType is ParameterizedType pt
&& pt.FullName == "System.Linq.Expressions.Expression"
&& pt.TypeArguments.Count == 1;
IType UnwrapExpressionTree(IType delegateType)
{
if (delegateType is ParameterizedType pt && pt.FullName == "System.Linq.Expressions.Expression" && pt.TypeArguments.Count == 1)
{
return pt.TypeArguments[0];
}
return delegateType;
}
(Func, IType) ConvertArrayIndex(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var (array, arrayType) = ConvertInstruction(invocation.Arguments[0]);
if (array == null)
return (null, SpecialType.UnknownType);
if (!(arrayType is ArrayType type))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
arguments = new[] { invocation.Arguments[1] };
ILInstruction Convert()
{
Func[] toBeConverted = new Func[arguments.Count];
for (int i = 0; i < arguments.Count; i++)
{
var (converted, indexType) = ConvertInstruction(arguments[i]);
if (converted == null)
return null;
toBeConverted[i] = converted;
}
return new LdObj(new LdElema(type.ElementType, array(), toBeConverted.SelectArray(f => f())) { DelayExceptions = true }, type.ElementType);
}
return (Convert, type.ElementType);
}
(Func, IType) ConvertArrayLength(CallInstruction invocation)
{
if (invocation.Arguments.Count != 1)
return (null, SpecialType.UnknownType);
var (converted, _) = ConvertInstruction(invocation.Arguments[0]);
if (converted == null)
return (null, SpecialType.UnknownType);
return (() => new LdLen(StackType.I4, converted()), context.TypeSystem.FindType(KnownTypeCode.Int32));
}
(Func, IType) ConvertBinaryNumericOperator(CallInstruction invocation, BinaryNumericOperator op, bool? isChecked = null)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
var (left, leftType) = ConvertInstruction(invocation.Arguments[0]);
if (left == null)
return (null, SpecialType.UnknownType);
var (right, rightType) = ConvertInstruction(invocation.Arguments[1]);
if (right == null)
return (null, SpecialType.UnknownType);
IMember method;
switch (invocation.Arguments.Count)
{
case 2:
if (op == BinaryNumericOperator.ShiftLeft || op == BinaryNumericOperator.ShiftRight)
{
if (!NullableType.GetUnderlyingType(rightType).IsKnownType(KnownTypeCode.Int32))
return (null, SpecialType.UnknownType);
}
else
{
if (!rightType.Equals(leftType))
return (null, SpecialType.UnknownType);
}
return (() => new BinaryNumericInstruction(op, left(), right(),
NullableType.GetUnderlyingType(leftType).GetStackType(),
NullableType.GetUnderlyingType(rightType).GetStackType(),
isChecked == true,
leftType.GetSign(),
isLifted: NullableType.IsNullable(leftType) || NullableType.IsNullable(rightType)), leftType);
case 3:
if (!MatchGetMethodFromHandle(invocation.Arguments[2], out method))
return (null, SpecialType.UnknownType);
return (() => new Call((IMethod)method) {
Arguments = { left(), right() }
}, method.ReturnType);
case 4:
if (!invocation.Arguments[2].MatchLdcI4(out var isLiftedToNull))
return (null, SpecialType.UnknownType);
if (!MatchGetMethodFromHandle(invocation.Arguments[3], out method))
return (null, SpecialType.UnknownType);
bool isLifted = NullableType.IsNullable(leftType);
if (isLifted)
method = CSharpOperators.LiftUserDefinedOperator((IMethod)method);
return (() => new Call((IMethod)method) {
Arguments = { left(), right() }
}, isLiftedToNull != 0 ? NullableType.Create(method.Compilation, method.ReturnType) : method.ReturnType);
default:
return (null, SpecialType.UnknownType);
}
}
(Func, IType) ConvertBind(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var (value, typeValue) = ConvertInstruction(invocation.Arguments[1]);
if (value == null)
return (null, SpecialType.UnknownType);
if (MatchGetMethodFromHandle(invocation.Arguments[0], out var member))
{
}
else if (MatchGetFieldFromHandle(invocation.Arguments[0], out member))
{
}
else
{
return (null, SpecialType.UnknownType);
}
switch (member)
{
case IMethod method:
if (method.IsStatic)
return (targetVariable => new Call(method) { Arguments = { new LdLoc(targetVariable), value() } }, method.ReturnType);
else
return (targetVariable => new CallVirt(method) { Arguments = { new LdLoc(targetVariable), value() } }, method.ReturnType);
case IField field:
return (targetVariable => new StObj(new LdFlda(new LdLoc(targetVariable), (IField)member) { DelayExceptions = true }, value(), member.ReturnType), field.ReturnType);
}
return (null, SpecialType.UnknownType);
}
(Func, IType) ConvertCall(CallInstruction invocation)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
IList arguments = null;
Func targetConverter = null;
IType targetType = null;
if (MatchGetMethodFromHandle(invocation.Arguments[0], out var member))
{
// static method
if (invocation.Arguments.Count != 2 || !MatchArgumentList(invocation.Arguments[1], out arguments))
{
arguments = new List(invocation.Arguments.Skip(1));
}
}
else if (MatchGetMethodFromHandle(invocation.Arguments[1], out member))
{
if (invocation.Arguments.Count != 3 || !MatchArgumentList(invocation.Arguments[2], out arguments))
{
arguments = new List(invocation.Arguments.Skip(2));
}
if (!invocation.Arguments[0].MatchLdNull())
{
(targetConverter, targetType) = ConvertInstruction(invocation.Arguments[0]);
if (targetConverter == null)
return (null, SpecialType.UnknownType);
}
}
if (arguments == null)
return (null, SpecialType.UnknownType);
IMethod method = (IMethod)member;
var convertedArguments = ConvertCallArguments(arguments, method);
if (convertedArguments == null)
return (null, SpecialType.UnknownType);
if (method.FullName == "System.Reflection.MethodInfo.CreateDelegate" && method.Parameters.Count == 2)
{
if (!MatchGetMethodFromHandle(UnpackConstant(invocation.Arguments[0]), out var targetMethod))
return (null, SpecialType.UnknownType);
if (!MatchGetTypeFromHandle(UnpackConstant(arguments[0]), out var delegateType))
return (null, SpecialType.UnknownType);
return (() => new NewObj(delegateType.GetConstructors().Single()) {
Arguments = { convertedArguments[1](), new LdFtn((IMethod)targetMethod) }
}, delegateType);
}
CallInstruction BuildCall()
{
CallInstruction call;
if (method.IsStatic)
{
call = new Call(method);
}
else
{
call = new CallVirt(method);
}
if (targetConverter != null)
{
call.Arguments.Add(PrepareCallTarget(method.DeclaringType, targetConverter(), targetType));
}
call.Arguments.AddRange(convertedArguments.Select(f => f()));
return call;
}
return (BuildCall, method.ReturnType);
}
ILInstruction PrepareCallTarget(IType expectedType, ILInstruction target, IType targetType)
{
switch (CallInstruction.ExpectedTypeForThisPointer(expectedType))
{
case StackType.Ref:
if (target.ResultType == StackType.Ref)
{
return target;
}
else if (target is LdLoc ldloc)
{
return new LdLoca(ldloc.Variable).WithILRange(ldloc);
}
else
{
return new AddressOf(target, expectedType);
}
case StackType.O:
if (targetType.IsReferenceType == false)
{
return new Box(target, targetType);
}
else
{
return target;
}
default:
if (expectedType.Kind == TypeKind.Unknown && target.ResultType != StackType.Unknown)
{
return new Conv(target, PrimitiveType.Unknown, false, Sign.None);
}
return target;
}
}
ILInstruction UnpackConstant(ILInstruction inst)
{
if (!(inst is CallInstruction call && call.Method.FullName == "System.Linq.Expressions.Expression.Constant" && call.Arguments.Count == 2))
return inst;
return call.Arguments[0];
}
Func[] ConvertCallArguments(IList arguments, IMethod method)
{
var converted = new Func[arguments.Count];
Debug.Assert(arguments.Count == method.Parameters.Count);
for (int i = 0; i < arguments.Count; i++)
{
var expectedType = method.Parameters[i].Type;
var argument = ConvertInstruction(arguments[i], expectedType).Item1;
if (argument == null)
return null;
converted[i] = argument;
}
return converted;
}
(Func, IType) ConvertCast(CallInstruction invocation, bool isChecked)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
if (!MatchGetTypeFromHandle(invocation.Arguments[1], out var targetType))
return (null, SpecialType.UnknownType);
var (expr, exprType) = ConvertInstruction(invocation.Arguments[0]);
if (expr == null)
return (null, SpecialType.UnknownType);
if (exprType.IsSmallIntegerType() && targetType.IsKnownType(KnownTypeCode.Int32))
return (expr, targetType);
return (() => new ExpressionTreeCast(targetType, expr(), isChecked), targetType);
}
(Func, IType) ConvertCoalesce(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var (trueInst, trueInstType) = ConvertInstruction(invocation.Arguments[0]);
if (trueInst == null)
return (null, SpecialType.UnknownType);
var (fallbackInst, fallbackInstType) = ConvertInstruction(invocation.Arguments[1]);
if (fallbackInst == null)
return (null, SpecialType.UnknownType);
var kind = NullCoalescingKind.Ref;
var trueInstTypeNonNullable = NullableType.GetUnderlyingType(trueInstType);
IType targetType;
if (NullableType.IsNullable(trueInstType) && conversions.ImplicitConversion(fallbackInstType, trueInstTypeNonNullable).IsValid)
{
targetType = trueInstTypeNonNullable;
kind = NullableType.IsNullable(fallbackInstType) ? NullCoalescingKind.Nullable : NullCoalescingKind.NullableWithValueFallback;
}
else if (conversions.ImplicitConversion(fallbackInstType, trueInstType).IsValid)
{
targetType = trueInstType;
}
else
{
targetType = fallbackInstType;
}
return (() => new NullCoalescingInstruction(kind, trueInst(), fallbackInst()) {
UnderlyingResultType = trueInstTypeNonNullable.GetStackType()
}, targetType);
}
(Func, IType) ConvertComparison(CallInstruction invocation, ComparisonKind kind)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
var (left, leftType) = ConvertInstruction(invocation.Arguments[0]);
if (left == null)
return (null, SpecialType.UnknownType);
var (right, rightType) = ConvertInstruction(invocation.Arguments[1]);
if (right == null)
return (null, SpecialType.UnknownType);
if (invocation.Arguments.Count == 4 && invocation.Arguments[2].MatchLdcI4(out var isLiftedToNull) && MatchGetMethodFromHandle(invocation.Arguments[3], out var method))
{
bool isLifted = NullableType.IsNullable(leftType);
if (isLifted)
method = CSharpOperators.LiftUserDefinedOperator((IMethod)method);
return (() => new Call((IMethod)method) { Arguments = { left(), right() } }, isLiftedToNull != 0 ? NullableType.Create(method.Compilation, method.ReturnType) : method.ReturnType);
}
var rr = resolver.ResolveBinaryOperator(kind.ToBinaryOperatorType(), new ResolveResult(leftType), new ResolveResult(rightType)) as OperatorResolveResult;
if (rr != null && !rr.IsError && rr.UserDefinedOperatorMethod != null)
{
return (() => new Call(rr.UserDefinedOperatorMethod) { Arguments = { left(), right() } }, rr.UserDefinedOperatorMethod.ReturnType);
}
if (leftType.IsKnownType(KnownTypeCode.String) && rightType.IsKnownType(KnownTypeCode.String))
{
IMethod operatorMethod;
switch (kind)
{
case ComparisonKind.Equality:
operatorMethod = leftType.GetMethods(m => m.IsOperator && m.Name == "op_Equality" && m.Parameters.Count == 2).FirstOrDefault(m => m.Parameters[0].Type.IsKnownType(KnownTypeCode.String) && m.Parameters[1].Type.IsKnownType(KnownTypeCode.String));
if (operatorMethod == null)
return (null, SpecialType.UnknownType);
break;
case ComparisonKind.Inequality:
operatorMethod = leftType.GetMethods(m => m.IsOperator && m.Name == "op_Inequality" && m.Parameters.Count == 2).FirstOrDefault(m => m.Parameters[0].Type.IsKnownType(KnownTypeCode.String) && m.Parameters[1].Type.IsKnownType(KnownTypeCode.String));
if (operatorMethod == null)
return (null, SpecialType.UnknownType);
break;
default:
return (null, SpecialType.UnknownType);
}
return (() => new Call(operatorMethod) { Arguments = { left(), right() } }, operatorMethod.ReturnType);
}
var resultType = context.TypeSystem.FindType(KnownTypeCode.Boolean);
var lifting = NullableType.IsNullable(leftType) ? ComparisonLiftingKind.CSharp : ComparisonLiftingKind.None;
var utype = NullableType.GetUnderlyingType(leftType);
return (() => new Comp(kind, lifting, utype.GetStackType(), utype.GetSign(), left(), right()), resultType);
}
(Func, IType) ConvertCondition(CallInstruction invocation)
{
if (invocation.Arguments.Count != 3)
return (null, SpecialType.UnknownType);
var (condition, conditionType) = ConvertInstruction(invocation.Arguments[0]);
if (condition == null || !conditionType.IsKnownType(KnownTypeCode.Boolean))
return (null, SpecialType.UnknownType);
var (trueInst, trueInstType) = ConvertInstruction(invocation.Arguments[1]);
if (trueInst == null)
return (null, SpecialType.UnknownType);
var (falseInst, falseInstType) = ConvertInstruction(invocation.Arguments[2]);
if (falseInst == null)
return (null, SpecialType.UnknownType);
if (!NormalizeTypeVisitor.TypeErasure.EquivalentTypes(trueInstType, falseInstType))
return (null, SpecialType.UnknownType);
return (() => new IfInstruction(condition(), trueInst(), falseInst()), trueInstType);
}
(Func, IType) ConvertConstant(CallInstruction invocation)
{
if (!MatchConstantCall(invocation, out var value, out var type))
return (null, SpecialType.UnknownType);
if (value.MatchBox(out var arg, out var boxType))
{
if (boxType.Kind == TypeKind.Enum || boxType.IsKnownType(KnownTypeCode.Boolean))
return (() => new ExpressionTreeCast(boxType, ConvertValue(arg, invocation), false), boxType);
return (() => ConvertValue(arg, invocation), type);
}
return (() => ConvertValue(value, invocation), type);
}
(Func, IType) ConvertElementInit(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
if (!MatchGetMethodFromHandle(invocation.Arguments[0], out var member))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
return (null, SpecialType.UnknownType);
var args = new Func[arguments.Count];
for (int i = 0; i < arguments.Count; i++)
{
var arg = ConvertInstruction(arguments[i]).Item1;
if (arg == null)
return (null, SpecialType.UnknownType);
args[i] = arg;
}
ILInstruction BuildCall()
{
CallInstruction call = member.IsStatic
? (CallInstruction)new Call((IMethod)member)
: new CallVirt((IMethod)member);
call.Arguments.AddRange(args.Select(f => f()));
return call;
}
return (BuildCall, member.ReturnType);
}
(Func, IType) ConvertField(CallInstruction invocation, IType typeHint)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
Func targetConverter = null;
if (!invocation.Arguments[0].MatchLdNull())
{
targetConverter = ConvertInstruction(invocation.Arguments[0]).Item1;
if (targetConverter == null)
return (null, SpecialType.UnknownType);
}
if (!MatchGetFieldFromHandle(invocation.Arguments[1], out var member))
return (null, SpecialType.UnknownType);
IType type = member.ReturnType;
if (typeHint.SkipModifiers() is ByReferenceType && !member.ReturnType.IsByRefLike)
{
type = typeHint;
}
return (BuildField, type);
ILInstruction BuildField()
{
ILInstruction inst;
if (targetConverter == null)
{
inst = new LdsFlda((IField)member);
}
else
{
var target = targetConverter();
if (member.DeclaringType.IsReferenceType == true)
{
inst = new LdFlda(target, (IField)member) { DelayExceptions = true };
}
else
{
inst = new LdFlda(new AddressOf(target, member.DeclaringType), (IField)member) { DelayExceptions = true };
}
}
if (!(typeHint.SkipModifiers() is ByReferenceType && !member.ReturnType.IsByRefLike))
{
inst = new LdObj(inst, member.ReturnType);
}
return inst;
}
}
(Func, IType) ConvertInvoke(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var (targetConverter, targetType) = ConvertInstruction(invocation.Arguments[0]);
if (targetConverter == null)
return (null, SpecialType.UnknownType);
var invokeMethod = targetType.GetDelegateInvokeMethod();
if (invokeMethod == null)
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
return (null, SpecialType.UnknownType);
var convertedArguments = ConvertCallArguments(arguments, invokeMethod);
if (convertedArguments == null)
return (null, SpecialType.UnknownType);
ILInstruction BuildCall()
{
var call = new CallVirt(invokeMethod);
call.Arguments.Add(targetConverter());
call.Arguments.AddRange(convertedArguments.Select(f => f()));
return call;
}
return (BuildCall, invokeMethod.ReturnType);
}
(Func, IType) ConvertListInit(CallInstruction invocation)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
var newObj = ConvertInstruction(invocation.Arguments[0]).Item1;
if (newObj == null)
return (null, SpecialType.UnknownType);
if (!MatchNew((CallInstruction)invocation.Arguments[0], out var ctor))
return (null, SpecialType.UnknownType);
IList arguments;
if (!MatchGetMethodFromHandle(invocation.Arguments[1], out var member))
{
if (!MatchArgumentList(invocation.Arguments[1], out arguments))
return (null, SpecialType.UnknownType);
}
else
{
if (invocation.Arguments.Count != 3 || !MatchArgumentList(invocation.Arguments[2], out arguments))
return (null, SpecialType.UnknownType);
}
if (arguments == null || arguments.Count == 0)
return (null, SpecialType.UnknownType);
Func[] convertedArguments = new Func[arguments.Count];
for (int i = 0; i < arguments.Count; i++)
{
if (arguments[i] is CallInstruction elementInit && elementInit.Method.FullName == "System.Linq.Expressions.Expression.ElementInit")
{
var arg = ConvertElementInit(elementInit).Item1;
if (arg == null)
return (null, SpecialType.UnknownType);
convertedArguments[i] = v => { var a = arg(); ((CallInstruction)a).Arguments.Insert(0, new LdLoc(v)); return a; };
}
else
{
var arg = ConvertInstruction(arguments[i]).Item1;
if (arg == null)
return (null, SpecialType.UnknownType);
convertedArguments[i] = v => arg();
}
}
Block BuildBlock()
{
var initializerBlock = new Block(BlockKind.CollectionInitializer);
ILFunction function = lambdaStack.Peek();
var initializer = function.RegisterVariable(VariableKind.InitializerTarget, ctor.DeclaringType);
initializerBlock.FinalInstruction = new LdLoc(initializer);
initializerBlock.Instructions.Add(new StLoc(initializer, newObj()));
initializerBlock.Instructions.AddRange(convertedArguments.Select(f => f(initializer)));
return initializerBlock;
}
return (BuildBlock, ctor.DeclaringType);
}
(Func, IType) ConvertLogicOperator(CallInstruction invocation, bool and)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
var (left, leftType) = ConvertInstruction(invocation.Arguments[0]);
if (left == null)
return (null, SpecialType.UnknownType);
var (right, rightType) = ConvertInstruction(invocation.Arguments[1]);
if (right == null)
return (null, SpecialType.UnknownType);
IMember method;
switch (invocation.Arguments.Count)
{
case 2:
var resultType = context.TypeSystem.FindType(KnownTypeCode.Boolean);
return (() => and ? IfInstruction.LogicAnd(left(), right()) : IfInstruction.LogicOr(left(), right()), resultType);
case 3:
if (!MatchGetMethodFromHandle(invocation.Arguments[2], out method))
return (null, SpecialType.UnknownType);
return (() => new Call((IMethod)method) {
Arguments = { left(), right() }
}, method.ReturnType);
case 4:
if (!invocation.Arguments[2].MatchLdcI4(out var isLiftedToNull))
return (null, SpecialType.UnknownType);
if (!MatchGetMethodFromHandle(invocation.Arguments[3], out method))
return (null, SpecialType.UnknownType);
bool isLifted = NullableType.IsNullable(leftType);
if (isLifted)
method = CSharpOperators.LiftUserDefinedOperator((IMethod)method);
return (() => new Call((IMethod)method) {
Arguments = { left(), right() }
}, isLiftedToNull != 0 ? NullableType.Create(method.Compilation, method.ReturnType) : method.ReturnType);
default:
return (null, SpecialType.UnknownType);
}
}
(Func, IType) ConvertMemberInit(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var newObj = ConvertInstruction(invocation.Arguments[0]).Item1;
if (newObj == null)
return (null, SpecialType.UnknownType);
if (!MatchNew((CallInstruction)invocation.Arguments[0], out var ctor))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
return (null, SpecialType.UnknownType);
if (arguments == null || arguments.Count == 0)
return (null, SpecialType.UnknownType);
Func[] convertedArguments = new Func[arguments.Count];
for (int i = 0; i < arguments.Count; i++)
{
Func arg;
if (arguments[i] is CallInstruction bind && bind.Method.FullName == "System.Linq.Expressions.Expression.Bind")
{
arg = ConvertBind(bind).Item1;
if (arg == null)
return (null, SpecialType.UnknownType);
}
else
{
return (null, SpecialType.UnknownType);
}
convertedArguments[i] = arg;
}
ILInstruction BuildBlock()
{
var function = lambdaStack.Peek();
var initializer = function.RegisterVariable(VariableKind.InitializerTarget, ctor.DeclaringType);
var initializerBlock = new Block(BlockKind.CollectionInitializer);
initializerBlock.FinalInstruction = new LdLoc(initializer);
initializerBlock.Instructions.Add(new StLoc(initializer, newObj()));
initializerBlock.Instructions.AddRange(convertedArguments.Select(f => f(initializer)));
return initializerBlock;
}
return (BuildBlock, ctor.DeclaringType);
}
(Func, IType) ConvertNewArrayBounds(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
if (!MatchGetTypeFromHandle(invocation.Arguments[0], out var type))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
return (null, SpecialType.UnknownType);
if (arguments.Count == 0)
return (null, SpecialType.UnknownType);
var indices = new Func[arguments.Count];
for (int i = 0; i < arguments.Count; i++)
{
var index = ConvertInstruction(arguments[i]).Item1;
if (index == null)
return (null, SpecialType.UnknownType);
indices[i] = index;
}
return (() => new NewArr(type, indices.SelectArray(f => f())), new ArrayType(context.TypeSystem, type, arguments.Count));
}
(Func, IType) ConvertNewArrayInit(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
if (!MatchGetTypeFromHandle(invocation.Arguments[0], out var type))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
return (null, SpecialType.UnknownType);
ArrayType arrayType = new ArrayType(context.BlockContext.TypeSystem, type);
if (arguments.Count == 0)
return (() => new NewArr(type, new LdcI4(0)), arrayType);
var convertedArguments = new Func[arguments.Count];
for (int i = 0; i < arguments.Count; i++)
{
ILInstruction item = arguments[i];
var value = ConvertInstruction(item).Item1;
if (value == null)
return (null, SpecialType.UnknownType);
convertedArguments[i] = value;
}
ILInstruction BuildInitializer()
{
var block = (Block)invocation.Arguments[1];
var function = lambdaStack.Peek();
var variable = function.RegisterVariable(VariableKind.InitializerTarget, arrayType);
Block initializer = new Block(BlockKind.ArrayInitializer);
initializer.Instructions.Add(new StLoc(variable, new NewArr(type, new LdcI4(convertedArguments.Length))));
for (int i = 0; i < convertedArguments.Length; i++)
{
initializer.Instructions.Add(new StObj(new LdElema(type, new LdLoc(variable), new LdcI4(i)) { DelayExceptions = true }, convertedArguments[i](), type));
}
initializer.FinalInstruction = new LdLoc(variable);
return initializer;
}
return (BuildInitializer, arrayType);
}
bool MatchNew(CallInstruction invocation, out IMethod ctor)
{
ctor = null;
if (invocation.Method.Name != "New")
return false;
switch (invocation.Arguments.Count)
{
case 1:
if (MatchGetTypeFromHandle(invocation.Arguments[0], out var type))
{
ctor = type.GetConstructors(c => c.Parameters.Count == 0).FirstOrDefault();
return ctor != null;
}
if (MatchGetConstructorFromHandle(invocation.Arguments[0], out var member))
{
ctor = (IMethod)member;
return true;
}
return false;
case 2:
case 3:
if (!MatchGetConstructorFromHandle(invocation.Arguments[0], out member))
return false;
ctor = (IMethod)member;
return true;
default:
return false;
}
}
(Func, IType) ConvertNewObject(CallInstruction invocation)
{
switch (invocation.Arguments.Count)
{
case 1:
if (MatchGetTypeFromHandle(invocation.Arguments[0], out var type))
{
var ctor = type.GetConstructors(c => c.Parameters.Count == 0).FirstOrDefault();
if (ctor == null)
return (null, SpecialType.UnknownType);
return (() => new NewObj(ctor), type);
}
if (MatchGetConstructorFromHandle(invocation.Arguments[0], out var member))
{
return (() => new NewObj((IMethod)member), member.DeclaringType);
}
return (null, SpecialType.UnknownType);
case 2:
if (!MatchGetConstructorFromHandle(invocation.Arguments[0], out member))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out var arguments))
return (null, SpecialType.UnknownType);
IMethod method = (IMethod)member;
Func[] convertedArguments = ConvertCallArguments(arguments, method);
if (convertedArguments == null)
return (null, SpecialType.UnknownType);
return (() => BuildNewObj(method, convertedArguments), member.DeclaringType);
case 3:
if (!MatchGetConstructorFromHandle(invocation.Arguments[0], out member))
return (null, SpecialType.UnknownType);
if (!MatchArgumentList(invocation.Arguments[1], out arguments))
return (null, SpecialType.UnknownType);
method = (IMethod)member;
convertedArguments = ConvertCallArguments(arguments, method);
if (convertedArguments == null)
return (null, SpecialType.UnknownType);
return (() => BuildNewObj(method, convertedArguments), member.DeclaringType);
}
ILInstruction BuildNewObj(IMethod method, Func[] args)
{
var newObj = new NewObj(method);
newObj.Arguments.AddRange(args.Select(f => f()));
return newObj;
}
return (null, SpecialType.UnknownType);
}
(Func, IType) ConvertNotOperator(CallInstruction invocation)
{
if (invocation.Arguments.Count < 1)
return (null, SpecialType.UnknownType);
var (argument, argumentType) = ConvertInstruction(invocation.Arguments[0]);
if (argument == null)
return (null, SpecialType.UnknownType);
var underlyingType = NullableType.GetUnderlyingType(argumentType);
switch (invocation.Arguments.Count)
{
case 1:
bool isLifted = NullableType.IsNullable(argumentType);
return (() => underlyingType.IsKnownType(KnownTypeCode.Boolean)
? Comp.LogicNot(argument(), isLifted)
: (ILInstruction)new BitNot(argument(), isLifted, underlyingType.GetStackType()), argumentType);
case 2:
if (!MatchGetMethodFromHandle(invocation.Arguments[1], out var method))
return (null, SpecialType.UnknownType);
return (() => new Call((IMethod)method) {
Arguments = { argument() }
}, method.ReturnType);
default:
return (null, SpecialType.UnknownType);
}
}
(Func, IType) ConvertProperty(CallInstruction invocation)
{
if (invocation.Arguments.Count < 2)
return (null, SpecialType.UnknownType);
Func targetConverter = null;
IType targetType = null;
if (!invocation.Arguments[0].MatchLdNull())
{
(targetConverter, targetType) = ConvertInstruction(invocation.Arguments[0]);
if (targetConverter == null)
return (null, SpecialType.UnknownType);
}
if (!MatchGetMethodFromHandle(invocation.Arguments[1], out var member))
return (null, SpecialType.UnknownType);
IList arguments;
if (invocation.Arguments.Count != 3 || !MatchArgumentList(invocation.Arguments[2], out arguments))
{
arguments = new List();
}
var convertedArguments = ConvertCallArguments(arguments, (IMethod)member);
if (convertedArguments == null)
return (null, SpecialType.UnknownType);
ILInstruction BuildProperty()
{
CallInstruction call;
if (member.IsStatic)
{
call = new Call((IMethod)member);
}
else
{
call = new CallVirt((IMethod)member);
}
if (targetConverter != null)
{
call.Arguments.Add(PrepareCallTarget(member.DeclaringType, targetConverter(), targetType));
}
call.Arguments.AddRange(convertedArguments.Select(f => f()));
return call;
}
return (BuildProperty, member.ReturnType);
}
(Func, IType) ConvertTypeAs(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var converted = ConvertInstruction(invocation.Arguments[0]).Item1;
if (!MatchGetTypeFromHandle(invocation.Arguments[1], out var type))
return (null, SpecialType.UnknownType);
if (converted == null)
return (null, SpecialType.UnknownType);
ILInstruction BuildTypeAs()
{
ILInstruction inst = new IsInst(converted(), type);
// We must follow ECMA-335, III.4.6:
// If typeTok is a nullable type, Nullable, it is interpreted as "boxed" T.
if (type.IsKnownType(KnownTypeCode.NullableOfT))
inst = new UnboxAny(inst, type);
return inst;
}
return (BuildTypeAs, type);
}
(Func, IType) ConvertTypeIs(CallInstruction invocation)
{
if (invocation.Arguments.Count != 2)
return (null, SpecialType.UnknownType);
var converted = ConvertInstruction(invocation.Arguments[0]).Item1;
if (!MatchGetTypeFromHandle(invocation.Arguments[1], out var type))
return (null, SpecialType.UnknownType);
var resultType = context.TypeSystem.FindType(KnownTypeCode.Boolean);
if (converted != null)
return (() => new Comp(ComparisonKind.Inequality, Sign.None, new IsInst(converted(), type), new LdNull()), resultType);
return (null, SpecialType.UnknownType);
}
(Func, IType) ConvertUnaryNumericOperator(CallInstruction invocation, BinaryNumericOperator op, bool? isChecked = null)
{
if (invocation.Arguments.Count < 1)
return (null, SpecialType.UnknownType);
var (argument, argumentType) = ConvertInstruction(invocation.Arguments[0]);
if (argument == null)
return (null, SpecialType.UnknownType);
switch (invocation.Arguments.Count)
{
case 1:
ILInstruction left;
var underlyingType = NullableType.GetUnderlyingType(argumentType);
switch (underlyingType.GetStackType())
{
case StackType.I4:
left = new LdcI4(0);
break;
case StackType.I8:
left = new LdcI8(0);
break;
case StackType.I:
left = new Conv(new LdcI4(0), PrimitiveType.I, false, Sign.None);
break;
case StackType.F4:
left = new LdcF4(0);
break;
case StackType.F8:
left = new LdcF8(0);
break;
case StackType.O when underlyingType.IsKnownType(KnownTypeCode.Decimal):
left = new LdcDecimal(0);
break;
default:
return (null, SpecialType.UnknownType);
}
return (() => new BinaryNumericInstruction(op, left, argument(),
underlyingType.GetStackType(),
underlyingType.GetStackType(),
isChecked == true,
argumentType.GetSign(),
isLifted: NullableType.IsNullable(argumentType)), argumentType);
case 2:
if (!MatchGetMethodFromHandle(invocation.Arguments[1], out var method))
return (null, SpecialType.UnknownType);
return (() => new Call((IMethod)method) {
Arguments = { argument() }
}, method.ReturnType);
}
return (null, SpecialType.UnknownType);
}
ILInstruction ConvertValue(ILInstruction value, ILInstruction context)
{
switch (value)
{
case LdLoc ldloc:
if (IsExpressionTreeParameter(ldloc.Variable))
{
if (!parameterMapping.TryGetValue(ldloc.Variable, out var v))
return ldloc.Clone();
if (context is CallInstruction parentCall
&& parentCall.Method.FullName == "System.Linq.Expressions.Expression.Call"
&& v.StackType.IsIntegerType())
return new LdLoca(v).WithILRange(ldloc);
return null;
}
else if (IsClosureReference(ldloc.Variable))
{
if (ldloc.Variable.Kind == VariableKind.Local)
{
ldloc.Variable.Kind = VariableKind.DisplayClassLocal;
}
if (ldloc.Variable.CaptureScope == null)
{
ldloc.Variable.CaptureScope = BlockContainer.FindClosestContainer(context);
var f = ldloc.Variable.CaptureScope.Ancestors.OfType().FirstOrDefault();
if (f != null)
{
f.CapturedVariables.Add(ldloc.Variable);
}
}
return ldloc;
}
else
{
return ldloc;
}
default:
return value.Clone();
}
}
bool IsClosureReference(ILVariable variable)
{
if (!variable.IsSingleDefinition || !(variable.StoreInstructions.SingleOrDefault() is StLoc store))
return false;
if (!(store.Value is NewObj newObj))
return false;
return TransformDisplayClassUsage.IsPotentialClosure(this.context, newObj);
}
bool IsExpressionTreeParameter(ILVariable variable)
{
return variable.Type.FullName == "System.Linq.Expressions.ParameterExpression";
}
bool MatchConstantCall(ILInstruction inst, out ILInstruction value, out IType type)
{
value = null;
type = null;
if (inst is CallInstruction call && call.Method.FullName == "System.Linq.Expressions.Expression.Constant")
{
value = call.Arguments[0];
if (call.Arguments.Count == 2)
return MatchGetTypeFromHandle(call.Arguments[1], out type);
type = value.InferType(context.TypeSystem);
return true;
}
return false;
}
internal static bool MatchGetTypeFromHandle(ILInstruction inst, out IType type)
{
type = null;
return inst is CallInstruction getTypeCall
&& getTypeCall.Method.FullName == "System.Type.GetTypeFromHandle"
&& getTypeCall.Arguments.Count == 1
&& getTypeCall.Arguments[0].MatchLdTypeToken(out type);
}
bool MatchGetMethodFromHandle(ILInstruction inst, out IMember member)
{
member = null;
//castclass System.Reflection.MethodInfo(call GetMethodFromHandle(ldmembertoken op_Addition))
if (!inst.MatchCastClass(out var arg, out var type))
return false;
if (type.FullName != "System.Reflection.MethodInfo")
return false;
if (!(arg is CallInstruction call && call.Method.FullName == "System.Reflection.MethodBase.GetMethodFromHandle"))
return false;
return MatchFromHandleParameterList(call, out member);
}
bool MatchGetConstructorFromHandle(ILInstruction inst, out IMember member)
{
member = null;
//castclass System.Reflection.ConstructorInfo(call GetMethodFromHandle(ldmembertoken op_Addition))
if (!inst.MatchCastClass(out var arg, out var type))
return false;
if (type.FullName != "System.Reflection.ConstructorInfo")
return false;
if (!(arg is CallInstruction call && call.Method.FullName == "System.Reflection.MethodBase.GetMethodFromHandle"))
return false;
return MatchFromHandleParameterList(call, out member);
}
bool MatchGetFieldFromHandle(ILInstruction inst, out IMember member)
{
member = null;
if (!(inst is CallInstruction call && call.Method.FullName == "System.Reflection.FieldInfo.GetFieldFromHandle"))
return false;
return MatchFromHandleParameterList(call, out member);
}
static bool MatchFromHandleParameterList(CallInstruction call, out IMember member)
{
member = null;
switch (call.Arguments.Count)
{
case 1:
if (!call.Arguments[0].MatchLdMemberToken(out member))
return false;
break;
case 2:
if (!call.Arguments[0].MatchLdMemberToken(out member))
return false;
if (!call.Arguments[1].MatchLdTypeToken(out _))
return false;
break;
default:
return false;
}
return true;
}
bool MatchArgumentList(ILInstruction inst, out IList arguments)
{
arguments = null;
if (!(inst is Block block && block.Kind == BlockKind.ArrayInitializer))
{
if (IsEmptyParameterList(inst))
{
arguments = new List();
return true;
}
return false;
}
int i = 0;
arguments = new List();
foreach (var item in block.Instructions.OfType())
{
if (!(item.Target is LdElema ldelem && ldelem.Indices.Single().MatchLdcI4(i)))
return false;
arguments.Add(item.Value);
i++;
}
return true;
}
}
}