File: System\ServiceModel\Dispatcher\QueryIntervalOp.cs
Project: ndp\cdf\src\WCF\ServiceModel\System.ServiceModel.csproj (System.ServiceModel)
//------------------------------------------------------------
// Copyright (c) Microsoft Corporation.  All rights reserved.
//------------------------------------------------------------
namespace System.ServiceModel.Dispatcher
{
    using System.Collections;
    using System.Collections.Generic;
    using System.Runtime;
 
    // Based on the Paper: The IBS Tree: A Datastructure for finding all intervals that overlap a point.
    // by Eric Hanson & Moez Chaabouni, Nov, 1994
 
    internal enum IntervalOp : byte
    {
        LessThan,
        LessThanEquals
    }
 
    internal class Interval
    {
        QueryBranch branch;
        double lowerBound;
        IntervalOp lowerOp;
        double upperBound;
        IntervalOp upperOp;
 
        // Converts expressions of the form:
        // x < 5 => -infinity <= x and x < 5
        // x <= 5 => -infinity <= x and x <= 5
        // x > 5 => 5 < x <= infinity
        // x >= 5 => 5 <= x <= infinity
        //
        // The variable is always to the left
        internal Interval(double literal, RelationOperator op)
        {
            this.lowerBound = double.MinValue;
            this.upperBound = double.MaxValue;
            this.lowerOp = IntervalOp.LessThanEquals;
            this.upperOp = IntervalOp.LessThanEquals;
 
            Fx.Assert(RelationOperator.Eq != op && RelationOperator.Ne != op, "");
            switch (op)
            {
                case RelationOperator.Lt:
                    this.upperBound = literal;
                    this.upperOp = IntervalOp.LessThan;
                    break;
                case RelationOperator.Le:
                    this.upperBound = literal;
                    break;
                case RelationOperator.Gt:
                    this.lowerBound = literal;
                    this.lowerOp = IntervalOp.LessThan;
                    break;
                case RelationOperator.Ge:
                    this.lowerBound = literal;
                    break;
            }
        }
 
#if NO
        internal Interval(double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            Fx.Assert(lowerBound <= upperBound, "");
 
            this.lowerBound = lowerBound;
            this.upperBound = upperBound;
            if (this.lowerBound == this.upperBound)
            {
                this.lowerOp = IntervalOp.LessThanEquals;
                this.upperOp = IntervalOp.LessThanEquals;
            }
            else
            {
                this.lowerOp = lowerOp;
                this.upperOp = upperOp;
            }
        }
#endif
        internal QueryBranch Branch
        {
            get
            {
                return this.branch;
            }
            set
            {
                this.branch = value;
            }
        }
 
        internal double LowerBound
        {
            get
            {
                return this.lowerBound;
            }
        }
 
        internal IntervalOp LowerOp
        {
            get
            {
                return this.lowerOp;
            }
        }
 
        internal double UpperBound
        {
            get
            {
                return this.upperBound;
            }
        }
 
        internal IntervalOp UpperOp
        {
            get
            {
                return this.upperOp;
            }
        }
 
#if NO
        internal bool Equals(Interval interval)
        {
            Fx.Assert(null != interval, "");
            return this.Equals(interval.lowerBound, interval.lowerOp, interval.upperBound, interval.upperOp);
        }
#endif
        internal bool Equals(double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            return (this.lowerBound == lowerBound && this.lowerOp == lowerOp && this.upperBound == upperBound && this.upperOp == upperOp);
        }
 
        internal bool HasMatchingEndPoint(double endpoint)
        {
            return (this.lowerBound == endpoint || this.upperBound == endpoint);
        }
    }
 
    /// <summary>
    /// IntervalCollection
    /// All Contains/Find operations are currently linear, since they are required only during
    /// Inserts/Removes from the Interval Tree, which is not anticipated to be performance critical.
    /// </summary>
    internal class IntervalCollection : ArrayList
    {
        internal IntervalCollection()
            : base(1)
        {
        }
 
