// Copyright 2019 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef BASE_MESSAGE_LOOP_MESSAGE_PUMP_KQUEUE_H_ #define BASE_MESSAGE_LOOP_MESSAGE_PUMP_KQUEUE_H_ #include #include #include #include #include "base/containers/id_map.h" #include "base/files/scoped_file.h" #include "base/location.h" #include "base/mac/scoped_mach_port.h" #include "base/macros.h" #include "base/memory/weak_ptr.h" #include "base/message_loop/message_pump.h" #include "base/message_loop/watchable_io_message_pump_posix.h" namespace base { // MessagePumpKqueue is used on macOS to drive an IO MessageLoop that is // capable of watching both POSIX file descriptors and Mach ports. class BASE_EXPORT MessagePumpKqueue : public MessagePump, public WatchableIOMessagePumpPosix { public: class FdWatchController : public FdWatchControllerInterface { public: explicit FdWatchController(const Location& from_here); ~FdWatchController() override; // FdWatchControllerInterface: bool StopWatchingFileDescriptor() override; protected: friend class MessagePumpKqueue; void Init(WeakPtr pump, int fd, int mode, FdWatcher* watcher); void Reset(); int fd() { return fd_; } int mode() { return mode_; } FdWatcher* watcher() { return watcher_; } private: int fd_ = -1; int mode_ = 0; FdWatcher* watcher_ = nullptr; WeakPtr pump_; DISALLOW_COPY_AND_ASSIGN(FdWatchController); }; // Delegate interface that provides notifications of Mach message receive // events. class MachPortWatcher { public: virtual ~MachPortWatcher() {} virtual void OnMachMessageReceived(mach_port_t port) = 0; }; // Controller interface that is used to stop receiving events for an // installed MachPortWatcher. class MachPortWatchController { public: explicit MachPortWatchController(const Location& from_here); ~MachPortWatchController(); bool StopWatchingMachPort(); protected: friend class MessagePumpKqueue; void Init(WeakPtr pump, mach_port_t port, MachPortWatcher* watcher); void Reset(); mach_port_t port() { return port_; } MachPortWatcher* watcher() { return watcher_; } private: mach_port_t port_ = MACH_PORT_NULL; MachPortWatcher* watcher_ = nullptr; WeakPtr pump_; const Location from_here_; DISALLOW_COPY_AND_ASSIGN(MachPortWatchController); }; MessagePumpKqueue(); ~MessagePumpKqueue() override; // MessagePump: void Run(Delegate* delegate) override; void Quit() override; void ScheduleWork() override; void ScheduleDelayedWork(const TimeTicks& delayed_work_time) override; // Begins watching the Mach receive right named by |port|. The |controller| // can be used to stop watching for incoming messages, and new message // notifications are delivered to the |delegate|. Returns true if the watch // was successfully set-up and false on error. bool WatchMachReceivePort(mach_port_t port, MachPortWatchController* controller, MachPortWatcher* delegate); // WatchableIOMessagePumpPosix: bool WatchFileDescriptor(int fd, bool persistent, int mode, FdWatchController* controller, FdWatcher* delegate); private: // Called by the watch controller implementations to stop watching the // respective types of handles. bool StopWatchingMachPort(MachPortWatchController* controller); bool StopWatchingFileDescriptor(FdWatchController* controller); // Checks the |kqueue_| for events. If |next_work_info| is null, then the // kqueue will be polled for events. If it is non-null, it will wait for the // amount of time specified by the NextWorkInfo or until an event is // triggered. Returns whether any events were dispatched, with the events // stored in |events_|. bool DoInternalWork(Delegate::NextWorkInfo* next_work_info); // Called by DoInternalWork() to dispatch the user events stored in |events_| // that were triggered. |count| is the number of events to process. Returns // true if work was done, or false if no work was done. bool ProcessEvents(int count); // Receive right to which an empty Mach message is sent to wake up the pump // in response to ScheduleWork(). mac::ScopedMachReceiveRight wakeup_; // Scratch buffer that is used to receive the message sent to |wakeup_|. mach_msg_empty_rcv_t wakeup_buffer_; // A Mach port set used to watch ports from WatchMachReceivePort(). This is // only used on macOS <10.12, where kqueues cannot watch ports directly. mac::ScopedMachPortSet port_set_; // Watch controllers for FDs. IDs are generated by the map and are stored in // the kevent64_s::udata field. IDMap fd_controllers_; // Watch controllers for Mach ports. IDs are the port being watched. IDMap port_controllers_; // The kqueue that drives the pump. ScopedFD kqueue_; // Whether the pump has been Quit() or not. bool keep_running_ = true; // The number of events scheduled on the |kqueue_|. There is always at least // 1, for the |wakeup_| port (or |port_set_|). size_t event_count_ = 1; // Buffer used by DoInternalWork() to be notified of triggered events. This // is always at least |event_count_|-sized. std::vector events_{event_count_}; WeakPtrFactory weak_factory_; DISALLOW_COPY_AND_ASSIGN(MessagePumpKqueue); }; } // namespace base #endif // BASE_MESSAGE_LOOP_MESSAGE_PUMP_KQUEUE_H_