| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167 |
- // 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.
- using System.Reactive.Concurrency;
- using System.Reactive.Disposables;
- using System.Reactive.Subjects;
- using System.Threading;
- namespace System.Reactive.Linq.ObservableImpl
- {
- internal static class RefCount<TSource>
- {
- internal sealed class Eager : Producer<TSource, Eager._>
- {
- private readonly IConnectableObservable<TSource> _source;
- private readonly object _gate;
- private int _count;
- private IDisposable _connectableSubscription;
- public Eager(IConnectableObservable<TSource> source)
- {
- _source = source;
- _gate = new object();
- _count = 0;
- _connectableSubscription = default(IDisposable);
- }
- protected override _ CreateSink(IObserver<TSource> observer) => new _(observer, this);
- protected override void Run(_ sink) => sink.Run();
- internal sealed class _ : IdentitySink<TSource>
- {
- readonly Eager _parent;
- public _(IObserver<TSource> observer, Eager parent)
- : base(observer)
- {
- this._parent = parent;
- }
- public void Run()
- {
- base.Run(_parent._source);
- lock (_parent._gate)
- {
- if (++_parent._count == 1)
- {
- // We need to set _connectableSubscription to something
- // before Connect because if Connect terminates synchronously,
- // Dispose(bool) gets executed and will try to dispose
- // _connectableSubscription of null.
- // ?.Dispose() is no good because the dispose action has to be
- // executed anyway.
- // We can't inline SAD either because the IDisposable of Connect
- // may belong to the wrong connection.
- var sad = new SingleAssignmentDisposable();
- _parent._connectableSubscription = sad;
- sad.Disposable = _parent._source.Connect();
- }
- }
- }
- protected override void Dispose(bool disposing)
- {
- base.Dispose(disposing);
- if (disposing)
- {
- lock (_parent._gate)
- {
- if (--_parent._count == 0)
- {
- _parent._connectableSubscription.Dispose();
- }
- }
- }
- }
- }
- }
- internal sealed class Lazy : Producer<TSource, Lazy._>
- {
- private readonly object _gate;
- private readonly IScheduler _scheduler;
- private readonly TimeSpan _disconnectTime;
- private readonly IConnectableObservable<TSource> _source;
- private IDisposable _serial;
- private int _count;
- private IDisposable _connectableSubscription;
- public Lazy(IConnectableObservable<TSource> source, TimeSpan disconnectTime, IScheduler scheduler)
- {
- _source = source;
- _gate = new object();
- _disconnectTime = disconnectTime;
- _scheduler = scheduler;
- _count = 0;
- _connectableSubscription = default(IDisposable);
- }
- protected override _ CreateSink(IObserver<TSource> observer) => new _(observer);
- protected override void Run(_ sink) => sink.Run(this);
- internal sealed class _ : IdentitySink<TSource>
- {
- public _(IObserver<TSource> observer)
- : base(observer)
- {
- }
- public void Run(Lazy parent)
- {
- var subscription = parent._source.SubscribeSafe(this);
- lock (parent._gate)
- {
- if (++parent._count == 1)
- {
- if (parent._connectableSubscription == null)
- parent._connectableSubscription = parent._source.Connect();
- Disposable.TrySetSerial(ref parent._serial, new SingleAssignmentDisposable());
- }
- }
- SetUpstream(Disposable.Create(
- (parent, subscription),
- tuple =>
- {
- var (closureParent, closureSubscription) = tuple;
- closureSubscription.Dispose();
- lock (closureParent._gate)
- {
- if (--closureParent._count == 0)
- {
- var cancelable = (SingleAssignmentDisposable)Volatile.Read(ref closureParent._serial);
- cancelable.Disposable = closureParent._scheduler.Schedule((cancelable, closureParent), closureParent._disconnectTime, (self, tuple2) =>
- {
- lock (tuple2.closureParent._gate)
- {
- if (object.ReferenceEquals(Volatile.Read(ref tuple2.closureParent._serial), tuple2.cancelable))
- {
- tuple2.closureParent._connectableSubscription.Dispose();
- tuple2.closureParent._connectableSubscription = null;
- }
- }
- return Disposable.Empty;
- });
- }
- }
- }));
- }
- }
- }
- }
- }
|