        internal bool HasIntervals
        {
            get
            {
                return (this.Count > 0);
            }
        }
 
        internal new Interval this[int index]
        {
            get
            {
                return (Interval)base[index];
            }
        }
 
        internal int Add(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.Capacity = this.Count + 1;
            return base.Add(interval);
        }
 
        internal int AddUnique(Interval interval)
        {
            Fx.Assert(null != interval, "");
            int index = this.IndexOf(interval);
            if (-1 == index)
            {
                return this.Add(interval);
            }
            return index;
        }
 
        internal IntervalCollection GetIntervalsWithEndPoint(double endPoint)
        {
            IntervalCollection matches = new IntervalCollection();
 
            int count = this.Count;
            for (int i = 0; i < count; ++i)
            {
                Interval interval = this[i];
                if (interval.HasMatchingEndPoint(endPoint))
                {
                    matches.Add(interval);
                }
            }
 
            return matches;
        }
 
        internal int IndexOf(Interval interval)
        {
            Fx.Assert(null != interval, "");
            return base.IndexOf(interval);
        }
 
        internal int IndexOf(double endPoint)
        {
            int count = this.Count;
            for (int i = 0; i < count; ++i)
            {
                Interval interval = this[i];
                if (interval.HasMatchingEndPoint(endPoint))
                {
                    return i;
                }
            }
 
            return -1;
        }
 
        internal int IndexOf(double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            int count = this.Count;
            for (int i = 0; i < count; ++i)
            {
                if (this[i].Equals(lowerBound, lowerOp, upperBound, upperOp))
                {
                    return i;
                }
            }
            return -1;
        }
 
        internal void Remove(Interval interval)
        {
            Fx.Assert(null != interval, "");
            base.Remove(interval);
            this.TrimToSize();
        }
 
        internal void Trim()
        {
            this.TrimToSize();
        }
    }
 
    internal class IntervalBoundary
    {
        IntervalCollection eqSlot;
        IntervalCollection gtSlot;
        IntervalBoundary left;
        IntervalCollection ltSlot;
        IntervalBoundary parent;
        IntervalBoundary right;
        double val;
 
        internal IntervalBoundary(double val, IntervalBoundary parent)
        {
            this.val = val;
            this.parent = parent;
        }
 
        internal IntervalCollection EqSlot
        {
            get
            {
                return this.eqSlot;
            }
        }
 
        internal IntervalCollection GtSlot
        {
            get
            {
                return this.gtSlot;
            }
        }
 
        internal IntervalBoundary Left
        {
            get
            {
                return this.left;
            }
            set
            {
                this.left = value;
            }
        }
 
        internal IntervalCollection LtSlot
        {
            get
            {
                return this.ltSlot;
            }
        }
 
        internal IntervalBoundary Parent
        {
            get
            {
                return this.parent;
            }
            set
            {
                this.parent = value;
            }
        }
 
        internal IntervalBoundary Right
        {
            get
            {
                return this.right;
            }
            set
            {
                this.right = value;
            }
        }
 
        internal double Value
        {
            get
            {
                return this.val;
            }
            set
            {
                this.val = value;
            }
        }
 
        internal void AddToEqSlot(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.AddToSlot(ref this.eqSlot, interval);
        }
 
        internal void AddToGtSlot(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.AddToSlot(ref this.gtSlot, interval);
        }
 
        internal void AddToLtSlot(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.AddToSlot(ref this.ltSlot, interval);
        }
 
        void AddToSlot(ref IntervalCollection slot, Interval interval)
        {
            if (null == slot)
            {
                slot = new IntervalCollection();
            }
            slot.AddUnique(interval);
        }
 
        internal IntervalBoundary EnsureLeft(double val)
        {
            if (null == this.left)
            {
                this.left = new IntervalBoundary(val, this);
            }
 
            return this.left;
        }
 
