/Runtime/Microsoft.Dynamic/Actions/ComboActionRewriter.cs

http://github.com/IronLanguages/main · C# · 209 lines · 133 code · 32 blank · 44 comment · 38 complexity · f3c1e0841ad485ec7f593949852479b7 MD5 · raw file

  1. /* ****************************************************************************
  2. *
  3. * Copyright (c) Microsoft Corporation.
  4. *
  5. * This source code is subject to terms and conditions of the Apache License, Version 2.0. A
  6. * copy of the license can be found in the License.html file at the root of this distribution. If
  7. * you cannot locate the Apache License, Version 2.0, please send an email to
  8. * dlr@microsoft.com. By using this source code in any fashion, you are agreeing to be bound
  9. * by the terms of the Apache License, Version 2.0.
  10. *
  11. * You must not remove this notice, or any other, from this software.
  12. *
  13. *
  14. * ***************************************************************************/
  15. #if FEATURE_CORE_DLR
  16. using System.Linq.Expressions;
  17. #else
  18. using Microsoft.Scripting.Ast;
  19. #endif
  20. using System;
  21. using System.Collections.Generic;
  22. using System.Diagnostics;
  23. using System.Dynamic;
  24. using System.Reflection;
  25. using Microsoft.Scripting.Utils;
  26. using AstUtils = Microsoft.Scripting.Ast.Utils;
  27. namespace Microsoft.Scripting.Actions {
  28. /// <summary>
  29. /// A tree rewriter which will find dynamic sites which consume dynamic sites and
  30. /// turn them into a single combo dynamic site. The combo dynamic site will then run the
  31. /// individual meta binders and produce the resulting code in a single dynamic site.
  32. /// </summary>
  33. public class ComboActionRewriter : DynamicExpressionVisitor {
  34. /// <summary>
  35. /// A reducible node which we use to generate the combo dynamic sites. Each time we encounter
  36. /// a dynamic site we replace it with a ComboDynamicSiteExpression. When a child of a dynamic site
  37. /// turns out to be a ComboDynamicSiteExpression we will then merge the child with the parent updating
  38. /// the binding mapping info. If any of the inputs cause side effects then we'll stop the combination.
  39. /// </summary>
  40. class ComboDynamicSiteExpression : Expression {
  41. private readonly Expression[] _inputs;
  42. private readonly List<BinderMappingInfo> _binders;
  43. private readonly Type _type;
  44. public ComboDynamicSiteExpression(Type type, List<BinderMappingInfo> binders, Expression[] inputs) {
  45. _binders = binders;
  46. _inputs = inputs;
  47. _type = type;
  48. }
  49. public override bool CanReduce {
  50. get { return true; }
  51. }
  52. public sealed override Type Type {
  53. get { return _type; }
  54. }
  55. public sealed override ExpressionType NodeType {
  56. get { return ExpressionType.Extension; }
  57. }
  58. public Expression[] Inputs {
  59. get {
  60. return _inputs;
  61. }
  62. }
  63. public List<BinderMappingInfo> Binders {
  64. get {
  65. return _binders;
  66. }
  67. }
  68. public override Expression Reduce() {
  69. // we just reduce to a simple DynamicExpression
  70. return DynamicExpression.Dynamic(
  71. new ComboBinder(_binders),
  72. Type,
  73. _inputs
  74. );
  75. }
  76. }
  77. protected override Expression VisitDynamic(DynamicExpression node) {
  78. DynamicMetaObjectBinder metaBinder = node.Binder as DynamicMetaObjectBinder;
  79. if (metaBinder == null) {
  80. // don't rewrite non meta-binder nodes, we can't compose them
  81. return node;
  82. }
  83. // gather the real arguments for the new dynamic site node
  84. var args = node.Arguments;
  85. bool foundSideEffectingArgs = false;
  86. List<Expression> inputs = new List<Expression>();
  87. // parameter mapping is 1 List<ComboParameterMappingInfo> for each meta binder, the inner list
  88. // contains the mapping info for each particular binder
  89. List<BinderMappingInfo> binders = new List<BinderMappingInfo>();
  90. List<ParameterMappingInfo> myInfo = new List<ParameterMappingInfo>();
  91. int actionCount = 0;
  92. for (int i = 0; i < args.Count; i++) {
  93. Expression e = args[i];
  94. if (!foundSideEffectingArgs) {
  95. // attempt to combine the arguments...
  96. Expression rewritten = Visit(e);
  97. ComboDynamicSiteExpression combo = rewritten as ComboDynamicSiteExpression;
  98. ConstantExpression ce;
  99. if (combo != null) {
  100. // an action expression we can combine with our own expression
  101. // remember how many actions we have so far - if any of our children consume
  102. // actions their offset is bumped up
  103. int baseActionCount = actionCount;
  104. foreach (BinderMappingInfo comboInfo in combo.Binders) {
  105. List<ParameterMappingInfo> newInfo = new List<ParameterMappingInfo>();
  106. foreach (ParameterMappingInfo info in comboInfo.MappingInfo) {
  107. if (info.IsParameter) {
  108. // all of the inputs from the child now become ours
  109. newInfo.Add(ParameterMappingInfo.Parameter(inputs.Count));
  110. inputs.Add(combo.Inputs[info.ParameterIndex]);
  111. } else if (info.IsAction) {
  112. newInfo.Add(ParameterMappingInfo.Action(info.ActionIndex + baseActionCount));
  113. actionCount++;
  114. } else {
  115. Debug.Assert(info.Constant != null);
  116. // constants can just flow through
  117. newInfo.Add(info);
  118. }
  119. }
  120. binders.Add(new BinderMappingInfo(comboInfo.Binder, newInfo));
  121. }
  122. myInfo.Add(ParameterMappingInfo.Action(actionCount++));
  123. } else if ((ce = rewritten as ConstantExpression) != null) {
  124. // we can hoist the constant into the combo
  125. myInfo.Add(ParameterMappingInfo.Fixed(ce));
  126. } else if (IsSideEffectFree(rewritten)) {
  127. // we can treat this as an input parameter
  128. myInfo.Add(ParameterMappingInfo.Parameter(inputs.Count));
  129. inputs.Add(rewritten);
  130. } else {
  131. // this argument is doing something we don't understand - we have to leave
  132. // it as is (an input we consume) and all the remaining arguments need to be
  133. // evaluated normally as this could have side effects on them.
  134. foundSideEffectingArgs = true;
  135. myInfo.Add(ParameterMappingInfo.Parameter(inputs.Count));
  136. inputs.Add(e);
  137. }
  138. } else {
  139. // we've already seen an argument which may have side effects, don't do
  140. // any more combinations.
  141. myInfo.Add(ParameterMappingInfo.Parameter(inputs.Count));
  142. inputs.Add(e);
  143. }
  144. }
  145. binders.Add(new BinderMappingInfo(metaBinder, myInfo));
  146. // TODO: Remove any duplicate inputs (e.g. locals being fed in multiple times)
  147. return new ComboDynamicSiteExpression(node.Type, binders, inputs.ToArray());
  148. }
  149. private bool IsSideEffectFree(Expression rewritten) {
  150. if (rewritten is ParameterExpression) {
  151. return true;
  152. }
  153. if (rewritten.NodeType == ExpressionType.TypeIs) {
  154. return IsSideEffectFree(((UnaryExpression)rewritten).Operand);
  155. }
  156. BinaryExpression be = rewritten as BinaryExpression;
  157. if (be != null) {
  158. if (be.Method == null && IsSideEffectFree(be.Left) && IsSideEffectFree(be.Right)) {
  159. return true;
  160. }
  161. }
  162. MethodCallExpression mc = rewritten as MethodCallExpression;
  163. if (mc != null && mc.Method != null) {
  164. return mc.Method.IsDefined(typeof(NoSideEffectsAttribute), false);
  165. }
  166. ConditionalExpression ce = rewritten as ConditionalExpression;
  167. if (ce != null) {
  168. return IsSideEffectFree(ce.Test) && IsSideEffectFree(ce.IfTrue) && IsSideEffectFree(ce.IfFalse);
  169. }
  170. MemberExpression me = rewritten as MemberExpression;
  171. if (me != null && me.Member is System.Reflection.FieldInfo) {
  172. return false;
  173. }
  174. return false;
  175. }
  176. }
  177. }