| // Copyright 2019 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/message_loop/message_pump_kqueue.h" |
| |
| #include <mach/mach.h> |
| #include <mach/message.h> |
| |
| #include <utility> |
| |
| #include "base/functional/bind.h" |
| #include "base/memory/ptr_util.h" |
| #include "base/memory/raw_ptr.h" |
| #include "base/memory/scoped_refptr.h" |
| #include "base/run_loop.h" |
| #include "base/task/single_thread_task_executor.h" |
| #include "base/task/single_thread_task_runner.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace base { |
| namespace { |
| |
| class MessagePumpKqueueTest : public testing::Test { |
| public: |
| MessagePumpKqueueTest() |
| : pump_(new MessagePumpKqueue()), executor_(WrapUnique(pump_.get())) {} |
| |
| MessagePumpKqueue* pump() { return pump_; } |
| |
| static void CreatePortPair(mac::ScopedMachReceiveRight* receive, |
| mac::ScopedMachSendRight* send) { |
| mach_port_options_t options{}; |
| options.flags = MPO_INSERT_SEND_RIGHT; |
| mac::ScopedMachReceiveRight port; |
| kern_return_t kr = mach_port_construct( |
| mach_task_self(), &options, 0, |
| mac::ScopedMachReceiveRight::Receiver(*receive).get()); |
| ASSERT_EQ(kr, KERN_SUCCESS); |
| *send = mac::ScopedMachSendRight(receive->get()); |
| } |
| |
| static mach_msg_return_t SendEmptyMessage(mach_port_t remote_port, |
| mach_msg_id_t msgid) { |
| mach_msg_empty_send_t message{}; |
| message.header.msgh_bits = MACH_MSGH_BITS_REMOTE(MACH_MSG_TYPE_COPY_SEND); |
| message.header.msgh_size = sizeof(message); |
| message.header.msgh_remote_port = remote_port; |
| message.header.msgh_id = msgid; |
| return mach_msg_send(&message.header); |
| } |
| |
| private: |
| raw_ptr<MessagePumpKqueue> pump_; // Weak, owned by |executor_|. |
| SingleThreadTaskExecutor executor_; |
| }; |
| |
| class PortWatcher : public MessagePumpKqueue::MachPortWatcher { |
| public: |
| PortWatcher(RepeatingClosure callback) : callback_(std::move(callback)) {} |
| ~PortWatcher() override {} |
| |
| void OnMachMessageReceived(mach_port_t port) override { |
| mach_msg_empty_rcv_t message{}; |
| kern_return_t kr = mach_msg(&message.header, MACH_RCV_MSG, 0, |
| sizeof(message), port, 0, MACH_PORT_NULL); |
| ASSERT_EQ(kr, KERN_SUCCESS); |
| |
| messages_.push_back(message.header); |
| |
| callback_.Run(); |
| } |
| |
| std::vector<mach_msg_header_t> messages_; |
| |
| private: |
| RepeatingClosure callback_; |
| }; |
| |
| TEST_F(MessagePumpKqueueTest, MachPortBasicWatch) { |
| mac::ScopedMachReceiveRight port; |
| mac::ScopedMachSendRight send_right; |
| CreatePortPair(&port, &send_right); |
| |
| mach_msg_id_t msgid = 'helo'; |
| |
| RunLoop run_loop; |
| PortWatcher watcher(run_loop.QuitClosure()); |
| MessagePumpKqueue::MachPortWatchController controller(FROM_HERE); |
| |
| SingleThreadTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, BindOnce( |
| [](mach_port_t port, mach_msg_id_t msgid, RunLoop* loop) { |
| mach_msg_return_t kr = SendEmptyMessage(port, msgid); |
| EXPECT_EQ(kr, KERN_SUCCESS); |
| if (kr != KERN_SUCCESS) { |
| loop->Quit(); |
| } |
| }, |
| port.get(), msgid, Unretained(&run_loop))); |
| |
| pump()->WatchMachReceivePort(port.get(), &controller, &watcher); |
| |
| run_loop.Run(); |
| |
| ASSERT_EQ(1u, watcher.messages_.size()); |
| EXPECT_EQ(port.get(), watcher.messages_[0].msgh_local_port); |
| EXPECT_EQ(msgid, watcher.messages_[0].msgh_id); |
| } |
| |
| TEST_F(MessagePumpKqueueTest, MachPortStopWatching) { |
| mac::ScopedMachReceiveRight port; |
| mac::ScopedMachSendRight send_right; |
| CreatePortPair(&port, &send_right); |
| |
| RunLoop run_loop; |
| PortWatcher watcher(run_loop.QuitClosure()); |
| MessagePumpKqueue::MachPortWatchController controller(FROM_HERE); |
| |
| pump()->WatchMachReceivePort(port.get(), &controller, &watcher); |
| |
| SingleThreadTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, |
| BindOnce( |
| [](MessagePumpKqueue::MachPortWatchController* controller) { |
| controller->StopWatchingMachPort(); |
| }, |
| Unretained(&controller))); |
| |
| SingleThreadTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, BindOnce( |
| [](mach_port_t port) { |
| EXPECT_EQ(KERN_SUCCESS, SendEmptyMessage(port, 100)); |
| }, |
| port.get())); |
| |
| run_loop.RunUntilIdle(); |
| |
| EXPECT_EQ(0u, watcher.messages_.size()); |
| } |
| |
| TEST_F(MessagePumpKqueueTest, MultipleMachWatchers) { |
| mac::ScopedMachReceiveRight port1, port2; |
| mac::ScopedMachSendRight send_right1, send_right2; |
| CreatePortPair(&port1, &send_right1); |
| CreatePortPair(&port2, &send_right2); |
| |
| RunLoop run_loop; |
| |
| int port1_count = 0, port2_count = 0; |
| |
| // Whenever port1 receives a message, it will send to port2. |
| // Whenever port2 receives a message, it will send to port1. |
| // When port2 has sent 3 messages to port1, it will stop. |
| |
| PortWatcher watcher1(BindRepeating( |
| [](mach_port_t port2, int* port2_count, RunLoop* loop) { |
| mach_msg_id_t id = (0x2 << 16) | ++(*port2_count); |
| mach_msg_return_t kr = SendEmptyMessage(port2, id); |
| EXPECT_EQ(kr, KERN_SUCCESS); |
| if (kr != KERN_SUCCESS) { |
| loop->Quit(); |
| } |
| }, |
| port2.get(), &port2_count, &run_loop)); |
| MessagePumpKqueue::MachPortWatchController controller1(FROM_HERE); |
| |
| PortWatcher watcher2(BindRepeating( |
| [](mach_port_t port1, int* port1_count, RunLoop* loop) { |
| if (*port1_count == 3) { |
| loop->Quit(); |
| return; |
| } |
| mach_msg_id_t id = (0x1 << 16) | ++(*port1_count); |
| mach_msg_return_t kr = SendEmptyMessage(port1, id); |
| EXPECT_EQ(kr, KERN_SUCCESS); |
| if (kr != KERN_SUCCESS) { |
| loop->Quit(); |
| } |
| }, |
| port1.get(), &port1_count, &run_loop)); |
| MessagePumpKqueue::MachPortWatchController controller2(FROM_HERE); |
| |
| pump()->WatchMachReceivePort(port1.get(), &controller1, &watcher1); |
| pump()->WatchMachReceivePort(port2.get(), &controller2, &watcher2); |
| |
| // Start ping-ponging with by sending the first message to port1. |
| SingleThreadTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, BindOnce( |
| [](mach_port_t port1) { |
| ASSERT_EQ(KERN_SUCCESS, |
| SendEmptyMessage(port1, 0xf000f)); |
| }, |
| port1.get())); |
| |
| run_loop.Run(); |
| |
| ASSERT_EQ(4u, watcher1.messages_.size()); |
| ASSERT_EQ(4u, watcher2.messages_.size()); |
| |
| EXPECT_EQ(0xf000f, watcher1.messages_[0].msgh_id); |
| EXPECT_EQ(0x10001, watcher1.messages_[1].msgh_id); |
| EXPECT_EQ(0x10002, watcher1.messages_[2].msgh_id); |
| EXPECT_EQ(0x10003, watcher1.messages_[3].msgh_id); |
| |
| EXPECT_EQ(0x20001, watcher2.messages_[0].msgh_id); |
| EXPECT_EQ(0x20002, watcher2.messages_[1].msgh_id); |
| EXPECT_EQ(0x20003, watcher2.messages_[2].msgh_id); |
| EXPECT_EQ(0x20004, watcher2.messages_[3].msgh_id); |
| } |
| |
| } // namespace |
| } // namespace base |