        internal IntervalBoundary EnsureRight(double val)
        {
            if (null == this.right)
            {
                this.right = new IntervalBoundary(val, this);
            }
 
            return this.right;
        }
#if NO
        internal Interval GetInterval(double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            Interval interval;
            if (
                null != (interval = this.GetIntervalFromSlot(this.eqSlot, lowerBound, lowerOp, upperBound, upperOp))
                || null != (interval = this.GetIntervalFromSlot(this.ltSlot, lowerBound, lowerOp, upperBound, upperOp))
                || null != (interval = this.GetIntervalFromSlot(this.gtSlot, lowerBound, lowerOp, upperBound, upperOp))
                )
            {
                return interval;
            }
 
            return null;
        }
 
        internal Interval GetIntervalByData(object data)
        {
            Interval interval;
            if (
                null != (interval = this.GetIntervalFromSlot(this.eqSlot, data))
                || null != (interval = this.GetIntervalFromSlot(this.ltSlot, data))
                || null != (interval = this.GetIntervalFromSlot(this.gtSlot, data))
                )
            {
                return interval;
            }
 
            return null;
        }
 
        Interval GetIntervalFromSlot(IntervalCollection slot, object data)
        {
            int index;
            if (null != slot && -1 != (index = slot.IndexOf(data)))
            {
                return slot[index];
            }
            return null;
        }
 
        Interval GetIntervalFromSlot(IntervalCollection slot, double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            int index;
            if (null != slot && -1 != (index = slot.IndexOf(lowerBound, lowerOp, upperBound, upperOp)))
            {
                return slot[index];
            }
            return null;
        }
#endif
        internal void RemoveFromEqSlot(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.RemoveFromSlot(ref this.eqSlot, interval);
        }
 
        internal void RemoveFromGtSlot(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.RemoveFromSlot(ref this.gtSlot, interval);
        }
 
        internal void RemoveFromLtSlot(Interval interval)
        {
            Fx.Assert(null != interval, "");
            this.RemoveFromSlot(ref this.ltSlot, interval);
        }
 
        void RemoveFromSlot(ref IntervalCollection slot, Interval interval)
        {
            if (null != slot)
            {
                slot.Remove(interval);
                if (!slot.HasIntervals)
                {
                    slot = null;
                }
            }
        }
 
        internal void Trim()
        {
            if (this.eqSlot != null)
            {
                this.eqSlot.Trim();
            }
 
            if (this.gtSlot != null)
            {
                this.gtSlot.Trim();
            }
 
            if (this.ltSlot != null)
            {
                this.ltSlot.Trim();
            }
 
            if (this.left != null)
            {
                this.left.Trim();
            }
 
            if (this.right != null)
            {
                this.right.Trim();
            }
        }
    }
 
    internal struct IntervalTreeTraverser
    {
        IntervalBoundary currentNode;
        IntervalBoundary nextNode;
        IntervalCollection slot;
        double val;
 
        internal IntervalTreeTraverser(double val, IntervalBoundary root)
        {
            Fx.Assert(null != root, "");
            this.currentNode = null;
            this.slot = null;
            this.nextNode = root;
            this.val = val;
        }
 
        internal IntervalCollection Slot
        {
            get
            {
                return this.slot;
            }
        }
 
        internal bool MoveNext()
        {
            while (null != this.nextNode)
            {
                this.currentNode = this.nextNode;
                double currentVal = this.currentNode.Value;
                if (val < currentVal)
                {
                    this.slot = this.currentNode.LtSlot;
                    this.nextNode = this.currentNode.Left;
                }
                else if (val > currentVal)
                {
                    this.slot = this.currentNode.GtSlot;
                    this.nextNode = this.currentNode.Right;
                }
                else
                {
                    this.slot = this.currentNode.EqSlot;
                    this.nextNode = null;
                }
                if (null != this.slot)
                {
                    return true;
                }
            }
 
            return false;
        }
    }
 
