| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 | // Licensed to the .NET Foundation under one or more agreements.// The .NET Foundation licenses this file to you under the Apache 2.0 License.// See the LICENSE file in the project root for more information. #if NO_EXPRESSIONVISITORusing System;using System.Collections.Generic;using System.Collections.ObjectModel;using System.Reflection;using System.Linq.Expressions;namespace System.Reactive.Linq{    internal abstract class ExpressionVisitor    {        internal ExpressionVisitor()        {        }        public virtual Expression Visit(Expression exp)        {            if (exp == null)                return exp;            switch (exp.NodeType)            {                case ExpressionType.UnaryPlus:                case ExpressionType.Negate:                case ExpressionType.NegateChecked:                case ExpressionType.Not:                case ExpressionType.Convert:                case ExpressionType.ConvertChecked:                case ExpressionType.ArrayLength:                case ExpressionType.Quote:                case ExpressionType.TypeAs:                    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.Modulo:                case ExpressionType.Power:                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 InvalidOperationException();            }        }        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 InvalidOperationException();            }        }        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)        {            List<Expression> list = null;            for (int i = 0, n = original.Count; i < n; i++)            {                Expression p = this.Visit(original[i]);                if (list != null)                {                    list.Add(p);                }                else if (p != original[i])                {                    list = new List<Expression>(n);                    for (int j = 0; j < i; j++)                    {                        list.Add(original[j]);                    }                    list.Add(p);                }            }            if (list != null)                return new ReadOnlyCollection<Expression>(list);            return original;        }        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)        {            List<MemberBinding> list = null;            for (int i = 0, n = original.Count; i < n; i++)            {                MemberBinding b = this.VisitBinding(original[i]);                if (list != null)                {                    list.Add(b);                }                else if (b != original[i])                {                    list = new List<MemberBinding>(n);                    for (int j = 0; j < i; j++)                    {                        list.Add(original[j]);                    }                    list.Add(b);                }            }            if (list != null)                return list;            return original;        }        protected virtual IEnumerable<ElementInit> VisitElementInitializerList(ReadOnlyCollection<ElementInit> original)        {            List<ElementInit> list = null;            for (int i = 0, n = original.Count; i < n; i++)            {                ElementInit init = this.VisitElementInitializer(original[i]);                if (list != null)                {                    list.Add(init);                }                else if (init != original[i])                {                    list = new List<ElementInit>(n);                    for (int j = 0; j < i; j++)                    {                        list.Add(original[j]);                    }                    list.Add(init);                }            }            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;        }    }}#endif
 |