/* * Copyright (c) 2023, Tim Flynn * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include namespace JS { JS_DEFINE_ALLOCATOR(IteratorHelper); ThrowCompletionOr> IteratorHelper::create(Realm& realm, NonnullGCPtr underlying_iterator, Closure closure, Optional abrupt_closure) { return realm.heap().allocate(realm, realm, realm.intrinsics().iterator_helper_prototype(), move(underlying_iterator), move(closure), move(abrupt_closure)); } IteratorHelper::IteratorHelper(Realm& realm, Object& prototype, NonnullGCPtr underlying_iterator, Closure closure, Optional abrupt_closure) : GeneratorObject(realm, prototype, realm.vm().running_execution_context().copy(), "Iterator Helper"sv) , m_underlying_iterator(move(underlying_iterator)) , m_closure(move(closure)) , m_abrupt_closure(move(abrupt_closure)) { } void IteratorHelper::visit_edges(Visitor& visitor) { Base::visit_edges(visitor); visitor.visit(m_underlying_iterator); } Value IteratorHelper::result(Value value) { set_generator_state(value.is_undefined() ? GeneratorState::Completed : GeneratorState::SuspendedYield); return value; } ThrowCompletionOr IteratorHelper::close_result(VM& vm, Completion completion) { set_generator_state(GeneratorState::Completed); return *TRY(iterator_close(vm, underlying_iterator(), move(completion))); } ThrowCompletionOr IteratorHelper::execute(VM& vm, JS::Completion const& completion) { ScopeGuard guard { [&] { vm.pop_execution_context(); } }; if (completion.is_abrupt()) { if (m_abrupt_closure.has_value()) return (*m_abrupt_closure)(vm, *this, completion); return close_result(vm, completion); } auto result_value = m_closure(vm, *this); if (result_value.is_throw_completion()) { set_generator_state(GeneratorState::Completed); return result_value; } return create_iterator_result_object(vm, result(result_value.release_value()), generator_state() == GeneratorState::Completed); } }