    internal class IntervalTree
    {
        IntervalCollection intervals;
        IntervalBoundary root;
 
        internal IntervalTree()
        {
        }
 
        internal int Count
        {
            get
            {
                return (null != this.intervals) ? this.intervals.Count : 0;
            }
        }
 
        internal IntervalCollection Intervals
        {
            get
            {
                if (null == this.intervals)
                {
                    return new IntervalCollection();
                }
                return this.intervals;
            }
        }
 
#if NO
        internal bool IsEmpty
        {
            get
            {
                return (this.root == null);
            }
        }
#endif
        internal IntervalBoundary Root
        {
            get
            {
                return this.root;
            }
        }
 
        internal void Add(Interval interval)
        {
            Fx.Assert(null != interval, "");
 
            this.AddIntervalToTree(interval);
            this.EnsureIntervals();
            this.intervals.Add(interval);
        }
 
        void AddIntervalToTree(Interval interval)
        {
            this.EditLeft(interval, true);
            this.EditRight(interval, true);
        }
 
#if NO
        internal bool Contains(double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            if (null != this.intervals)
            {
                return (this.intervals.IndexOf(lowerBound, lowerOp, upperBound, upperOp) >= 0);
            }
 
            return false;
        }
#endif
        void EditLeft(Interval interval, bool add)
        {
            if (add)
            {
                this.EnsureRoot(interval.LowerBound);
            }
 
            IntervalBoundary root = this.root;
            IntervalBoundary leftAncestor = null;
 
            while (true)
            {
                double rootVal = root.Value;
 
                if (rootVal < interval.LowerBound)
                {
                    // root is outside the interval range because it is < the lower bound
                    root = add ? root.EnsureRight(interval.LowerBound) : root.Right;
                    continue;
                }
 
                // rootVal is >= to interval.lowerBound
                //
                // All values in thee subtree at 'root' are < leftAncestor.Value
                // All values to the left of this node cannot be in the interval because they are < the interval's
                // lower bound.
                // Values to the right of this node lie in range (root.Value to leftAncestor.Value)
                // Thus, the entire right subtree of root will be inside the range if the interval.upperBound
                // is >= leftAncestor.Value
                if (null != leftAncestor && leftAncestor.Value <= interval.UpperBound)
                {
                    if (add)
                    {
                        root.AddToGtSlot(interval);
                    }
                    else
                    {
                        root.RemoveFromGtSlot(interval);
                    }
                }
 
                if (rootVal > interval.LowerBound)
                { // This node itself lies in the range if it is also < the upper bound
                    if (rootVal < interval.UpperBound)
                    {
                        if (add)
                        {
                            root.AddToEqSlot(interval);
                        }
                        else
                        {
                            root.RemoveFromEqSlot(interval);
                        }
                    }
                    leftAncestor = root;
                    root = add ? root.EnsureLeft(interval.LowerBound) : root.Left;
                    continue;
                }
 
                // lowerBound == rootVal. We're done.
                if (IntervalOp.LessThanEquals == interval.LowerOp)
                {
                    // If the range is inclusive of the lower bound (>=), then since this node == lowerBound,
                    // it must be in the range.
                    if (add)
                    {
                        root.AddToEqSlot(interval);
                    }
                    else
                    {
                        root.RemoveFromEqSlot(interval);
                    }
                }
                break;
            }
        }
 
