ladybird/Libraries/LibCore/EventLoopImplementation.h
Andreas Kling 96ce468b60 LibCore: Add thread-safe weak deferred_invoke()
Add a thread-safe deferred_invoke() API on WeakEventLoopReference that
queues work onto the owning thread's event queue and wakes that thread
via EventLoopManager hooks. This avoids calling wake() from foreign
threads during teardown.

Implement current_thread_handle()/wake_thread() in each backend and
track per-thread data so handles are validated before waking:

- Unix: wake via per-thread wake pipe
- Windows: wake via thread wake event
- macOS: wake via stored CFRunLoopRef
- Qt: wake via event target or QEventLoop::wakeUp()
- Android: wake via stored ALooper
2026-01-25 09:32:51 +01:00

68 lines
1.7 KiB
C++

/*
* Copyright (c) 2022-2023, Andreas Kling <andreas@ladybird.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#pragma once
#include <AK/Function.h>
#include <AK/Types.h>
#include <LibCore/Forward.h>
namespace Core {
class EventLoopImplementation;
class ThreadEventQueue;
using EventLoopThreadHandle = FlatPtr;
class EventLoopManager {
public:
static EventLoopManager& the();
static void install(EventLoopManager&);
virtual ~EventLoopManager();
virtual NonnullOwnPtr<EventLoopImplementation> make_implementation() = 0;
virtual intptr_t register_timer(EventReceiver&, int milliseconds, bool should_reload) = 0;
virtual void unregister_timer(intptr_t timer_id) = 0;
virtual void register_notifier(Notifier&) = 0;
virtual void unregister_notifier(Notifier&) = 0;
virtual void did_post_event() = 0;
virtual EventLoopThreadHandle current_thread_handle() = 0;
virtual void wake_thread(EventLoopThreadHandle) = 0;
// FIXME: These APIs only exist for obscure use-cases inside SerenityOS. Try to get rid of them.
virtual int register_signal(int signal_number, Function<void(int)> handler) = 0;
virtual void unregister_signal(int handler_id) = 0;
protected:
EventLoopManager();
};
class EventLoopImplementation {
public:
virtual ~EventLoopImplementation();
enum class PumpMode {
WaitForEvents,
DontWaitForEvents,
};
virtual int exec() = 0;
virtual size_t pump(PumpMode) = 0;
virtual void quit(int) = 0;
virtual void wake() = 0;
virtual bool was_exit_requested() const = 0;
virtual void deferred_invoke(Function<void()>&&);
protected:
EventLoopImplementation();
ThreadEventQueue& m_thread_event_queue;
};
}