/* -*- Mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 40 -*- */ /* vim: set ts=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/. */ #include "mozilla/dom/PromiseResolver.h" #include "mozilla/dom/PromiseBinding.h" #include "mozilla/dom/Promise.h" #include "PromiseCallback.h" namespace mozilla { namespace dom { // PromiseResolverTask // This class processes the promise's callbacks with promise's result. class PromiseResolverTask MOZ_FINAL : public nsRunnable { public: PromiseResolverTask(PromiseResolver* aResolver, const JS::Handle aValue, Promise::PromiseState aState) : mResolver(aResolver) , mValue(aValue) , mState(aState) { MOZ_ASSERT(aResolver); MOZ_ASSERT(mState != Promise::Pending); MOZ_COUNT_CTOR(PromiseResolverTask); JSContext* cx = nsContentUtils::GetSafeJSContext(); JS_AddNamedValueRootRT(JS_GetRuntime(cx), &mValue, "PromiseResolverTask.mValue"); } ~PromiseResolverTask() { MOZ_COUNT_DTOR(PromiseResolverTask); JSContext* cx = nsContentUtils::GetSafeJSContext(); JS_RemoveValueRootRT(JS_GetRuntime(cx), &mValue); } NS_IMETHOD Run() { mResolver->RunTask(JS::Handle::fromMarkedLocation(&mValue), mState, PromiseResolver::SyncTask); return NS_OK; } private: nsRefPtr mResolver; JS::Value mValue; Promise::PromiseState mState; }; // PromiseResolver NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE_1(PromiseResolver, mPromise) NS_IMPL_CYCLE_COLLECTING_ADDREF(PromiseResolver) NS_IMPL_CYCLE_COLLECTING_RELEASE(PromiseResolver) NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(PromiseResolver) NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY NS_INTERFACE_MAP_ENTRY(nsISupports) NS_INTERFACE_MAP_END PromiseResolver::PromiseResolver(Promise* aPromise) : mPromise(aPromise) , mResolvePending(false) { SetIsDOMBinding(); } JSObject* PromiseResolver::WrapObject(JSContext* aCx, JS::Handle aScope) { return PromiseResolverBinding::Wrap(aCx, aScope, this); } void PromiseResolver::Resolve(JSContext* aCx, const Optional >& aValue, PromiseTaskSync aAsynchronous) { if (mResolvePending) { return; } ResolveInternal(aCx, aValue, aAsynchronous); } void PromiseResolver::ResolveInternal(JSContext* aCx, const Optional >& aValue, PromiseTaskSync aAsynchronous) { mResolvePending = true; // TODO: Bug 879245 - Then-able objects if (aValue.WasPassed() && aValue.Value().isObject()) { JS::Rooted valueObj(aCx, &aValue.Value().toObject()); Promise* nextPromise; nsresult rv = UnwrapObject(aCx, valueObj, nextPromise); if (NS_SUCCEEDED(rv)) { nsRefPtr resolveCb = new ResolvePromiseCallback(this); nsRefPtr rejectCb = new RejectPromiseCallback(this); nextPromise->AppendCallbacks(resolveCb, rejectCb); return; } } // If the synchronous flag is set, process promise's resolve callbacks with // value. Otherwise, the synchronous flag is unset, queue a task to process // promise's resolve callbacks with value. Otherwise, the synchronous flag is // unset, queue a task to process promise's resolve callbacks with value. RunTask(aValue.WasPassed() ? aValue.Value() : JS::UndefinedHandleValue, Promise::Resolved, aAsynchronous); } void PromiseResolver::Reject(JSContext* aCx, const Optional >& aValue, PromiseTaskSync aAsynchronous) { if (mResolvePending) { return; } RejectInternal(aCx, aValue, aAsynchronous); } void PromiseResolver::RejectInternal(JSContext* aCx, const Optional >& aValue, PromiseTaskSync aAsynchronous) { mResolvePending = true; // If the synchronous flag is set, process promise's reject callbacks with // value. Otherwise, the synchronous flag is unset, queue a task to process // promise's reject callbacks with value. RunTask(aValue.WasPassed() ? aValue.Value() : JS::UndefinedHandleValue, Promise::Rejected, aAsynchronous); } void PromiseResolver::RunTask(JS::Handle aValue, Promise::PromiseState aState, PromiseTaskSync aAsynchronous) { // If the synchronous flag is unset, queue a task to process promise's // accept callbacks with value. if (aAsynchronous == AsyncTask) { nsRefPtr task = new PromiseResolverTask(this, aValue, aState); NS_DispatchToCurrentThread(task); return; } mPromise->SetResult(aValue); mPromise->SetState(aState); mPromise->RunTask(); mPromise = nullptr; } } // namespace dom } // namespace mozilla