/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*-*/ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef WAITFOR_H_ #define WAITFOR_H_ #include "MediaEventSource.h" #include "MediaUtils.h" #include "mozilla/Maybe.h" #include "mozilla/MozPromise.h" #include "mozilla/ResultVariant.h" #include "mozilla/SpinEventLoopUntil.h" namespace mozilla { /** * Waits for an occurrence of aEvent on the current thread (by blocking it, * except tasks added to the event loop may run) and returns the event's * templated value, if it's non-void. * * The caller must be wary of eventloop issues, in * particular cases where we rely on a stable state runnable, but there is never * a task to trigger stable state. In such cases it is the responsibility of the * caller to create the needed tasks, as JS would. A noteworthy API that relies * on stable state is MediaTrackGraph::GetInstance. */ template T WaitFor(MediaEventSource& aEvent) { Maybe value; MediaEventListener listener = aEvent.Connect( AbstractThread::GetCurrent(), [&](T aValue) { value = Some(aValue); }); SpinEventLoopUntil( "WaitFor(MediaEventSource& aEvent)"_ns, [&] { return value.isSome(); }); listener.Disconnect(); return value.value(); } /** * Specialization of WaitFor for void. */ void WaitFor(MediaEventSource& aEvent); /** * Variant of WaitFor that blocks the caller until a MozPromise has either been * resolved or rejected. */ template Result WaitFor(const RefPtr>& aPromise) { Maybe success; Maybe error; aPromise->Then( GetCurrentSerialEventTarget(), __func__, [&](R aResult) { success = Some(aResult); }, [&](E aError) { error = Some(aError); }); SpinEventLoopUntil( "WaitFor(const RefPtr>& aPromise)"_ns, [&] { return success.isSome() || error.isSome(); }); if (success.isSome()) { return success.extract(); } return Err(error.extract()); } /** * A variation of WaitFor that takes a callback to be called each time aEvent is * raised. Blocks the caller until the callback function returns true. */ template void WaitUntil(MediaEventSource& aEvent, CallbackFunction&& aF) { bool done = false; MediaEventListener listener = aEvent.Connect(AbstractThread::GetCurrent(), [&](Args... aValue) { if (!done) { done = aF(std::forward(aValue)...); } }); SpinEventLoopUntil( "WaitUntil(MediaEventSource& aEvent, CallbackFunction&& aF)"_ns, [&] { return done; }); listener.Disconnect(); } template using TakeNPromise = MozPromise>, bool, true>; template auto TakeN(MediaEventSourceImpl& aEvent, size_t aN) -> RefPtr> { using Storage = std::vector>; using Promise = TakeNPromise; using Values = media::Refcountable; using Listener = media::Refcountable; RefPtr values = MakeRefPtr(); values->reserve(aN); RefPtr listener = MakeRefPtr(); auto promise = InvokeAsync( AbstractThread::GetCurrent(), __func__, [values, aN]() mutable { SpinEventLoopUntil( "TakeN(MediaEventSourceImpl& aEvent, size_t aN)"_ns, [&] { return values->size() == aN; }); return Promise::CreateAndResolve(std::move(*values), __func__); }); *listener = aEvent.Connect(AbstractThread::GetCurrent(), [values, listener, aN](Args... aValue) { values->push_back({aValue...}); if (values->size() == aN) { listener->Disconnect(); } }); return promise; } /** * Helper that, given that canonicals have just been updated on the current * thread, will block its execution until mirrors and their watchers have * executed on aTarget. */ inline void WaitForMirrors(const RefPtr& aTarget) { Unused << WaitFor(InvokeAsync(aTarget, __func__, [] { return GenericPromise::CreateAndResolve(true, "WaitForMirrors resolver"); })); } /** * Short form of WaitForMirrors that assumes mirrors are on the current thread * (like canonicals). */ inline void WaitForMirrors() { WaitForMirrors(GetCurrentSerialEventTarget()); } } // namespace mozilla #endif // WAITFOR_H_