        void EditRight(Interval interval, bool add)
        {
            if (add)
            {
                this.EnsureRoot(interval.UpperBound);
            }
 
            IntervalBoundary root = this.root;
            IntervalBoundary rightAncestor = null;
 
            while (true)
            {
                double rootVal = root.Value;
 
                if (rootVal > interval.UpperBound)
                {
                    // root is outside the interval range because it is > the upper bound
                    root = add ? root.EnsureLeft(interval.UpperBound) : root.Left;
                    continue;
                }
 
                // rootVal is <= to interval.UpperBound
                //
                // All values in the subtree at 'root' are > leftAncestor.Value
                // All values to the right of this node cannot be in the interval because they are > the interval's
                // upper bound.
                // Values to the left of this node lie in range (rightAncestor.Value to root.Value)
                // Thus, the entire left subtree of root will be inside the range if the interval.lowerBound
                // is <= rightAncestor.Value
                if (null != rightAncestor && rightAncestor.Value >= interval.LowerBound)
                {
                    if (add)
                    {
                        root.AddToLtSlot(interval);
                    }
                    else
                    {
                        root.RemoveFromLtSlot(interval);
                    }
                }
 
                if (rootVal < interval.UpperBound)
                {
                    // This node itself lies in the range if it is also > the lower bound
                    if (rootVal > interval.LowerBound)
                    {
                        if (add)
                        {
                            root.AddToEqSlot(interval);
                        }
                        else
                        {
                            root.RemoveFromEqSlot(interval);
                        }
                    }
                    rightAncestor = root;
                    root = add ? root.EnsureRight(interval.UpperBound) : root.Right;
                    continue;
                }
 
                // upperBound == rootVal. We're done.
                // If upperBound == lowerBound, we already inserted this when doing AddLeft
                if (IntervalOp.LessThanEquals == interval.UpperOp)
                {
                    // If the range is inclusive of the upper bound, then since this node == upperBound,
                    // it must be in the range.
                    if (add)
                    {
                        root.AddToEqSlot(interval);
                    }
                    else
                    {
                        root.RemoveFromEqSlot(interval);
                    }
                }
                break;
            }
        }
 
        void EnsureIntervals()
        {
            if (null == this.intervals)
            {
                this.intervals = new IntervalCollection();
            }
        }
 
        void EnsureRoot(double val)
        {
            if (null == this.root)
            {
                this.root = new IntervalBoundary(val, null);
            }
        }
 
        internal IntervalBoundary FindBoundaryNode(double val)
        {
            return this.FindBoundaryNode(root, val);
        }
 
        internal IntervalBoundary FindBoundaryNode(IntervalBoundary root, double val)
        {
            IntervalBoundary boundary = null;
            if (null != root)
            {
                if (root.Value == val)
                {
                    boundary = root;
                }
                else
                {
                    if (null == (boundary = this.FindBoundaryNode(root.Left, val)))
                    {
                        boundary = this.FindBoundaryNode(root.Right, val);
                    }
                }
            }
            return boundary;
        }
 
        internal Interval FindInterval(Interval interval)
        {
            return this.FindInterval(interval.LowerBound, interval.LowerOp, interval.UpperBound, interval.UpperOp);
        }
 
        internal Interval FindInterval(double lowerBound, IntervalOp lowerOp, double upperBound, IntervalOp upperOp)
        {
            if (null != this.intervals)
            {
                int index;
                if (-1 != (index = this.intervals.IndexOf(lowerBound, lowerOp, upperBound, upperOp)))
                {
                    return this.intervals[index];
                }
            }
            return null;
        }
 
        /// <summary>
        /// An interval has been removed. Prune the tree appropriately
        /// </summary>
        /// <param name="intervalRemoved">interval that was removed</param>
        void PruneTree(Interval intervalRemoved)
        {
            // Delete endpoints if no other intervals have them
            int index;
            if (-1 == (index = this.intervals.IndexOf(intervalRemoved.LowerBound)))
            {
                this.RemoveBoundary(this.FindBoundaryNode(intervalRemoved.LowerBound));
            }
            if (intervalRemoved.LowerBound != intervalRemoved.UpperBound && -1 == (index = this.intervals.IndexOf(intervalRemoved.UpperBound)))
            {
                this.RemoveBoundary(this.FindBoundaryNode(intervalRemoved.UpperBound));
            }
        }
 
