LibJS: Convert FinalizationRegistry::cleanup to ThrowCompletionOr

This commit is contained in:
davidot 2022-02-07 14:42:24 +01:00 committed by Linus Groh
parent dcc284705b
commit f324d91106
Notes: sideshowbarker 2024-07-17 19:38:10 +09:00
3 changed files with 19 additions and 7 deletions

View file

@ -58,18 +58,30 @@ void FinalizationRegistry::remove_dead_cells(Badge<Heap>)
}
// 9.13 CleanupFinalizationRegistry ( finalizationRegistry ), https://tc39.es/ecma262/#sec-cleanup-finalization-registry
void FinalizationRegistry::cleanup(FunctionObject* callback)
ThrowCompletionOr<void> FinalizationRegistry::cleanup(FunctionObject* callback)
{
auto& vm = this->vm();
// 1. Assert: finalizationRegistry has [[Cells]] and [[CleanupCallback]] internal slots.
// Note: Ensured by type.
// 2. Let callback be finalizationRegistry.[[CleanupCallback]].
auto cleanup_callback = callback ?: m_cleanup_callback;
// 3. While finalizationRegistry.[[Cells]] contains a Record cell such that cell.[[WeakRefTarget]] is empty, an implementation may perform the following steps:
for (auto it = m_records.begin(); it != m_records.end(); ++it) {
// a. Choose any such cell.
if (it->target != nullptr)
continue;
(void)call(global_object(), *cleanup_callback, js_undefined(), it->held_value);
auto cell = *it;
// b. Remove cell from finalizationRegistry.[[Cells]].
it.remove(m_records);
if (vm.exception())
return;
// c. Perform ? HostCallJobCallback(callback, undefined, « cell.[[HeldValue]] »).
(void)TRY(call(global_object(), *cleanup_callback, js_undefined(), cell.held_value));
}
// 4. Return NormalCompletion(empty).
return {};
}
void FinalizationRegistry::visit_edges(Cell::Visitor& visitor)

View file

@ -28,7 +28,7 @@ public:
void add_finalization_record(Cell& target, Value held_value, Object* unregister_token);
bool remove_by_token(Object& unregister_token);
void cleanup(FunctionObject* callback = nullptr);
ThrowCompletionOr<void> cleanup(FunctionObject* callback = nullptr);
virtual void remove_dead_cells(Badge<Heap>) override;

View file

@ -41,7 +41,7 @@ JS_DEFINE_NATIVE_FUNCTION(FinalizationRegistryPrototype::cleanup_some)
if (vm.argument_count() > 0 && !callback.is_function())
return vm.throw_completion<TypeError>(global_object, ErrorType::NotAFunction, callback.to_string_without_side_effects());
finalization_registry->cleanup(callback.is_undefined() ? nullptr : &callback.as_function());
TRY(finalization_registry->cleanup(callback.is_undefined() ? nullptr : &callback.as_function()));
return js_undefined();
}