diff options
Diffstat (limited to 'dom/promise/Promise-inl.h')
-rw-r--r-- | dom/promise/Promise-inl.h | 341 |
1 files changed, 341 insertions, 0 deletions
diff --git a/dom/promise/Promise-inl.h b/dom/promise/Promise-inl.h new file mode 100644 index 0000000000..a7a6ebd0fd --- /dev/null +++ b/dom/promise/Promise-inl.h @@ -0,0 +1,341 @@ +/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ +/* vim: set ts=8 sts=2 et sw=2 tw=80: */ +/* 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 mozilla_dom_Promise_inl_h +#define mozilla_dom_Promise_inl_h + +#include <type_traits> +#include <utility> + +#include "mozilla/AlreadyAddRefed.h" +#include "mozilla/ResultExtensions.h" +#include "mozilla/dom/BindingUtils.h" +#include "mozilla/dom/Promise.h" +#include "mozilla/dom/PromiseNativeHandler.h" +#include "nsCycleCollectionParticipant.h" + +namespace mozilla::dom { + +class PromiseNativeThenHandlerBase : public PromiseNativeHandler { + NS_DECL_CYCLE_COLLECTING_ISUPPORTS + NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS(PromiseNativeThenHandlerBase) + + PromiseNativeThenHandlerBase(Promise* aPromise) : mPromise(aPromise) {} + + virtual bool HasResolvedCallback() = 0; + virtual bool HasRejectedCallback() = 0; + + MOZ_CAN_RUN_SCRIPT void ResolvedCallback(JSContext* aCx, + JS::Handle<JS::Value> aValue, + ErrorResult& aRv) override; + + MOZ_CAN_RUN_SCRIPT void RejectedCallback(JSContext* aCx, + JS::Handle<JS::Value> aValue, + ErrorResult& aRv) override; + + protected: + virtual ~PromiseNativeThenHandlerBase() = default; + + MOZ_CAN_RUN_SCRIPT virtual already_AddRefed<Promise> CallResolveCallback( + JSContext* aCx, JS::Handle<JS::Value> aValue, ErrorResult& aRv) = 0; + MOZ_CAN_RUN_SCRIPT virtual already_AddRefed<Promise> CallRejectCallback( + JSContext* aCx, JS::Handle<JS::Value> aValue, ErrorResult& aRv) = 0; + + virtual void Traverse(nsCycleCollectionTraversalCallback&) = 0; + virtual void Unlink() = 0; + virtual void Trace(const TraceCallbacks& aCallbacks, void* aClosure) = 0; + + RefPtr<Promise> mPromise; +}; + +namespace { + +template <typename T, bool = IsRefcounted<std::remove_pointer_t<T>>::value, + bool = (std::is_convertible_v<T, nsISupports*> || + std::is_convertible_v<T*, nsISupports*>)> +struct StorageTypeHelper { + using Type = T; +}; + +template <typename T> +struct StorageTypeHelper<T, true, true> { + using Type = nsCOMPtr<T>; +}; + +template <typename T> +struct StorageTypeHelper<nsCOMPtr<T>, true, true> { + using Type = nsCOMPtr<T>; +}; + +template <typename T> +struct StorageTypeHelper<T*, true, false> { + using Type = RefPtr<T>; +}; + +template <typename T> +struct StorageTypeHelper<JS::Handle<T>, false, false> { + using Type = JS::Heap<T>; +}; + +template <template <typename> class SmartPtr, typename T> +struct StorageTypeHelper<SmartPtr<T>, true, false> + : std::enable_if<std::is_convertible_v<SmartPtr<T>, T*>, RefPtr<T>> { + using Type = typename StorageTypeHelper::enable_if::type; +}; + +template <typename T> +using StorageType = typename StorageTypeHelper<std::decay_t<T>>::Type; + +// Helpers to choose the correct argument type based on the storage type. Smart +// pointers are converted to the corresponding raw pointer type. Everything else +// is passed by move reference. +// +// Note: We can't just use std::forward for this because the input type may be a +// raw pointer which does not match the argument type, and while the +// spec-compliant behavior there should still give us the expected results, MSVC +// considers it an illegal use of std::forward. +template <template <typename> class SmartPtr, typename T> +decltype(std::declval<SmartPtr<T>>().get()) ArgType(SmartPtr<T>& aVal) { + return aVal.get(); +} + +template <typename T> +T&& ArgType(T& aVal) { + return std::move(aVal); +} + +using ::ImplCycleCollectionUnlink; + +template <typename ResolveCallback, typename RejectCallback, typename ArgsTuple, + typename JSArgsTuple> +class NativeThenHandler; + +template <typename ResolveCallback, typename RejectCallback, typename... Args, + typename... JSArgs> +class NativeThenHandler<ResolveCallback, RejectCallback, std::tuple<Args...>, + std::tuple<JSArgs...>> + final : public PromiseNativeThenHandlerBase { + public: + /** + * @param aPromise A promise that will be settled by the result of the + * callbacks. Any thrown value to ErrorResult passed to those callbacks will + * be used to reject the promise, otherwise the promise will be resolved with + * the return value. + * @param aOnResolve A resolve callback + * @param aOnReject A reject callback + * @param aArgs The custom arguments to be passed to the both callbacks. The + * handler class will grab them to make them live long enough and to allow + * cycle collection. + * @param aJSArgs The JS arguments to be passed to the both callbacks, after + * native arguments. The handler will also grab them and allow garbage + * collection. + * + * XXX(krosylight): ideally there should be two signatures, with or without a + * promise parameter. Unfortunately doing so confuses the compiler and errors + * out, because nothing prevents promise from being ResolveCallback. + */ + NativeThenHandler(Promise* aPromise, Maybe<ResolveCallback>&& aOnResolve, + Maybe<RejectCallback>&& aOnReject, + std::tuple<std::remove_reference_t<Args>...>&& aArgs, + std::tuple<std::remove_reference_t<JSArgs>...>&& aJSArgs) + : PromiseNativeThenHandlerBase(aPromise), + mOnResolve(std::forward<Maybe<ResolveCallback>>(aOnResolve)), + mOnReject(std::forward<Maybe<RejectCallback>>(aOnReject)), + mArgs(std::forward<decltype(aArgs)>(aArgs)), + mJSArgs(std::forward<decltype(aJSArgs)>(aJSArgs)) { + if constexpr (std::tuple_size<decltype(mJSArgs)>::value > 0) { + mozilla::HoldJSObjects(this); + } + } + + protected: + ~NativeThenHandler() override { + if constexpr (std::tuple_size<decltype(mJSArgs)>::value > 0) { + mozilla::DropJSObjects(this); + } + } + + void Traverse(nsCycleCollectionTraversalCallback& cb) override { + auto* tmp = this; + NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mArgs) + } + + void Unlink() override { + auto* tmp = this; + NS_IMPL_CYCLE_COLLECTION_UNLINK(mArgs) + NS_IMPL_CYCLE_COLLECTION_UNLINK(mJSArgs) + } + + void Trace(const TraceCallbacks& aCallbacks, void* aClosure) override { + std::apply( + [&aCallbacks, aClosure](auto&&... aArgs) { + (aCallbacks.Trace(&aArgs, "mJSArgs[]", aClosure), ...); + }, + mJSArgs); + } + + bool HasResolvedCallback() override { return mOnResolve.isSome(); } + bool HasRejectedCallback() override { return mOnReject.isSome(); } + + MOZ_CAN_RUN_SCRIPT already_AddRefed<Promise> CallResolveCallback( + JSContext* aCx, JS::Handle<JS::Value> aValue, ErrorResult& aRv) override { + return CallCallback(aCx, *mOnResolve, aValue, aRv); + } + MOZ_CAN_RUN_SCRIPT already_AddRefed<Promise> CallRejectCallback( + JSContext* aCx, JS::Handle<JS::Value> aValue, ErrorResult& aRv) override { + return CallCallback(aCx, *mOnReject, aValue, aRv); + } + + // mJSArgs are marked with Trace() above, so they can be safely converted to + // Handles. But we should not circumvent the read barrier, so call + // exposeToActiveJS explicitly. + template <typename T> + static JS::Handle<T> GetJSArgHandleForCall(JS::Heap<T>& aArg) { + aArg.exposeToActiveJS(); + return JS::Handle<T>::fromMarkedLocation(aArg.address()); + } + + template <typename TCallback, size_t... Indices, size_t... JSIndices> + MOZ_CAN_RUN_SCRIPT already_AddRefed<Promise> CallCallback( + JSContext* aCx, const TCallback& aHandler, JS::Handle<JS::Value> aValue, + ErrorResult& aRv, std::index_sequence<Indices...>, + std::index_sequence<JSIndices...>) { + return aHandler(aCx, aValue, aRv, ArgType(std::get<Indices>(mArgs))..., + GetJSArgHandleForCall(std::get<JSIndices>(mJSArgs))...); + } + + template <typename TCallback> + MOZ_CAN_RUN_SCRIPT already_AddRefed<Promise> CallCallback( + JSContext* aCx, const TCallback& aHandler, JS::Handle<JS::Value> aValue, + ErrorResult& aRv) { + return CallCallback(aCx, aHandler, aValue, aRv, + std::index_sequence_for<Args...>{}, + std::index_sequence_for<JSArgs...>{}); + } + + Maybe<ResolveCallback> mOnResolve; + Maybe<RejectCallback> mOnReject; + + std::tuple<StorageType<Args>...> mArgs; + std::tuple<StorageType<JSArgs>...> mJSArgs; +}; + +} // anonymous namespace + +template <typename ResolveCallback, typename RejectCallback, typename... Args, + typename... JSArgs> +Result<RefPtr<Promise>, nsresult> +Promise::ThenCatchWithCycleCollectedArgsJSImpl( + Maybe<ResolveCallback>&& aOnResolve, Maybe<RejectCallback>&& aOnReject, + std::tuple<Args...>&& aArgs, std::tuple<JSArgs...>&& aJSArgs) { + using HandlerType = + NativeThenHandler<ResolveCallback, RejectCallback, std::tuple<Args...>, + std::tuple<JSArgs...>>; + + ErrorResult rv; + RefPtr<Promise> promise = Promise::Create(GetParentObject(), rv); + if (rv.Failed()) { + return Err(rv.StealNSResult()); + } + + auto* handler = new (fallible) + HandlerType(promise, std::forward<Maybe<ResolveCallback>>(aOnResolve), + std::forward<Maybe<RejectCallback>>(aOnReject), + std::forward<std::tuple<Args...>>(aArgs), + std::forward<std::tuple<JSArgs...>>(aJSArgs)); + + if (!handler) { + return Err(NS_ERROR_OUT_OF_MEMORY); + } + + AppendNativeHandler(handler); + return std::move(promise); +} + +template <typename ResolveCallback, typename RejectCallback, typename... Args> +Promise::ThenResult<ResolveCallback, Args...> +Promise::ThenCatchWithCycleCollectedArgsImpl( + Maybe<ResolveCallback>&& aOnResolve, Maybe<RejectCallback>&& aOnReject, + Args&&... aArgs) { + return ThenCatchWithCycleCollectedArgsJSImpl( + std::forward<Maybe<ResolveCallback>>(aOnResolve), + std::forward<Maybe<RejectCallback>>(aOnReject), + std::make_tuple(std::forward<Args>(aArgs)...), std::make_tuple()); +} + +template <typename ResolveCallback, typename RejectCallback, typename... Args> +Promise::ThenResult<ResolveCallback, Args...> +Promise::ThenCatchWithCycleCollectedArgs(ResolveCallback&& aOnResolve, + RejectCallback&& aOnReject, + Args&&... aArgs) { + return ThenCatchWithCycleCollectedArgsImpl(Some(aOnResolve), Some(aOnReject), + std::forward<Args>(aArgs)...); +} + +template <typename Callback, typename... Args> +Promise::ThenResult<Callback, Args...> Promise::ThenWithCycleCollectedArgs( + Callback&& aOnResolve, Args&&... aArgs) { + return ThenCatchWithCycleCollectedArgsImpl(Some(aOnResolve), + Maybe<Callback>(Nothing()), + std::forward<Args>(aArgs)...); +} + +template <typename Callback, typename... Args> +Promise::ThenResult<Callback, Args...> Promise::CatchWithCycleCollectedArgs( + Callback&& aOnReject, Args&&... aArgs) { + return ThenCatchWithCycleCollectedArgsImpl(Maybe<Callback>(Nothing()), + Some(aOnReject), + std::forward<Args>(aArgs)...); +} + +template <typename ResolveCallback, typename RejectCallback, typename ArgsTuple, + typename JSArgsTuple> +Result<RefPtr<Promise>, nsresult> Promise::ThenCatchWithCycleCollectedArgsJS( + ResolveCallback&& aOnResolve, RejectCallback&& aOnReject, ArgsTuple&& aArgs, + JSArgsTuple&& aJSArgs) { + return ThenCatchWithCycleCollectedArgsJSImpl( + Some(aOnResolve), Some(aOnReject), std::forward<ArgsTuple>(aArgs), + std::forward<JSArgsTuple>(aJSArgs)); +} + +template <typename Callback, typename ArgsTuple, typename JSArgsTuple> +Result<RefPtr<Promise>, nsresult> Promise::ThenWithCycleCollectedArgsJS( + Callback&& aOnResolve, ArgsTuple&& aArgs, JSArgsTuple&& aJSArgs) { + return ThenCatchWithCycleCollectedArgsJSImpl( + Some(aOnResolve), Maybe<Callback>(Nothing()), + std::forward<ArgsTuple>(aArgs), std::forward<JSArgsTuple>(aJSArgs)); +} + +template <typename ResolveCallback, typename RejectCallback, typename... Args> +void Promise::AddCallbacksWithCycleCollectedArgs(ResolveCallback&& aOnResolve, + RejectCallback&& aOnReject, + Args&&... aArgs) { + auto onResolve = + [aOnResolve](JSContext* aCx, JS::Handle<JS::Value> value, + ErrorResult& aRv, + StorageType<Args>&&... aArgs) -> already_AddRefed<Promise> { + aOnResolve(aCx, value, aRv, aArgs...); + return nullptr; + }; + auto onReject = + [aOnReject](JSContext* aCx, JS::Handle<JS::Value> value, ErrorResult& aRv, + StorageType<Args>&&... aArgs) -> already_AddRefed<Promise> { + aOnReject(aCx, value, aRv, aArgs...); + return nullptr; + }; + + // Note: explicit template parameters for clang<7/gcc<8 without "Template + // argument deduction for class templates" support + AppendNativeHandler( + new NativeThenHandler<decltype(onResolve), decltype(onReject), + std::tuple<Args...>, std::tuple<>>( + nullptr, Some(onResolve), Some(onReject), + std::make_tuple(std::forward<Args>(aArgs)...), std::make_tuple())); +} + +} // namespace mozilla::dom + +#endif // mozilla_dom_Promise_inl_h |