        internal void Remove(Interval interval)
        {
            Fx.Assert(null != interval, "");
            Fx.Assert(null != this.intervals, "");
 
            // First, delete all occurences of interval in the tree. Note: we do a reference equals
            this.RemoveIntervalFromTree(interval);
            // Remove interval from interval collection
            this.intervals.Remove(interval);
            // It may be possible to prune the tree.. this will do the necessary, if required
            this.PruneTree(interval);
        }
 
        void RemoveBoundary(IntervalBoundary boundary)
        {
            IntervalCollection replacementIntervals = null;
            int replacementCount = 0;
 
            if (null != boundary.Left && null != boundary.Right)
            {
                // Neither left/right are null. Typical binary tree node removal - replace the removed node
                // with the symmetric order predecessor
                IntervalBoundary replacement = boundary.Left;
                while (null != replacement.Right)
                {
                    replacement = replacement.Right;
                }
                // Find all intervals with endpoint y in the tree
                replacementIntervals = this.intervals.GetIntervalsWithEndPoint(replacement.Value);
 
                // Remove the intervals from the tree
                replacementCount = replacementIntervals.Count;
                for (int i = 0; i < replacementCount; ++i)
                {
                    this.RemoveIntervalFromTree(replacementIntervals[i]);
                }
 
                double val = boundary.Value;
                boundary.Value = replacement.Value;
                replacement.Value = val;
                boundary = replacement;
            }
 
            if (null != boundary.Left)
            {
                this.Replace(boundary, boundary.Left);
            }
            else
            {
                this.Replace(boundary, boundary.Right);
            }
 
            // Discard the node
            boundary.Parent = null;
            boundary.Left = null;
            boundary.Right = null;
 
            // Reinstall Intervals
            for (int i = 0; i < replacementCount; ++i)
            {
                this.AddIntervalToTree(replacementIntervals[i]);
            }
        }
 
        void RemoveIntervalFromTree(Interval interval)
        {
            this.EditLeft(interval, false);
            this.EditRight(interval, false);
        }
 
        void Replace(IntervalBoundary replace, IntervalBoundary with)
        {
            IntervalBoundary parent = replace.Parent;
            if (null != parent)
            {
                if (replace == parent.Left)
                {
                    parent.Left = with;
                }
                else if (replace == parent.Right)
                {
                    parent.Right = with;
                }
            }
            else
            {
                // Replacing root
                this.root = with;
            }
            if (null != with)
            {
                with.Parent = parent;
            }
        }
 
        internal void Trim()
        {
            this.intervals.Trim();
            this.root.Trim();
        }
    }
 
    internal class NumberRelationOpcode : LiteralRelationOpcode
    {
        double literal;
        RelationOperator op;
 
        internal NumberRelationOpcode(double literal, RelationOperator op)
            : this(OpcodeID.NumberRelation, literal, op)
        {
        }
 
        protected NumberRelationOpcode(OpcodeID id, double literal, RelationOperator op)
            : base(id)
        {
            this.literal = literal;
            this.op = op;
        }
#if NO
        internal override ValueDataType DataType
        {
            get
            {
                return ValueDataType.Double;
            }
        }
#endif
 
        internal override object Literal
        {
            get
            {
                return this.literal;
            }
        }
#if NO
        internal RelationOperator Op
        {
            get
            {
                return this.op;
            }
        }
#endif
        internal override bool Equals(Opcode opcode)
        {
            if (base.Equals(opcode))
            {
                NumberRelationOpcode numOp = (NumberRelationOpcode)opcode;
                return (numOp.op == this.op && numOp.literal == this.literal);
            }
 
            return false;
        }
 
        internal override Opcode Eval(ProcessingContext context)
        {
            Value[] values = context.Values;
            StackFrame arg = context.TopArg;
 
            for (int i = arg.basePtr; i <= arg.endPtr; ++i)
            {
                values[i].Update(context, values[i].CompareTo(this.literal, op));
            }
            return this.next;
        }
 
