/* -*- 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/. */ #include "js/Value.h" #include "nsThreadUtils.h" #include "mozilla/CycleCollectedJSRuntime.h" #include "mozilla/ThreadLocal.h" #include "mozilla/TimeStamp.h" #include "mozilla/dom/BindingDeclarations.h" #include "mozilla/dom/ContentChild.h" #include "mozilla/dom/Promise.h" #include "mozilla/dom/PromiseBinding.h" #include "mozilla/dom/PromiseDebugging.h" #include "mozilla/dom/PromiseDebuggingBinding.h" namespace mozilla { namespace dom { class FlushRejections: public nsCancelableRunnable { public: static void Init() { if (!sDispatched.init()) { MOZ_CRASH("Could not initialize FlushRejections::sDispatched"); } sDispatched.set(false); } static void DispatchNeeded() { if (sDispatched.get()) { // An instance of `FlushRejections` has already been dispatched // and not run yet. No need to dispatch another one. return; } sDispatched.set(true); NS_DispatchToCurrentThread(new FlushRejections()); } static void FlushSync() { sDispatched.set(false); // Call the callbacks if necessary. // Note that these callbacks may in turn cause Promise to turn // uncaught or consumed. Since `sDispatched` is `false`, // `FlushRejections` will be called once again, on an ulterior // tick. PromiseDebugging::FlushUncaughtRejectionsInternal(); } NS_IMETHOD Run() { FlushSync(); return NS_OK; } private: // `true` if an instance of `FlushRejections` is currently dispatched // and has not been executed yet. static MOZ_THREAD_LOCAL(bool) sDispatched; }; /* static */ MOZ_THREAD_LOCAL(bool) FlushRejections::sDispatched; static Promise* UnwrapPromise(JS::Handle aPromise, ErrorResult& aRv) { Promise* promise; if (NS_WARN_IF(NS_FAILED(UNWRAP_OBJECT(Promise, aPromise, promise)))) { aRv.ThrowTypeError(NS_LITERAL_STRING("Argument")); return nullptr; } return promise; } /* static */ void PromiseDebugging::GetState(GlobalObject&, JS::Handle aPromise, PromiseDebuggingStateHolder& aState, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return; } switch (promise->mState) { case Promise::Pending: aState.mState = PromiseDebuggingState::Pending; break; case Promise::Resolved: aState.mState = PromiseDebuggingState::Fulfilled; JS::ExposeValueToActiveJS(promise->mResult); aState.mValue = promise->mResult; break; case Promise::Rejected: aState.mState = PromiseDebuggingState::Rejected; JS::ExposeValueToActiveJS(promise->mResult); aState.mReason = promise->mResult; break; } } /*static */ nsString PromiseDebugging::sIDPrefix; /* static */ void PromiseDebugging::Init() { FlushRejections::Init(); // Generate a prefix for identifiers: "PromiseDebugging.$processid." sIDPrefix = NS_LITERAL_STRING("PromiseDebugging."); if (XRE_IsContentProcess()) { sIDPrefix.AppendInt(ContentChild::GetSingleton()->GetID()); sIDPrefix.Append('.'); } else { sIDPrefix.AppendLiteral("0."); } } /* static */ void PromiseDebugging::Shutdown() { sIDPrefix.SetIsVoid(true); } /* static */ void PromiseDebugging::FlushUncaughtRejections() { MOZ_ASSERT(!NS_IsMainThread()); FlushRejections::FlushSync(); } /* static */ void PromiseDebugging::GetAllocationStack(GlobalObject&, JS::Handle aPromise, JS::MutableHandle aStack, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return; } aStack.set(promise->mAllocationStack); } /* static */ void PromiseDebugging::GetRejectionStack(GlobalObject&, JS::Handle aPromise, JS::MutableHandle aStack, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return; } aStack.set(promise->mRejectionStack); } /* static */ void PromiseDebugging::GetFullfillmentStack(GlobalObject&, JS::Handle aPromise, JS::MutableHandle aStack, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return; } aStack.set(promise->mFullfillmentStack); } /* static */ void PromiseDebugging::GetDependentPromises(GlobalObject&, JS::Handle aPromise, nsTArray>& aPromises, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return; } promise->GetDependentPromises(aPromises); } /* static */ double PromiseDebugging::GetPromiseLifetime(GlobalObject&, JS::Handle aPromise, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return 0; } return (TimeStamp::Now() - promise->mCreationTimestamp).ToMilliseconds(); } /* static */ double PromiseDebugging::GetTimeToSettle(GlobalObject&, JS::Handle aPromise, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return 0; } if (promise->mState == Promise::Pending) { aRv.Throw(NS_ERROR_UNEXPECTED); return 0; } return (promise->mSettlementTimestamp - promise->mCreationTimestamp).ToMilliseconds(); } /* static */ void PromiseDebugging::AddUncaughtRejectionObserver(GlobalObject&, UncaughtRejectionObserver& aObserver) { CycleCollectedJSRuntime* storage = CycleCollectedJSRuntime::Get(); nsTArray>& observers = storage->mUncaughtRejectionObservers; observers.AppendElement(&aObserver); } /* static */ bool PromiseDebugging::RemoveUncaughtRejectionObserver(GlobalObject&, UncaughtRejectionObserver& aObserver) { CycleCollectedJSRuntime* storage = CycleCollectedJSRuntime::Get(); nsTArray>& observers = storage->mUncaughtRejectionObservers; for (size_t i = 0; i < observers.Length(); ++i) { UncaughtRejectionObserver* observer = static_cast(observers[i].get()); if (*observer == aObserver) { observers.RemoveElementAt(i); return true; } } return false; } /* static */ void PromiseDebugging::AddUncaughtRejection(Promise& aPromise) { CycleCollectedJSRuntime::Get()->mUncaughtRejections.AppendElement(&aPromise); FlushRejections::DispatchNeeded(); } /* void */ void PromiseDebugging::AddConsumedRejection(Promise& aPromise) { CycleCollectedJSRuntime::Get()->mConsumedRejections.AppendElement(&aPromise); FlushRejections::DispatchNeeded(); } /* static */ void PromiseDebugging::GetPromiseID(GlobalObject&, JS::Handle aPromise, nsString& aID, ErrorResult& aRv) { Promise* promise = UnwrapPromise(aPromise, aRv); if (aRv.Failed()) { return; } uint64_t promiseID = promise->GetID(); aID = sIDPrefix; aID.AppendInt(promiseID); } /* static */ void PromiseDebugging::FlushUncaughtRejectionsInternal() { CycleCollectedJSRuntime* storage = CycleCollectedJSRuntime::Get(); // The Promise that have been left uncaught (rejected and last in // their chain) since the last call to this function. nsTArray> uncaught; storage->mUncaughtRejections.SwapElements(uncaught); // The Promise that have been left uncaught at some point, but that // have eventually had their `then` method called. nsTArray> consumed; storage->mConsumedRejections.SwapElements(consumed); nsTArray>& observers = storage->mUncaughtRejectionObservers; nsresult rv; // Notify observers of uncaught Promise. for (size_t i = 0; i < uncaught.Length(); ++i) { nsCOMPtr promise = do_QueryInterface(uncaught[i], &rv); MOZ_ASSERT(NS_SUCCEEDED(rv)); if (!promise->IsLastInChain()) { // This promise is not the last in the chain anymore, // so the error has been caught at some point. continue; } // For the moment, the Promise is still at the end of the // chain. Let's inform observers, so that they may decide whether // to report it. for (size_t j = 0; j < observers.Length(); ++j) { ErrorResult err; RefPtr obs = static_cast(observers[j].get()); obs->OnLeftUncaught(*promise, err); // Ignore errors } promise->SetNotifiedAsUncaught(); } // Notify observers of consumed Promise. for (size_t i = 0; i < consumed.Length(); ++i) { nsCOMPtr promise = do_QueryInterface(consumed[i], &rv); MOZ_ASSERT(NS_SUCCEEDED(rv)); if (!promise->WasNotifiedAsUncaught()) { continue; } MOZ_ASSERT(!promise->IsLastInChain()); for (size_t j = 0; j < observers.Length(); ++j) { ErrorResult err; RefPtr obs = static_cast(observers[j].get()); obs->OnConsumed(*promise, err); // Ignore errors } } } } // namespace dom } // namespace mozilla