ladybird/Libraries/LibWeb/HTML/EventLoop/Task.cpp
Andreas Kling 164ed80244 Meta: Enable exit-time destructor warnings for libraries
Enable -Wexit-time-destructors for all in-tree library targets and
update process-lifetime library statics so they no longer register
exit-time destructors. Long-lived caches, lookup tables, singleton
registries, and generated constants now use NeverDestroyed or leaked
references where the data is intended to live until process exit.

Update LibWeb, LibLine, and the binding generators so regenerated
sources follow the same rule instead of reintroducing destructed
statics.
2026-06-04 19:20:49 +02:00

95 lines
2.2 KiB
C++

/*
* Copyright (c) 2021-2024, Andreas Kling <andreas@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <AK/IDAllocator.h>
#include <AK/NeverDestroyed.h>
#include <LibWeb/DOM/Document.h>
#include <LibWeb/HTML/EventLoop/Task.h>
namespace Web::HTML {
GC_DEFINE_ALLOCATOR(Task);
static IDAllocator& unique_task_source_allocator()
{
static NeverDestroyed<IDAllocator> allocator { static_cast<int>(Task::Source::UniqueTaskSourceStart) };
return *allocator;
}
[[nodiscard]] static TaskID allocate_task_id()
{
static u64 next_task_id = 1;
return next_task_id++;
}
GC::Ref<Task> Task::create(JS::VM& vm, Source source, GC::Ptr<DOM::Document const> document, GC::Ref<GC::Function<void()>> steps)
{
return vm.heap().allocate<Task>(source, document, move(steps));
}
Task::Task(Source source, GC::Ptr<DOM::Document const> document, GC::Ref<GC::Function<void()>> steps)
: m_id(allocate_task_id())
, m_source(source)
, m_steps(steps)
, m_document(document)
{
}
Task::~Task() = default;
void Task::visit_edges(Visitor& visitor)
{
Base::visit_edges(visitor);
visitor.visit(m_steps);
visitor.visit(m_document);
}
void Task::execute()
{
m_steps->function()();
}
// https://html.spec.whatwg.org/multipage/webappapis.html#concept-task-runnable
bool Task::is_runnable() const
{
// A task is runnable if its document is either null or fully active.
return !m_document || m_document->is_fully_active();
}
bool Task::is_permanently_unrunnable() const
{
return m_document && m_document->has_been_destroyed();
}
DOM::Document const* Task::document() const
{
return m_document.ptr();
}
UniqueTaskSource::UniqueTaskSource()
: source(static_cast<Task::Source>(unique_task_source_allocator().allocate()))
{
}
UniqueTaskSource::~UniqueTaskSource()
{
unique_task_source_allocator().deallocate(static_cast<int>(source));
}
NonnullRefPtr<ParallelQueue> ParallelQueue::create()
{
return adopt_ref(*new (nothrow) ParallelQueue);
}
TaskID ParallelQueue::enqueue(GC::Ref<GC::Function<void()>> algorithm)
{
auto& event_loop = HTML::main_thread_event_loop();
auto task = HTML::Task::create(event_loop.vm(), m_task_source.source, nullptr, algorithm);
event_loop.task_queue().add(task);
return task->id();
}
}