        internal Interval ToInterval()
        {
            return new Interval(this.literal, this.op);
        }
    }
 
    internal class NumberIntervalOpcode : NumberRelationOpcode
    {
        Interval interval;
 
        internal NumberIntervalOpcode(double literal, RelationOperator op)
            : base(OpcodeID.NumberInterval, literal, op)
        {
        }
 
        internal override object Literal
        {
            get
            {
                if (null == this.interval)
                {
                    this.interval = this.ToInterval();
                }
 
                return this.interval;
            }
        }
 
        internal override void Add(Opcode op)
        {
            NumberIntervalOpcode intervalOp = op as NumberIntervalOpcode;
            if (null == intervalOp)
            {
                base.Add(op);
                return;
            }
 
            Fx.Assert(null != this.prev, "");
 
            NumberIntervalBranchOpcode branch = new NumberIntervalBranchOpcode();
            this.prev.Replace(this, branch);
            branch.Add(this);
            branch.Add(intervalOp);
        }
    }
 
    internal class IntervalBranchIndex : QueryBranchIndex
    {
        IntervalTree intervalTree;
 
        internal IntervalBranchIndex()
        {
            this.intervalTree = new IntervalTree();
        }
 
        internal override int Count
        {
            get
            {
                return this.intervalTree.Count;
            }
        }
 
        internal override QueryBranch this[object key]
        {
            get
            {
                Interval interval = this.intervalTree.FindInterval((Interval)key);
                if (null != interval)
                {
                    return interval.Branch;
                }
                return null;
            }
            set
            {
                Interval interval = (Interval)key;
                interval.Branch = value;
                this.intervalTree.Add(interval);
            }
        }
 
        internal override void CollectXPathFilters(ICollection<MessageFilter> filters)
        {
            for (int i = 0; i < this.intervalTree.Intervals.Count; ++i)
            {
                this.intervalTree.Intervals[i].Branch.Branch.CollectXPathFilters(filters);
            }
        }
 
#if NO

        internal override IEnumerator GetEnumerator()
        {
            return this.intervalTree.Intervals.GetEnumerator();
        }
        
#endif
 
        void Match(int valIndex, double point, QueryBranchResultSet results)
        {
            IntervalTreeTraverser traverser = new IntervalTreeTraverser(point, this.intervalTree.Root);
            while (traverser.MoveNext())
            {
                IntervalCollection matches = traverser.Slot;
                for (int i = 0, count = matches.Count; i < count; ++i)
                {
                    QueryBranch branch = matches[i].Branch;
                    if (null != branch)
                    {
                        results.Add(branch, valIndex);
                    }
                }
            }
        }
 
        internal override void Match(int valIndex, ref Value val, QueryBranchResultSet results)
        {
            if (ValueDataType.Sequence == val.Type)
            {
                NodeSequence sequence = val.Sequence;
                for (int i = 0; i < sequence.Count; ++i)
                {
                    this.Match(valIndex, sequence.Items[i].NumberValue(), results);
                }
            }
            else
            {
                this.Match(valIndex, val.ToDouble(), results);
            }
        }
 
        internal override void Remove(object key)
        {
            this.intervalTree.Remove((Interval)key);
        }
 
        internal override void Trim()
        {
            this.intervalTree.Trim();
        }
    }
 
    internal class NumberIntervalBranchOpcode : QueryConditionalBranchOpcode
    {
        internal NumberIntervalBranchOpcode()
            : base(OpcodeID.NumberIntervalBranch, new IntervalBranchIndex())
        {
        }
 
        internal override LiteralRelationOpcode ValidateOpcode(Opcode opcode)
        {
            NumberIntervalOpcode numOp = opcode as NumberIntervalOpcode;
            if (null != numOp)
            {
                return numOp;
            }
 
            return null;
        }
 
    }
}