ExpressionTransformer.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. //
  2. // ExpressionTransformer.cs
  3. //
  4. // Authors:
  5. // Roei Erez ([email protected])
  6. // Jb Evain ([email protected])
  7. //
  8. // Copyright (C) 2007 Novell, Inc (http://www.novell.com)
  9. //
  10. // Permission is hereby granted, free of charge, to any person obtaining
  11. // a copy of this software and associated documentation files (the
  12. // "Software"), to deal in the Software without restriction, including
  13. // without limitation the rights to use, copy, modify, merge, publish,
  14. // distribute, sublicense, and/or sell copies of the Software, and to
  15. // permit persons to whom the Software is furnished to do so, subject to
  16. // the following conditions:
  17. //
  18. // The above copyright notice and this permission notice shall be
  19. // included in all copies or substantial portions of the Software.
  20. //
  21. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  22. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
  23. // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  24. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
  25. // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
  26. // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
  27. // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
  28. //
  29. using System;
  30. using System.Collections.ObjectModel;
  31. using System.Collections.Generic;
  32. using System.Linq.Expressions;
  33. namespace System.Linq.Expressions {
  34. abstract class ExpressionTransformer {
  35. public Expression Transform (Expression e)
  36. {
  37. return Visit (e);
  38. }
  39. protected virtual Expression Visit (Expression expression)
  40. {
  41. if (expression == null)
  42. return null;
  43. switch (expression.NodeType) {
  44. case ExpressionType.Negate:
  45. case ExpressionType.NegateChecked:
  46. case ExpressionType.Not:
  47. case ExpressionType.Convert:
  48. case ExpressionType.ConvertChecked:
  49. case ExpressionType.ArrayLength:
  50. case ExpressionType.Quote:
  51. case ExpressionType.TypeAs:
  52. case ExpressionType.UnaryPlus:
  53. return VisitUnary ((UnaryExpression) expression);
  54. case ExpressionType.Add:
  55. case ExpressionType.AddChecked:
  56. case ExpressionType.Subtract:
  57. case ExpressionType.SubtractChecked:
  58. case ExpressionType.Multiply:
  59. case ExpressionType.MultiplyChecked:
  60. case ExpressionType.Divide:
  61. case ExpressionType.Modulo:
  62. case ExpressionType.Power:
  63. case ExpressionType.And:
  64. case ExpressionType.AndAlso:
  65. case ExpressionType.Or:
  66. case ExpressionType.OrElse:
  67. case ExpressionType.LessThan:
  68. case ExpressionType.LessThanOrEqual:
  69. case ExpressionType.GreaterThan:
  70. case ExpressionType.GreaterThanOrEqual:
  71. case ExpressionType.Equal:
  72. case ExpressionType.NotEqual:
  73. case ExpressionType.Coalesce:
  74. case ExpressionType.ArrayIndex:
  75. case ExpressionType.RightShift:
  76. case ExpressionType.LeftShift:
  77. case ExpressionType.ExclusiveOr:
  78. return VisitBinary ((BinaryExpression) expression);
  79. case ExpressionType.TypeIs:
  80. return VisitTypeIs ((TypeBinaryExpression) expression);
  81. case ExpressionType.Conditional:
  82. return VisitConditional ((ConditionalExpression) expression);
  83. case ExpressionType.Constant:
  84. return VisitConstant ((ConstantExpression) expression);
  85. case ExpressionType.Parameter:
  86. return VisitParameter ((ParameterExpression) expression);
  87. case ExpressionType.MemberAccess:
  88. return VisitMemberAccess ((MemberExpression) expression);
  89. case ExpressionType.Call:
  90. return VisitMethodCall ((MethodCallExpression) expression);
  91. case ExpressionType.Lambda:
  92. return VisitLambda ((LambdaExpression) expression);
  93. case ExpressionType.New:
  94. return VisitNew ((NewExpression) expression);
  95. case ExpressionType.NewArrayInit:
  96. case ExpressionType.NewArrayBounds:
  97. return VisitNewArray ((NewArrayExpression) expression);
  98. case ExpressionType.Invoke:
  99. return VisitInvocation ((InvocationExpression) expression);
  100. case ExpressionType.MemberInit:
  101. return VisitMemberInit ((MemberInitExpression) expression);
  102. case ExpressionType.ListInit:
  103. return VisitListInit ((ListInitExpression) expression);
  104. default:
  105. throw new ArgumentException (string.Format ("Unhandled expression type: '{0}'", expression.NodeType));
  106. }
  107. }
  108. protected virtual MemberBinding VisitBinding (MemberBinding binding)
  109. {
  110. switch (binding.BindingType) {
  111. case MemberBindingType.Assignment:
  112. return VisitMemberAssignment ((MemberAssignment) binding);
  113. case MemberBindingType.MemberBinding:
  114. return VisitMemberMemberBinding ((MemberMemberBinding) binding);
  115. case MemberBindingType.ListBinding:
  116. return VisitMemberListBinding ((MemberListBinding) binding);
  117. default:
  118. throw new ArgumentException (string.Format ("Unhandled binding type '{0}'", binding.BindingType));
  119. }
  120. }
  121. protected virtual ElementInit VisitElementInitializer (ElementInit initializer)
  122. {
  123. ReadOnlyCollection<Expression> transformed = VisitExpressionList (initializer.Arguments);
  124. if (transformed != initializer.Arguments)
  125. return Expression.ElementInit (initializer.AddMethod, transformed);
  126. return initializer;
  127. }
  128. protected virtual UnaryExpression VisitUnary (UnaryExpression unary)
  129. {
  130. Expression transformedOperand = Visit (unary.Operand);
  131. if (transformedOperand != unary.Operand)
  132. return Expression.MakeUnary (unary.NodeType, transformedOperand, unary.Type, unary.Method);
  133. return unary;
  134. }
  135. protected virtual BinaryExpression VisitBinary (BinaryExpression binary)
  136. {
  137. Expression left = Visit (binary.Left);
  138. Expression right = Visit (binary.Right);
  139. LambdaExpression conversion = VisitLambda (binary.Conversion);
  140. if (left != binary.Left || right != binary.Right || conversion != binary.Conversion)
  141. return Expression.MakeBinary (binary.NodeType, left, right, binary.IsLiftedToNull, binary.Method, conversion);
  142. return binary;
  143. }
  144. protected virtual TypeBinaryExpression VisitTypeIs (TypeBinaryExpression type)
  145. {
  146. Expression inner = Visit (type.Expression);
  147. if (inner != type.Expression)
  148. return Expression.TypeIs (inner, type.TypeOperand);
  149. return type;
  150. }
  151. protected virtual ConstantExpression VisitConstant (ConstantExpression constant)
  152. {
  153. return constant;
  154. }
  155. protected virtual ConditionalExpression VisitConditional (ConditionalExpression conditional)
  156. {
  157. Expression test = Visit (conditional.Test);
  158. Expression ifTrue = Visit (conditional.IfTrue);
  159. Expression ifFalse = Visit (conditional.IfFalse);
  160. if (test != conditional.Test || ifTrue != conditional.IfTrue || ifFalse != conditional.IfFalse)
  161. return Expression.Condition (test, ifTrue, ifFalse);
  162. return conditional;
  163. }
  164. protected virtual ParameterExpression VisitParameter (ParameterExpression parameter)
  165. {
  166. return parameter;
  167. }
  168. protected virtual MemberExpression VisitMemberAccess (MemberExpression member)
  169. {
  170. Expression memberExp = Visit (member.Expression);
  171. if (memberExp != member.Expression)
  172. return Expression.MakeMemberAccess (memberExp, member.Member);
  173. return member;
  174. }
  175. protected virtual MethodCallExpression VisitMethodCall (MethodCallExpression methodCall)
  176. {
  177. Expression instance = Visit (methodCall.Object);
  178. ReadOnlyCollection<Expression> args = VisitExpressionList (methodCall.Arguments);
  179. if (instance != methodCall.Object || args != methodCall.Arguments)
  180. return Expression.Call (instance, methodCall.Method, args);
  181. return methodCall;
  182. }
  183. protected virtual ReadOnlyCollection<Expression> VisitExpressionList (ReadOnlyCollection<Expression> list)
  184. {
  185. return VisitList<Expression> (list, Visit);
  186. }
  187. ReadOnlyCollection<T> VisitList<T> (ReadOnlyCollection<T> original, Func<T, T> selector) where T :class
  188. {
  189. List<T> list = null;
  190. for (int i = 0; i < original.Count; i++) {
  191. var element = selector (original [i]);
  192. if (list != null) {
  193. list.Add (element);
  194. continue;
  195. }
  196. if (!EqualityComparer<T>.Default.Equals (element, original [i])) {
  197. list = new List<T> (original.Count);
  198. for (int j = 0; j < i; j++)
  199. list.Add (original [j]);
  200. list.Add(element);
  201. }
  202. }
  203. return list != null ? list.ToReadOnlyCollection () : original;
  204. }
  205. protected virtual MemberAssignment VisitMemberAssignment (MemberAssignment assignment)
  206. {
  207. Expression inner = Visit (assignment.Expression);
  208. if (inner != assignment.Expression)
  209. return Expression.Bind (assignment.Member, inner);
  210. return assignment;
  211. }
  212. protected virtual MemberMemberBinding VisitMemberMemberBinding (MemberMemberBinding binding)
  213. {
  214. ReadOnlyCollection<MemberBinding> bindingExp = VisitBindingList (binding.Bindings);
  215. if (bindingExp != binding.Bindings)
  216. return Expression.MemberBind (binding.Member, bindingExp);
  217. return binding;
  218. }
  219. protected virtual MemberListBinding VisitMemberListBinding (MemberListBinding binding)
  220. {
  221. ReadOnlyCollection<ElementInit> initializers =
  222. VisitElementInitializerList (binding.Initializers);
  223. if (initializers != binding.Initializers)
  224. return Expression.ListBind (binding.Member, initializers);
  225. return binding;
  226. }
  227. protected virtual ReadOnlyCollection<MemberBinding> VisitBindingList (ReadOnlyCollection<MemberBinding> list)
  228. {
  229. return VisitList<MemberBinding> (list, VisitBinding);
  230. }
  231. protected virtual ReadOnlyCollection<ElementInit> VisitElementInitializerList (ReadOnlyCollection<ElementInit> list)
  232. {
  233. return VisitList<ElementInit> (list, VisitElementInitializer);
  234. }
  235. protected virtual LambdaExpression VisitLambda (LambdaExpression lambda)
  236. {
  237. Expression body = Visit (lambda.Body);
  238. ReadOnlyCollection<ParameterExpression> parameters =
  239. VisitList<ParameterExpression> (lambda.Parameters, VisitParameter);
  240. if (body != lambda.Body || parameters != lambda.Parameters)
  241. return Expression.Lambda (body, parameters.ToArray());
  242. return lambda;
  243. }
  244. protected virtual NewExpression VisitNew (NewExpression nex)
  245. {
  246. ReadOnlyCollection<Expression> args = VisitList (nex.Arguments, Visit);
  247. if (args != nex.Arguments)
  248. return Expression.New (nex.Constructor, args);
  249. return nex;
  250. }
  251. protected virtual MemberInitExpression VisitMemberInit (MemberInitExpression init)
  252. {
  253. NewExpression newExp = VisitNew (init.NewExpression);
  254. ReadOnlyCollection<MemberBinding> bindings = VisitBindingList (init.Bindings);
  255. if (newExp != init.NewExpression || bindings != init.Bindings)
  256. return Expression.MemberInit (newExp, bindings);
  257. return init;
  258. }
  259. protected virtual ListInitExpression VisitListInit (ListInitExpression init)
  260. {
  261. NewExpression newExp = VisitNew (init.NewExpression);
  262. ReadOnlyCollection<ElementInit> initializers = VisitElementInitializerList (init.Initializers);
  263. if (newExp != init.NewExpression || initializers != init.Initializers)
  264. return Expression.ListInit (newExp, initializers.ToArray());
  265. return init;
  266. }
  267. protected virtual NewArrayExpression VisitNewArray (NewArrayExpression newArray)
  268. {
  269. ReadOnlyCollection<Expression> expressions = VisitExpressionList (newArray.Expressions);
  270. if (expressions != newArray.Expressions) {
  271. if (newArray.NodeType == ExpressionType.NewArrayBounds)
  272. return Expression.NewArrayBounds (newArray.Type, expressions);
  273. else
  274. return Expression.NewArrayInit (newArray.Type, expressions);
  275. }
  276. return newArray;
  277. }
  278. protected virtual InvocationExpression VisitInvocation (InvocationExpression invocation)
  279. {
  280. ReadOnlyCollection<Expression> args = VisitExpressionList (invocation.Arguments);
  281. Expression invocationExp = Visit (invocation.Expression);
  282. if (args != invocation.Arguments || invocationExp != invocation.Expression)
  283. return Expression.Invoke (invocationExp, args);
  284. return invocation;
  285. }
  286. }
  287. }