| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328 | // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.#if !NO_PERFusing System;using System.Collections.Generic;using System.Reactive.Disposables;namespace System.Reactive.Linq.ObservableImpl{    class SequenceEqual<TSource> : Producer<bool>    {        private readonly IObservable<TSource> _first;        private readonly IObservable<TSource> _second;        private readonly IEnumerable<TSource> _secondE;        private readonly IEqualityComparer<TSource> _comparer;        public SequenceEqual(IObservable<TSource> first, IObservable<TSource> second, IEqualityComparer<TSource> comparer)        {            _first = first;            _second = second;            _comparer = comparer;        }        public SequenceEqual(IObservable<TSource> first, IEnumerable<TSource> second, IEqualityComparer<TSource> comparer)        {            _first = first;            _secondE = second;            _comparer = comparer;        }        protected override IDisposable Run(IObserver<bool> observer, IDisposable cancel, Action<IDisposable> setSink)        {            if (_second != null)            {                var sink = new _(this, observer, cancel);                setSink(sink);                return sink.Run();            }            else            {                var sink = new SequenceEqualImpl(this, observer, cancel);                setSink(sink);                return sink.Run();            }        }        class _ : Sink<bool>        {            private readonly SequenceEqual<TSource> _parent;            public _(SequenceEqual<TSource> parent, IObserver<bool> observer, IDisposable cancel)                : base(observer, cancel)            {                _parent = parent;            }            private object _gate;            private bool _donel;            private bool _doner;            private Queue<TSource> _ql;            private Queue<TSource> _qr;            public IDisposable Run()            {                _gate = new object();                _donel = false;                _doner = false;                _ql = new Queue<TSource>();                _qr = new Queue<TSource>();                return StableCompositeDisposable.Create                (                    _parent._first.SubscribeSafe(new F(this)),                    _parent._second.SubscribeSafe(new S(this))                );            }            class F : IObserver<TSource>            {                private readonly _ _parent;                public F(_ parent)                {                    _parent = parent;                }                public void OnNext(TSource value)                {                    lock (_parent._gate)                    {                        if (_parent._qr.Count > 0)                        {                            var equal = false;                            var v = _parent._qr.Dequeue();                            try                            {                                equal = _parent._parent._comparer.Equals(value, v);                            }                            catch (Exception exception)                            {                                _parent._observer.OnError(exception);                                _parent.Dispose();                                return;                            }                            if (!equal)                            {                                _parent._observer.OnNext(false);                                _parent._observer.OnCompleted();                                _parent.Dispose();                            }                        }                        else if (_parent._doner)                        {                            _parent._observer.OnNext(false);                            _parent._observer.OnCompleted();                            _parent.Dispose();                        }                        else                            _parent._ql.Enqueue(value);                    }                }                public void OnError(Exception error)                {                    lock (_parent._gate)                    {                        _parent._observer.OnError(error);                        _parent.Dispose();                    }                }                public void OnCompleted()                {                    lock (_parent._gate)                    {                        _parent._donel = true;                        if (_parent._ql.Count == 0)                        {                            if (_parent._qr.Count > 0)                            {                                _parent._observer.OnNext(false);                                _parent._observer.OnCompleted();                                _parent.Dispose();                            }                            else if (_parent._doner)                            {                                _parent._observer.OnNext(true);                                _parent._observer.OnCompleted();                                _parent.Dispose();                            }                        }                    }                }            }            class S : IObserver<TSource>            {                private readonly _ _parent;                public S(_ parent)                {                    _parent = parent;                }                public void OnNext(TSource value)                {                    lock (_parent._gate)                    {                        if (_parent._ql.Count > 0)                        {                            var equal = false;                            var v = _parent._ql.Dequeue();                            try                            {                                equal = _parent._parent._comparer.Equals(v, value);                            }                            catch (Exception exception)                            {                                _parent._observer.OnError(exception);                                _parent.Dispose();                                return;                            }                            if (!equal)                            {                                _parent._observer.OnNext(false);                                _parent._observer.OnCompleted();                                _parent.Dispose();                            }                        }                        else if (_parent._donel)                        {                            _parent._observer.OnNext(false);                            _parent._observer.OnCompleted();                            _parent.Dispose();                        }                        else                            _parent._qr.Enqueue(value);                    }                }                public void OnError(Exception error)                {                    lock (_parent._gate)                    {                        _parent._observer.OnError(error);                        _parent.Dispose();                    }                }                public void OnCompleted()                {                    lock (_parent._gate)                    {                        _parent._doner = true;                        if (_parent._qr.Count == 0)                        {                            if (_parent._ql.Count > 0)                            {                                _parent._observer.OnNext(false);                                _parent._observer.OnCompleted();                                _parent.Dispose();                            }                            else if (_parent._donel)                            {                                _parent._observer.OnNext(true);                                _parent._observer.OnCompleted();                                _parent.Dispose();                            }                        }                    }                }            }        }        class SequenceEqualImpl : Sink<bool>, IObserver<TSource>        {            private readonly SequenceEqual<TSource> _parent;            public SequenceEqualImpl(SequenceEqual<TSource> parent, IObserver<bool> observer, IDisposable cancel)                : base(observer, cancel)            {                _parent = parent;            }            private IEnumerator<TSource> _enumerator;            public IDisposable Run()            {                //                // Notice the evaluation order of obtaining the enumerator and subscribing to the                // observable sequence is reversed compared to the operator's signature. This is                // required to make sure the enumerator is available as soon as the observer can                // be called. Otherwise, we end up having a race for the initialization and use                // of the _rightEnumerator field.                //                try                {                    _enumerator = _parent._secondE.GetEnumerator();                }                catch (Exception exception)                {                    base._observer.OnError(exception);                    base.Dispose();                    return Disposable.Empty;                }                return StableCompositeDisposable.Create(                    _parent._first.SubscribeSafe(this),                    _enumerator                );            }            public void OnNext(TSource value)            {                var equal = false;                try                {                    if (_enumerator.MoveNext())                    {                        var current = _enumerator.Current;                        equal = _parent._comparer.Equals(value, current);                    }                }                catch (Exception exception)                {                    base._observer.OnError(exception);                    base.Dispose();                    return;                }                if (!equal)                {                    base._observer.OnNext(false);                    base._observer.OnCompleted();                    base.Dispose();                }            }            public void OnError(Exception error)            {                base._observer.OnError(error);                base.Dispose();            }            public void OnCompleted()            {                var hasNext = false;                try                {                    hasNext = _enumerator.MoveNext();                }                catch (Exception exception)                {                    base._observer.OnError(exception);                    base.Dispose();                    return;                }                base._observer.OnNext(!hasNext);                base._observer.OnCompleted();                base.Dispose();            }        }    }}#endif
 |