ExpressionTransformer.cs 11 KB

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