// 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.Reactive.Disposables; namespace System.Reactive.Linq.ObservableImpl { internal sealed class WithLatestFrom : Producer._> { private readonly IObservable _first; private readonly IObservable _second; private readonly Func _resultSelector; public WithLatestFrom(IObservable first, IObservable second, Func resultSelector) { _first = first; _second = second; _resultSelector = resultSelector; } protected override _ CreateSink(IObserver observer) => new(_resultSelector, observer); protected override void Run(_ sink) => sink.Run(_first, _second); internal sealed class _ : IdentitySink { private readonly object _gate = new(); private readonly object _latestGate = new(); private readonly Func _resultSelector; public _(Func resultSelector, IObserver observer) : base(observer) { _resultSelector = resultSelector; } private volatile bool _hasLatest; private TSecond? _latest; private SingleAssignmentDisposableValue _secondDisposable; public void Run(IObservable first, IObservable second) { var fstO = new FirstObserver(this); var sndO = new SecondObserver(this); _secondDisposable.Disposable = second.SubscribeSafe(sndO); SetUpstream(first.SubscribeSafe(fstO)); } protected override void Dispose(bool disposing) { if (disposing) { _secondDisposable.Dispose(); } base.Dispose(disposing); } private sealed class FirstObserver : IObserver { private readonly _ _parent; public FirstObserver(_ parent) { _parent = parent; } public void OnCompleted() { lock (_parent._gate) { _parent.ForwardOnCompleted(); } } public void OnError(Exception error) { lock (_parent._gate) { _parent.ForwardOnError(error); } } public void OnNext(TFirst value) { if (_parent._hasLatest) // Volatile read { TSecond latest; lock (_parent._latestGate) { latest = _parent._latest!; // NB: Not null when hasLatest is true. } TResult res; try { res = _parent._resultSelector(value, latest); } catch (Exception ex) { lock (_parent._gate) { _parent.ForwardOnError(ex); } return; } lock (_parent._gate) { _parent.ForwardOnNext(res); } } } } private sealed class SecondObserver : IObserver { private readonly _ _parent; public SecondObserver(_ parent) { _parent = parent; } public void OnCompleted() { _parent._secondDisposable.Dispose(); } public void OnError(Exception error) { lock (_parent._gate) { _parent.ForwardOnError(error); } } public void OnNext(TSecond value) { lock (_parent._latestGate) { _parent._latest = value; } if (!_parent._hasLatest) { _parent._hasLatest = true; } } } } } }