// Licensed to the .NET Foundation under one or more agreements. // The .NET Foundation licenses this file to you under the MIT License. // See the LICENSE file in the project root for more information. using System.Collections.Generic; using System.Diagnostics.CodeAnalysis; using System.Reactive.Disposables; using System.Threading; namespace System.Reactive { internal abstract class TailRecursiveSink : IdentitySink { private readonly Stack>> _stack = new Stack>>(); private bool _isDisposed; private int _trampoline; private IDisposable? _currentSubscription; protected TailRecursiveSink(IObserver observer) : base(observer) { } public void Run(IEnumerable> sources) { if (!TryGetEnumerator(sources, out var current)) { return; } _stack.Push(current); Drain(); } protected override void Dispose(bool disposing) { if (disposing) { DisposeAll(); } base.Dispose(disposing); } private void Drain() { if (Interlocked.Increment(ref _trampoline) != 1) { return; } for (; ; ) { if (Volatile.Read(ref _isDisposed)) { while (_stack.Count != 0) { var enumerator = _stack.Pop(); enumerator.Dispose(); } Disposable.TryDispose(ref _currentSubscription); } else { if (_stack.Count != 0) { var currentEnumerator = _stack.Peek(); var currentObservable = default(IObservable); try { if (currentEnumerator.MoveNext()) { currentObservable = currentEnumerator.Current; } } catch (Exception ex) { currentEnumerator.Dispose(); ForwardOnError(ex); Volatile.Write(ref _isDisposed, true); continue; } IObservable? next; try { next = Helpers.Unpack(currentObservable); } catch (Exception ex) { if (!Fail(ex)) { Volatile.Write(ref _isDisposed, true); } continue; } if (next != null) { var nextSeq = Extract(next); if (nextSeq != null) { if (TryGetEnumerator(nextSeq, out var nextEnumerator)) { _stack.Push(nextEnumerator); continue; } Volatile.Write(ref _isDisposed, true); continue; } // we need an unique indicator for this as // Subscribe could return a Disposable.Empty or // a BooleanDisposable var sad = ReadyToken.Ready; // Swap in the Ready indicator so we know the sequence hasn't been disposed if (Disposable.TrySetSingle(ref _currentSubscription, sad) == TrySetSingleResult.Success) { // subscribe to the source var d = next.SubscribeSafe(this); // Try to swap in the returned disposable in place of the Ready indicator // Since this drain loop is the only one to use Ready, this should // be unambiguous var u = Interlocked.CompareExchange(ref _currentSubscription, d, sad); // sequence disposed or completed synchronously if (u != sad) { d.Dispose(); if (u == BooleanDisposable.True) { continue; } } } else { continue; } } else { _stack.Pop(); currentEnumerator.Dispose(); continue; } } else { Volatile.Write(ref _isDisposed, true); Done(); } } if (Interlocked.Decrement(ref _trampoline) == 0) { break; } } } private void DisposeAll() { Volatile.Write(ref _isDisposed, true); // the disposing of currentSubscription is deferred to drain due to some ObservableExTest.Iterate_Complete() // Interlocked.Exchange(ref currentSubscription, BooleanDisposable.True)?.Dispose(); Drain(); } protected void Recurse() { if (Disposable.TrySetSerial(ref _currentSubscription, null)) { Drain(); } } protected abstract IEnumerable>? Extract(IObservable source); private bool TryGetEnumerator(IEnumerable> sources, [NotNullWhen(true)] out IEnumerator>? result) { try { result = sources.GetEnumerator(); return true; } catch (Exception exception) { ForwardOnError(exception); result = null; return false; } } protected virtual void Done() { ForwardOnCompleted(); } protected virtual bool Fail(Exception error) { ForwardOnError(error); return false; } } /// /// Holds onto a singleton IDisposable indicating a ready state. /// internal static class ReadyToken { /// /// This indicates the operation has been prepared and ready for /// the next step. /// internal static readonly IDisposable Ready = new ReadyDisposable(); private sealed class ReadyDisposable : IDisposable { public void Dispose() { // deliberately no-op } } } }