// // EmitContext.cs // // Author: // Miguel de Icaza (miguel@novell.com) // Jb Evain (jbevain@novell.com) // // (C) 2008 Novell, Inc. (http://www.novell.com) // // 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. // #if !FULL_AOT_RUNTIME using System; using System.Collections.ObjectModel; using System.Collections.Generic; using System.IO; using System.Linq; using System.Reflection; using System.Reflection.Emit; using System.Runtime.CompilerServices; namespace System.Linq.Expressions { class CompilationContext { class ParameterReplacer : ExpressionTransformer { CompilationContext context; ExecutionScope scope; object [] locals; public ParameterReplacer (CompilationContext context, ExecutionScope scope, object [] locals) { this.context = context; this.scope = scope; this.locals = locals; } protected override Expression VisitParameter (ParameterExpression parameter) { var scope = this.scope; var locals = this.locals; while (scope != null) { int position = IndexOfHoistedLocal (scope, parameter); if (position != -1) return ReadHoistedLocalFromArray (locals, position); locals = scope.Locals; scope = scope.Parent; } return parameter; } Expression ReadHoistedLocalFromArray (object [] locals, int position) { return Expression.Field ( Expression.Convert ( Expression.ArrayIndex ( Expression.Constant (locals), Expression.Constant (position)), locals [position].GetType ()), "Value"); } int IndexOfHoistedLocal (ExecutionScope scope, ParameterExpression parameter) { return context.units [scope.compilation_unit].IndexOfHoistedLocal (parameter); } } class HoistedVariableDetector : ExpressionVisitor { Dictionary parameter_to_lambda = new Dictionary (); Dictionary> hoisted_map; LambdaExpression lambda; public Dictionary> Process (LambdaExpression lambda) { Visit (lambda); return hoisted_map; } protected override void VisitLambda (LambdaExpression lambda) { this.lambda = lambda; foreach (var parameter in lambda.Parameters) parameter_to_lambda [parameter] = lambda; base.VisitLambda (lambda); } protected override void VisitParameter (ParameterExpression parameter) { if (lambda.Parameters.Contains (parameter)) return; Hoist (parameter); } void Hoist (ParameterExpression parameter) { LambdaExpression lambda; if (!parameter_to_lambda.TryGetValue (parameter, out lambda)) return; if (hoisted_map == null) hoisted_map = new Dictionary> (); List hoisted; if (!hoisted_map.TryGetValue (lambda, out hoisted)) { hoisted = new List (); hoisted_map [lambda] = hoisted; } hoisted.Add (parameter); } } List globals = new List (); List units = new List (); Dictionary> hoisted_map; public int AddGlobal (object global) { return AddItemToList (global, globals); } public object [] GetGlobals () { return globals.ToArray (); } static int AddItemToList (T item, IList list) { list.Add (item); return list.Count - 1; } public int AddCompilationUnit (LambdaExpression lambda) { DetectHoistedVariables (lambda); return AddCompilationUnit (null, lambda); } public int AddCompilationUnit (EmitContext parent, LambdaExpression lambda) { var context = new EmitContext (this, parent, lambda); var unit = AddItemToList (context, units); context.Emit (); return unit; } void DetectHoistedVariables (LambdaExpression lambda) { hoisted_map = new HoistedVariableDetector ().Process (lambda); } public List GetHoistedLocals (LambdaExpression lambda) { if (hoisted_map == null) return null; List hoisted; hoisted_map.TryGetValue (lambda, out hoisted); return hoisted; } public object [] CreateHoistedLocals (int unit) { var hoisted = GetHoistedLocals (units [unit].Lambda); return new object [hoisted == null ? 0 : hoisted.Count]; } public Expression IsolateExpression (ExecutionScope scope, object [] locals, Expression expression) { return new ParameterReplacer (this, scope, locals).Transform (expression); } public Delegate CreateDelegate () { return CreateDelegate (0, new ExecutionScope (this)); } public Delegate CreateDelegate (int unit, ExecutionScope scope) { return units [unit].CreateDelegate (scope); } } class EmitContext { CompilationContext context; EmitContext parent; LambdaExpression lambda; DynamicMethod method; LocalBuilder hoisted_store; List hoisted; public readonly ILGenerator ig; public bool HasHoistedLocals { get { return hoisted != null && hoisted.Count > 0; } } public LambdaExpression Lambda { get { return lambda; } } public EmitContext (CompilationContext context, EmitContext parent, LambdaExpression lambda) { this.context = context; this.parent = parent; this.lambda = lambda; this.hoisted = context.GetHoistedLocals (lambda); method = new DynamicMethod ( "lambda_method", lambda.GetReturnType (), CreateParameterTypes (lambda.Parameters), typeof (ExecutionScope), true); ig = method.GetILGenerator (); } public void Emit () { if (HasHoistedLocals) EmitStoreHoistedLocals (); lambda.EmitBody (this); } static Type [] CreateParameterTypes (IList parameters) { var types = new Type [parameters.Count + 1]; types [0] = typeof (ExecutionScope); for (int i = 0; i < parameters.Count; i++) types [i + 1] = parameters [i].Type; return types; } public bool IsLocalParameter (ParameterExpression parameter, ref int position) { position = lambda.Parameters.IndexOf (parameter); if (position > -1) { position++; return true; } return false; } public Delegate CreateDelegate (ExecutionScope scope) { return method.CreateDelegate (lambda.Type, scope); } public void Emit (Expression expression) { expression.Emit (this); } public LocalBuilder EmitStored (Expression expression) { var local = ig.DeclareLocal (expression.Type); expression.Emit (this); ig.Emit (OpCodes.Stloc, local); return local; } public void EmitLoadAddress (Expression expression) { ig.Emit (OpCodes.Ldloca, EmitStored (expression)); } public void EmitLoadEnum (Expression expression) { expression.Emit (this); ig.Emit (OpCodes.Box, expression.Type); } public void EmitLoadEnum (LocalBuilder local) { ig.Emit (OpCodes.Ldloc, local); ig.Emit (OpCodes.Box, local.LocalType); } public void EmitLoadSubject (Expression expression) { if (expression.Type.IsEnum) { EmitLoadEnum (expression); return; } if (expression.Type.IsValueType) { EmitLoadAddress (expression); return; } Emit (expression); } public void EmitLoadSubject (LocalBuilder local) { if (local.LocalType.IsEnum) { EmitLoadEnum (local); return; } if (local.LocalType.IsValueType) { EmitLoadAddress (local); return; } EmitLoad (local); } public void EmitLoadAddress (LocalBuilder local) { ig.Emit (OpCodes.Ldloca, local); } public void EmitLoad (LocalBuilder local) { ig.Emit (OpCodes.Ldloc, local); } public void EmitCall (LocalBuilder local, IList arguments, MethodInfo method) { EmitLoadSubject (local); EmitArguments (method, arguments); EmitCall (method); } public void EmitCall (LocalBuilder local, MethodInfo method) { EmitLoadSubject (local); EmitCall (method); } public void EmitCall (Expression expression, MethodInfo method) { if (!method.IsStatic) EmitLoadSubject (expression); EmitCall (method); } public void EmitCall (Expression expression, IList arguments, MethodInfo method) { if (!method.IsStatic) EmitLoadSubject (expression); EmitArguments (method, arguments); EmitCall (method); } void EmitArguments (MethodInfo method, IList arguments) { var parameters = method.GetParameters (); for (int i = 0; i < parameters.Length; i++) { var parameter = parameters [i]; var argument = arguments [i]; if (parameter.ParameterType.IsByRef) { ig.Emit (OpCodes.Ldloca, EmitStored (argument)); continue; } Emit (arguments [i]); } } public void EmitCall (MethodInfo method) { ig.Emit ( method.IsVirtual ? OpCodes.Callvirt : OpCodes.Call, method); } public void EmitNullableHasValue (LocalBuilder local) { EmitCall (local, "get_HasValue"); } public void EmitNullableInitialize (LocalBuilder local) { ig.Emit (OpCodes.Ldloca, local); ig.Emit (OpCodes.Initobj, local.LocalType); ig.Emit (OpCodes.Ldloc, local); } public void EmitNullableGetValue (LocalBuilder local) { EmitCall (local, "get_Value"); } public void EmitNullableGetValueOrDefault (LocalBuilder local) { EmitCall (local, "GetValueOrDefault"); } void EmitCall (LocalBuilder local, string method_name) { EmitCall (local, local.LocalType.GetMethod (method_name, Type.EmptyTypes)); } public void EmitNullableNew (Type of) { ig.Emit (OpCodes.Newobj, of.GetConstructor (new [] { of.GetFirstGenericArgument () })); } public void EmitCollection (IEnumerable collection) where T : Expression { foreach (var expression in collection) expression.Emit (this); } public void EmitCollection (IEnumerable initializers, LocalBuilder local) { foreach (var initializer in initializers) initializer.Emit (this, local); } public void EmitCollection (IEnumerable bindings, LocalBuilder local) { foreach (var binding in bindings) binding.Emit (this, local); } public void EmitIsInst (Expression expression, Type candidate) { expression.Emit (this); var type = expression.Type; if (type.IsValueType) ig.Emit (OpCodes.Box, type); ig.Emit (OpCodes.Isinst, candidate); } public void EmitScope () { ig.Emit (OpCodes.Ldarg_0); } public void EmitReadGlobal (object global) { EmitReadGlobal (global, global.GetType ()); } public void EmitLoadGlobals () { EmitScope (); ig.Emit (OpCodes.Ldfld, typeof (ExecutionScope).GetField ("Globals")); } public void EmitReadGlobal (object global, Type type) { EmitLoadGlobals (); ig.Emit (OpCodes.Ldc_I4, AddGlobal (global, type)); ig.Emit (OpCodes.Ldelem, typeof (object)); EmitLoadStrongBoxValue (type); } public void EmitLoadStrongBoxValue (Type type) { var strongbox = type.MakeStrongBoxType (); ig.Emit (OpCodes.Isinst, strongbox); ig.Emit (OpCodes.Ldfld, strongbox.GetField ("Value")); } int AddGlobal (object value, Type type) { return context.AddGlobal (CreateStrongBox (value, type)); } public void EmitCreateDelegate (LambdaExpression lambda) { EmitScope (); ig.Emit (OpCodes.Ldc_I4, AddChildContext (lambda)); if (hoisted_store != null) ig.Emit (OpCodes.Ldloc, hoisted_store); else ig.Emit (OpCodes.Ldnull); ig.Emit (OpCodes.Callvirt, typeof (ExecutionScope).GetMethod ("CreateDelegate")); ig.Emit (OpCodes.Castclass, lambda.Type); } void EmitStoreHoistedLocals () { EmitHoistedLocalsStore (); for (int i = 0; i < hoisted.Count; i++) EmitStoreHoistedLocal (i, hoisted [i]); } void EmitStoreHoistedLocal (int position, ParameterExpression parameter) { ig.Emit (OpCodes.Ldloc, hoisted_store); ig.Emit (OpCodes.Ldc_I4, position); parameter.Emit (this); EmitCreateStrongBox (parameter.Type); ig.Emit (OpCodes.Stelem, typeof (object)); } public void EmitLoadHoistedLocalsStore () { ig.Emit (OpCodes.Ldloc, hoisted_store); } void EmitCreateStrongBox (Type type) { ig.Emit (OpCodes.Newobj, type.MakeStrongBoxType ().GetConstructor (new [] { type })); } void EmitHoistedLocalsStore () { EmitScope (); hoisted_store = ig.DeclareLocal (typeof (object [])); ig.Emit (OpCodes.Callvirt, typeof (ExecutionScope).GetMethod ("CreateHoistedLocals")); ig.Emit (OpCodes.Stloc, hoisted_store); } public void EmitLoadLocals () { ig.Emit (OpCodes.Ldfld, typeof (ExecutionScope).GetField ("Locals")); } public void EmitParentScope () { ig.Emit (OpCodes.Ldfld, typeof (ExecutionScope).GetField ("Parent")); } public void EmitIsolateExpression () { ig.Emit (OpCodes.Callvirt, typeof (ExecutionScope).GetMethod ("IsolateExpression")); } public int IndexOfHoistedLocal (ParameterExpression parameter) { if (!HasHoistedLocals) return -1; return hoisted.IndexOf (parameter); } public bool IsHoistedLocal (ParameterExpression parameter, ref int level, ref int position) { if (parent == null) return false; if (parent.hoisted != null) { position = parent.hoisted.IndexOf (parameter); if (position > -1) return true; } level++; return parent.IsHoistedLocal (parameter, ref level, ref position); } int AddChildContext (LambdaExpression lambda) { return context.AddCompilationUnit (this, lambda); } static object CreateStrongBox (object value, Type type) { return Activator.CreateInstance ( type.MakeStrongBoxType (), value); } } } #endif