Files
ladybird/Libraries/LibJS/Runtime/PromiseResolvingFunction.cpp
Aliaksandr Kalenik 65b09cc4e3 LibJS: Store AlreadyResolved bit on the resolve function itself
Previously, each pair of promise resolving functions allocated a
dedicated AlreadyResolved GC cell to hold a single shared bool. Move the
bit onto the resolve function as a plain member, and have the reject
function reach it through a GC::Ptr back to its paired resolve function.
2026-05-02 20:24:47 +02:00

69 lines
2.0 KiB
C++

/*
* Copyright (c) 2021-2023, Linus Groh <linusg@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibJS/Runtime/GlobalObject.h>
#include <LibJS/Runtime/NativeFunction.h>
#include <LibJS/Runtime/Promise.h>
#include <LibJS/Runtime/PromiseResolvingFunction.h>
namespace JS {
GC_DEFINE_ALLOCATOR(PromiseResolvingFunction);
GC::Ref<PromiseResolvingFunction> PromiseResolvingFunction::create_resolve(Realm& realm, Promise& promise)
{
return realm.create<PromiseResolvingFunction>(promise, Kind::Resolve, nullptr, realm.intrinsics().function_prototype());
}
GC::Ref<PromiseResolvingFunction> PromiseResolvingFunction::create_reject(Realm& realm, Promise& promise, PromiseResolvingFunction& resolve_function)
{
return realm.create<PromiseResolvingFunction>(promise, Kind::Reject, &resolve_function, realm.intrinsics().function_prototype());
}
PromiseResolvingFunction::PromiseResolvingFunction(Promise& promise, Kind kind, GC::Ptr<PromiseResolvingFunction> resolve_function, Object& prototype)
: NativeFunction(prototype)
, m_promise(promise)
, m_resolve_function(resolve_function)
, m_kind(kind)
{
}
void PromiseResolvingFunction::initialize(Realm& realm)
{
Base::initialize(realm);
define_direct_property(vm().names.length, Value(1), Attribute::Configurable);
}
ThrowCompletionOr<Value> PromiseResolvingFunction::call()
{
switch (m_kind) {
case Kind::Resolve:
return Promise::resolve_function_steps(vm(), m_promise, already_resolved());
case Kind::Reject:
return Promise::reject_function_steps(vm(), m_promise, already_resolved());
}
VERIFY_NOT_REACHED();
}
void PromiseResolvingFunction::visit_edges(Cell::Visitor& visitor)
{
Base::visit_edges(visitor);
visitor.visit(m_promise);
if (m_resolve_function)
visitor.visit(m_resolve_function);
}
bool& PromiseResolvingFunction::already_resolved()
{
if (m_kind == Kind::Resolve)
return m_already_resolved;
VERIFY(m_resolve_function);
return m_resolve_function->m_already_resolved;
}
}