| // Copyright 2014 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. |
| |
| #include "ipc/ipc_mojo_bootstrap.h" |
| |
| #include <cstdint> |
| #include <memory> |
| #include <utility> |
| |
| #include "base/message_loop/message_loop.h" |
| #include "base/run_loop.h" |
| #include "base/threading/thread_task_runner_handle.h" |
| #include "ipc/ipc.mojom.h" |
| #include "ipc/ipc_test_base.h" |
| #include "mojo/core/test/multiprocess_test_helper.h" |
| #include "mojo/public/cpp/bindings/associated_binding.h" |
| |
| namespace { |
| |
| constexpr int32_t kTestServerPid = 42; |
| constexpr int32_t kTestClientPid = 4242; |
| |
| class Connection { |
| public: |
| explicit Connection(std::unique_ptr<IPC::MojoBootstrap> bootstrap, |
| int32_t sender_id) |
| : bootstrap_(std::move(bootstrap)) { |
| bootstrap_->Connect(&sender_, &receiver_); |
| sender_->SetPeerPid(sender_id); |
| } |
| |
| void TakeReceiver(IPC::mojom::ChannelAssociatedRequest* receiver) { |
| *receiver = std::move(receiver_); |
| } |
| |
| IPC::mojom::ChannelAssociatedPtr& GetSender() { return sender_; } |
| |
| private: |
| IPC::mojom::ChannelAssociatedPtr sender_; |
| IPC::mojom::ChannelAssociatedRequest receiver_; |
| std::unique_ptr<IPC::MojoBootstrap> bootstrap_; |
| }; |
| |
| class PeerPidReceiver : public IPC::mojom::Channel { |
| public: |
| enum class MessageExpectation { |
| kNotExpected, |
| kExpectedValid, |
| kExpectedInvalid |
| }; |
| |
| PeerPidReceiver( |
| IPC::mojom::ChannelAssociatedRequest request, |
| const base::Closure& on_peer_pid_set, |
| MessageExpectation message_expectation = MessageExpectation::kNotExpected) |
| : binding_(this, std::move(request)), |
| on_peer_pid_set_(on_peer_pid_set), |
| message_expectation_(message_expectation) { |
| binding_.set_connection_error_handler(disconnect_run_loop_.QuitClosure()); |
| } |
| |
| ~PeerPidReceiver() override { |
| bool expected_message = |
| message_expectation_ != MessageExpectation::kNotExpected; |
| EXPECT_EQ(expected_message, received_message_); |
| } |
| |
| // mojom::Channel: |
| void SetPeerPid(int32_t pid) override { |
| peer_pid_ = pid; |
| on_peer_pid_set_.Run(); |
| } |
| |
| void Receive(IPC::MessageView message_view) override { |
| ASSERT_NE(MessageExpectation::kNotExpected, message_expectation_); |
| received_message_ = true; |
| |
| IPC::Message message(message_view.data(), message_view.size()); |
| bool expected_valid = |
| message_expectation_ == MessageExpectation::kExpectedValid; |
| EXPECT_EQ(expected_valid, message.IsValid()); |
| } |
| |
| void GetAssociatedInterface( |
| const std::string& name, |
| IPC::mojom::GenericInterfaceAssociatedRequest request) override {} |
| |
| int32_t peer_pid() const { return peer_pid_; } |
| |
| void RunUntilDisconnect() { disconnect_run_loop_.Run(); } |
| |
| private: |
| mojo::AssociatedBinding<IPC::mojom::Channel> binding_; |
| const base::Closure on_peer_pid_set_; |
| MessageExpectation message_expectation_; |
| int32_t peer_pid_ = -1; |
| bool received_message_ = false; |
| base::RunLoop disconnect_run_loop_; |
| |
| DISALLOW_COPY_AND_ASSIGN(PeerPidReceiver); |
| }; |
| |
| class IPCMojoBootstrapTest : public testing::Test { |
| protected: |
| mojo::core::test::MultiprocessTestHelper helper_; |
| }; |
| |
| TEST_F(IPCMojoBootstrapTest, Connect) { |
| base::MessageLoop message_loop; |
| Connection connection( |
| IPC::MojoBootstrap::Create( |
| helper_.StartChild("IPCMojoBootstrapTestClient"), |
| IPC::Channel::MODE_SERVER, base::ThreadTaskRunnerHandle::Get(), |
| base::ThreadTaskRunnerHandle::Get()), |
| kTestServerPid); |
| |
| IPC::mojom::ChannelAssociatedRequest receiver; |
| connection.TakeReceiver(&receiver); |
| |
| base::RunLoop run_loop; |
| PeerPidReceiver impl(std::move(receiver), run_loop.QuitClosure()); |
| run_loop.Run(); |
| |
| EXPECT_EQ(kTestClientPid, impl.peer_pid()); |
| |
| impl.RunUntilDisconnect(); |
| EXPECT_TRUE(helper_.WaitForChildTestShutdown()); |
| } |
| |
| // A long running process that connects to us. |
| MULTIPROCESS_TEST_MAIN_WITH_SETUP( |
| IPCMojoBootstrapTestClientTestChildMain, |
| ::mojo::core::test::MultiprocessTestHelper::ChildSetup) { |
| base::MessageLoop message_loop; |
| Connection connection( |
| IPC::MojoBootstrap::Create( |
| std::move(mojo::core::test::MultiprocessTestHelper::primordial_pipe), |
| IPC::Channel::MODE_CLIENT, base::ThreadTaskRunnerHandle::Get(), |
| base::ThreadTaskRunnerHandle::Get()), |
| kTestClientPid); |
| |
| IPC::mojom::ChannelAssociatedRequest receiver; |
| connection.TakeReceiver(&receiver); |
| |
| base::RunLoop run_loop; |
| PeerPidReceiver impl(std::move(receiver), run_loop.QuitClosure()); |
| run_loop.Run(); |
| |
| EXPECT_EQ(kTestServerPid, impl.peer_pid()); |
| |
| return 0; |
| } |
| |
| TEST_F(IPCMojoBootstrapTest, ReceiveEmptyMessage) { |
| base::MessageLoop message_loop; |
| Connection connection( |
| IPC::MojoBootstrap::Create( |
| helper_.StartChild("IPCMojoBootstrapTestEmptyMessage"), |
| IPC::Channel::MODE_SERVER, base::ThreadTaskRunnerHandle::Get(), |
| base::ThreadTaskRunnerHandle::Get()), |
| kTestServerPid); |
| |
| IPC::mojom::ChannelAssociatedRequest receiver; |
| connection.TakeReceiver(&receiver); |
| |
| base::RunLoop run_loop; |
| PeerPidReceiver impl(std::move(receiver), run_loop.QuitClosure(), |
| PeerPidReceiver::MessageExpectation::kExpectedInvalid); |
| run_loop.Run(); |
| |
| // Wait for the Channel to be disconnected so we can reasonably assert that |
| // the child's empty message must have been received before we pass the test. |
| impl.RunUntilDisconnect(); |
| |
| EXPECT_TRUE(helper_.WaitForChildTestShutdown()); |
| } |
| |
| // A long running process that connects to us. |
| MULTIPROCESS_TEST_MAIN_WITH_SETUP( |
| IPCMojoBootstrapTestEmptyMessageTestChildMain, |
| ::mojo::core::test::MultiprocessTestHelper::ChildSetup) { |
| base::MessageLoop message_loop; |
| Connection connection( |
| IPC::MojoBootstrap::Create( |
| std::move(mojo::core::test::MultiprocessTestHelper::primordial_pipe), |
| IPC::Channel::MODE_CLIENT, base::ThreadTaskRunnerHandle::Get(), |
| base::ThreadTaskRunnerHandle::Get()), |
| kTestClientPid); |
| |
| IPC::mojom::ChannelAssociatedRequest receiver; |
| connection.TakeReceiver(&receiver); |
| auto& sender = connection.GetSender(); |
| |
| uint8_t data = 0; |
| sender->Receive( |
| IPC::MessageView(mojo_base::BigBufferView(base::make_span(&data, 0)), |
| base::nullopt /* handles */)); |
| |
| base::RunLoop run_loop; |
| PeerPidReceiver impl(std::move(receiver), run_loop.QuitClosure()); |
| run_loop.Run(); |
| |
| return 0; |
| } |
| |
| } // namespace |