| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- //
- // ExpressionTransformer.cs
- //
- // Authors:
- // Roei Erez ([email protected])
- // Jb Evain ([email protected])
- //
- // Copyright (C) 2007 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.
- //
- using System;
- using System.Collections.ObjectModel;
- using System.Collections.Generic;
- using System.Linq.Expressions;
- namespace System.Linq.Expressions {
- abstract class ExpressionTransformer {
- public Expression Transform (Expression expression)
- {
- return Visit (expression);
- }
- protected virtual Expression Visit (Expression exp)
- {
- if (exp == null) return exp;
- switch (exp.NodeType) {
- case ExpressionType.Negate:
- case ExpressionType.NegateChecked:
- case ExpressionType.Not:
- case ExpressionType.Convert:
- case ExpressionType.ConvertChecked:
- case ExpressionType.ArrayLength:
- case ExpressionType.Quote:
- case ExpressionType.TypeAs:
- case ExpressionType.UnaryPlus:
- return this.VisitUnary ((UnaryExpression) exp);
- case ExpressionType.Add:
- case ExpressionType.AddChecked:
- case ExpressionType.Subtract:
- case ExpressionType.SubtractChecked:
- case ExpressionType.Multiply:
- case ExpressionType.MultiplyChecked:
- case ExpressionType.Divide:
- case ExpressionType.Power:
- case ExpressionType.Modulo:
- case ExpressionType.And:
- case ExpressionType.AndAlso:
- case ExpressionType.Or:
- case ExpressionType.OrElse:
- case ExpressionType.LessThan:
- case ExpressionType.LessThanOrEqual:
- case ExpressionType.GreaterThan:
- case ExpressionType.GreaterThanOrEqual:
- case ExpressionType.Equal:
- case ExpressionType.NotEqual:
- case ExpressionType.Coalesce:
- case ExpressionType.ArrayIndex:
- case ExpressionType.RightShift:
- case ExpressionType.LeftShift:
- case ExpressionType.ExclusiveOr:
- return this.VisitBinary ((BinaryExpression) exp);
- case ExpressionType.TypeIs:
- return this.VisitTypeIs ((TypeBinaryExpression) exp);
- case ExpressionType.Conditional:
- return this.VisitConditional ((ConditionalExpression) exp);
- case ExpressionType.Constant:
- return this.VisitConstant ((ConstantExpression) exp);
- case ExpressionType.Parameter:
- return this.VisitParameter ((ParameterExpression) exp);
- case ExpressionType.MemberAccess:
- return this.VisitMemberAccess ((MemberExpression) exp);
- case ExpressionType.Call:
- return this.VisitMethodCall ((MethodCallExpression) exp);
- case ExpressionType.Lambda:
- return this.VisitLambda ((LambdaExpression) exp);
- case ExpressionType.New:
- return this.VisitNew ((NewExpression) exp);
- case ExpressionType.NewArrayInit:
- case ExpressionType.NewArrayBounds:
- return this.VisitNewArray ((NewArrayExpression) exp);
- case ExpressionType.Invoke:
- return this.VisitInvocation ((InvocationExpression) exp);
- case ExpressionType.MemberInit:
- return this.VisitMemberInit ((MemberInitExpression) exp);
- case ExpressionType.ListInit:
- return this.VisitListInit ((ListInitExpression) exp);
- default:
- throw new Exception (string.Format ("Unhandled expression type: '{0}'", exp.NodeType));
- }
- }
- protected virtual MemberBinding VisitBinding (MemberBinding binding)
- {
- switch (binding.BindingType) {
- case MemberBindingType.Assignment:
- return this.VisitMemberAssignment ((MemberAssignment) binding);
- case MemberBindingType.MemberBinding:
- return this.VisitMemberMemberBinding ((MemberMemberBinding) binding);
- case MemberBindingType.ListBinding:
- return this.VisitMemberListBinding ((MemberListBinding) binding);
- default:
- throw new Exception (string.Format ("Unhandled binding type '{0}'", binding.BindingType));
- }
- }
- protected virtual ElementInit VisitElementInitializer (ElementInit initializer)
- {
- ReadOnlyCollection<Expression> arguments = this.VisitExpressionList (initializer.Arguments);
- if (arguments != initializer.Arguments) return Expression.ElementInit (initializer.AddMethod, arguments);
- return initializer;
- }
- protected virtual Expression VisitUnary (UnaryExpression u)
- {
- Expression operand = this.Visit (u.Operand);
- if (operand != u.Operand) return Expression.MakeUnary (u.NodeType, operand, u.Type, u.Method);
- return u;
- }
- protected virtual Expression VisitBinary (BinaryExpression b)
- {
- Expression left = this.Visit (b.Left);
- Expression right = this.Visit (b.Right);
- Expression conversion = this.Visit (b.Conversion);
- if (left != b.Left || right != b.Right || conversion != b.Conversion) {
- if (b.NodeType == ExpressionType.Coalesce && b.Conversion != null) {
- return Expression.Coalesce (left, right, conversion as LambdaExpression);
- } else {
- return Expression.MakeBinary (b.NodeType, left, right, b.IsLiftedToNull, b.Method);
- }
- }
- return b;
- }
- protected virtual Expression VisitTypeIs (TypeBinaryExpression b)
- {
- Expression expr = this.Visit (b.Expression);
- if (expr != b.Expression) {
- return Expression.TypeIs (expr, b.TypeOperand);
- }
- return b;
- }
- protected virtual Expression VisitConstant (ConstantExpression c)
- {
- return c;
- }
- protected virtual Expression VisitConditional (ConditionalExpression c)
- {
- Expression test = this.Visit (c.Test);
- Expression ifTrue = this.Visit (c.IfTrue);
- Expression ifFalse = this.Visit (c.IfFalse);
- if (test != c.Test || ifTrue != c.IfTrue || ifFalse != c.IfFalse) {
- return Expression.Condition (test, ifTrue, ifFalse);
- }
- return c;
- }
- protected virtual Expression VisitParameter (ParameterExpression p)
- {
- return p;
- }
- protected virtual Expression VisitMemberAccess (MemberExpression m)
- {
- Expression exp = this.Visit (m.Expression);
- if (exp != m.Expression) {
- return Expression.MakeMemberAccess (exp, m.Member);
- }
- return m;
- }
- protected virtual Expression VisitMethodCall (MethodCallExpression m)
- {
- Expression obj = this.Visit (m.Object);
- IEnumerable<Expression> args = this.VisitExpressionList (m.Arguments);
- if (obj != m.Object || args != m.Arguments) {
- return Expression.Call (obj, m.Method, args);
- }
- return m;
- }
- protected virtual ReadOnlyCollection<Expression> VisitExpressionList (ReadOnlyCollection<Expression> original)
- {
- var list = VisitList (original, Visit);
- if (list == null) return original;
- return new ReadOnlyCollection<Expression> (list);
- }
- protected virtual MemberAssignment VisitMemberAssignment (MemberAssignment assignment)
- {
- Expression e = this.Visit (assignment.Expression);
- if (e != assignment.Expression) return Expression.Bind (assignment.Member, e);
- return assignment;
- }
- protected virtual MemberMemberBinding VisitMemberMemberBinding (MemberMemberBinding binding)
- {
- IEnumerable<MemberBinding> bindings = this.VisitBindingList (binding.Bindings);
- if (bindings != binding.Bindings) return Expression.MemberBind (binding.Member, bindings);
- return binding;
- }
- protected virtual MemberListBinding VisitMemberListBinding (MemberListBinding binding)
- {
- IEnumerable<ElementInit> initializers = this.VisitElementInitializerList (binding.Initializers);
- if (initializers != binding.Initializers) return Expression.ListBind (binding.Member, initializers);
- return binding;
- }
- protected virtual IEnumerable<MemberBinding> VisitBindingList (ReadOnlyCollection<MemberBinding> original)
- {
- return VisitList (original, VisitBinding);
- }
- protected virtual IEnumerable<ElementInit> VisitElementInitializerList (ReadOnlyCollection<ElementInit> original)
- {
- return VisitList (original, VisitElementInitializer);
- }
- private IList<TElement> VisitList<TElement> (ReadOnlyCollection<TElement> original, Func<TElement, TElement> visit)
- {
- List<TElement> list = null;
- for (int i = 0, n = original.Count; i < n; i++) {
- TElement element = visit (original [i]);
- if (list != null) {
- list.Add (element);
- } else if (!EqualityComparer<TElement>.Default.Equals (element, original [i])) {
- list = new List<TElement> (n);
- for (int j = 0; j < i; j++) {
- list.Add (original [j]);
- }
- list.Add (element);
- }
- }
- if (list != null)
- return list;
- return original;
- }
- protected virtual Expression VisitLambda (LambdaExpression lambda)
- {
- Expression body = this.Visit (lambda.Body);
- if (body != lambda.Body) return Expression.Lambda (lambda.Type, body, lambda.Parameters);
- return lambda;
- }
- protected virtual NewExpression VisitNew (NewExpression nex)
- {
- IEnumerable<Expression> args = this.VisitExpressionList (nex.Arguments);
- if (args != nex.Arguments) {
- if (nex.Members != null)
- return Expression.New (nex.Constructor, args, nex.Members);
- else
- return Expression.New (nex.Constructor, args);
- }
- return nex;
- }
- protected virtual Expression VisitMemberInit (MemberInitExpression init)
- {
- NewExpression n = this.VisitNew (init.NewExpression);
- IEnumerable<MemberBinding> bindings = this.VisitBindingList (init.Bindings);
- if (n != init.NewExpression || bindings != init.Bindings) return Expression.MemberInit (n, bindings);
- return init;
- }
- protected virtual Expression VisitListInit (ListInitExpression init)
- {
- NewExpression n = this.VisitNew (init.NewExpression);
- IEnumerable<ElementInit> initializers = this.VisitElementInitializerList (init.Initializers);
- if (n != init.NewExpression || initializers != init.Initializers) return Expression.ListInit (n, initializers);
- return init;
- }
- protected virtual Expression VisitNewArray (NewArrayExpression na)
- {
- IEnumerable<Expression> exprs = this.VisitExpressionList (na.Expressions);
- if (exprs != na.Expressions) {
- if (na.NodeType == ExpressionType.NewArrayInit) {
- return Expression.NewArrayInit (na.Type.GetElementType (), exprs);
- } else {
- return Expression.NewArrayBounds (na.Type.GetElementType (), exprs);
- }
- }
- return na;
- }
- protected virtual Expression VisitInvocation (InvocationExpression iv)
- {
- IEnumerable<Expression> args = this.VisitExpressionList (iv.Arguments);
- Expression expr = this.Visit (iv.Expression);
- if (args != iv.Arguments || expr != iv.Expression) return Expression.Invoke (expr, args);
- return iv;
- }
- }
- }
|