|  | /* | 
|  | * Copyright (C) 2018 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include "src/ipc/host_impl.h" | 
|  |  | 
|  | #include <memory> | 
|  |  | 
|  | #include "gmock/gmock.h" | 
|  | #include "gtest/gtest.h" | 
|  | #include "perfetto/base/file_utils.h" | 
|  | #include "perfetto/base/scoped_file.h" | 
|  | #include "perfetto/base/temp_file.h" | 
|  | #include "perfetto/base/unix_socket.h" | 
|  | #include "perfetto/base/utils.h" | 
|  | #include "perfetto/ipc/service.h" | 
|  | #include "perfetto/ipc/service_descriptor.h" | 
|  | #include "src/base/test/test_task_runner.h" | 
|  | #include "src/ipc/buffered_frame_deserializer.h" | 
|  | #include "src/ipc/test/test_socket.h" | 
|  |  | 
|  | #include "src/ipc/test/client_unittest_messages.pb.h" | 
|  | #include "src/ipc/wire_protocol.pb.h" | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace ipc { | 
|  | namespace { | 
|  |  | 
|  | using ::testing::_; | 
|  | using ::testing::Invoke; | 
|  | using ::testing::InvokeWithoutArgs; | 
|  | using ::testing::Return; | 
|  |  | 
|  | constexpr char kSockName[] = TEST_SOCK_NAME("host_impl_unittest.sock"); | 
|  |  | 
|  | // RequestProto and ReplyProto are defined in client_unittest_messages.proto. | 
|  |  | 
|  | class FakeService : public Service { | 
|  | public: | 
|  | MOCK_METHOD2(OnFakeMethod1, void(const RequestProto&, DeferredBase*)); | 
|  |  | 
|  | static void Invoker(Service* service, | 
|  | const ProtoMessage& req, | 
|  | DeferredBase deferred_reply) { | 
|  | static_cast<FakeService*>(service)->OnFakeMethod1( | 
|  | static_cast<const RequestProto&>(req), &deferred_reply); | 
|  | } | 
|  |  | 
|  | static std::unique_ptr<ProtoMessage> RequestDecoder( | 
|  | const std::string& proto) { | 
|  | std::unique_ptr<ProtoMessage> reply(new RequestProto()); | 
|  | EXPECT_TRUE(reply->ParseFromString(proto)); | 
|  | return reply; | 
|  | } | 
|  |  | 
|  | explicit FakeService(const char* service_name) { | 
|  | descriptor_.service_name = service_name; | 
|  | descriptor_.methods.push_back( | 
|  | {"FakeMethod1", &RequestDecoder, nullptr, &Invoker}); | 
|  | } | 
|  |  | 
|  | const ServiceDescriptor& GetDescriptor() override { return descriptor_; } | 
|  |  | 
|  | base::ScopedFile TakeReceivedFD() { return ipc::Service::TakeReceivedFD(); } | 
|  |  | 
|  | base::ScopedFile received_fd_; | 
|  | ServiceDescriptor descriptor_; | 
|  | }; | 
|  |  | 
|  | class FakeClient : public base::UnixSocket::EventListener { | 
|  | public: | 
|  | MOCK_METHOD0(OnConnect, void()); | 
|  | MOCK_METHOD0(OnDisconnect, void()); | 
|  | MOCK_METHOD1(OnServiceBound, void(const Frame::BindServiceReply&)); | 
|  | MOCK_METHOD1(OnInvokeMethodReply, void(const Frame::InvokeMethodReply&)); | 
|  | MOCK_METHOD1(OnFileDescriptorReceived, void(int)); | 
|  | MOCK_METHOD0(OnRequestError, void()); | 
|  |  | 
|  | explicit FakeClient(base::TaskRunner* task_runner) { | 
|  | sock_ = base::UnixSocket::Connect(kSockName, this, task_runner); | 
|  | } | 
|  |  | 
|  | ~FakeClient() override = default; | 
|  |  | 
|  | void BindService(const std::string& service_name) { | 
|  | Frame frame; | 
|  | uint64_t request_id = requests_.empty() ? 1 : requests_.rbegin()->first + 1; | 
|  | requests_.emplace(request_id, 0); | 
|  | frame.set_request_id(request_id); | 
|  | frame.mutable_msg_bind_service()->set_service_name(service_name); | 
|  | SendFrame(frame); | 
|  | } | 
|  |  | 
|  | void InvokeMethod(ServiceID service_id, | 
|  | MethodID method_id, | 
|  | const ProtoMessage& args, | 
|  | bool drop_reply = false, | 
|  | int fd = -1) { | 
|  | Frame frame; | 
|  | uint64_t request_id = requests_.empty() ? 1 : requests_.rbegin()->first + 1; | 
|  | requests_.emplace(request_id, 0); | 
|  | frame.set_request_id(request_id); | 
|  | frame.mutable_msg_invoke_method()->set_service_id(service_id); | 
|  | frame.mutable_msg_invoke_method()->set_method_id(method_id); | 
|  | frame.mutable_msg_invoke_method()->set_drop_reply(drop_reply); | 
|  | frame.mutable_msg_invoke_method()->set_args_proto(args.SerializeAsString()); | 
|  | SendFrame(frame, fd); | 
|  | } | 
|  |  | 
|  | // base::UnixSocket::EventListener implementation. | 
|  | void OnConnect(base::UnixSocket*, bool success) override { | 
|  | ASSERT_TRUE(success); | 
|  | OnConnect(); | 
|  | } | 
|  |  | 
|  | void OnDisconnect(base::UnixSocket*) override { OnDisconnect(); } | 
|  |  | 
|  | void OnDataAvailable(base::UnixSocket* sock) override { | 
|  | ASSERT_EQ(sock_.get(), sock); | 
|  | auto buf = frame_deserializer_.BeginReceive(); | 
|  | base::ScopedFile fd; | 
|  | size_t rsize = sock->Receive(buf.data, buf.size, &fd); | 
|  | ASSERT_TRUE(frame_deserializer_.EndReceive(rsize)); | 
|  | if (fd) | 
|  | OnFileDescriptorReceived(*fd); | 
|  | while (std::unique_ptr<Frame> frame = frame_deserializer_.PopNextFrame()) { | 
|  | ASSERT_EQ(1u, requests_.count(frame->request_id())); | 
|  | EXPECT_EQ(0, requests_[frame->request_id()]++); | 
|  | if (frame->msg_case() == Frame::kMsgBindServiceReply) { | 
|  | if (frame->msg_bind_service_reply().success()) | 
|  | last_bound_service_id_ = frame->msg_bind_service_reply().service_id(); | 
|  | return OnServiceBound(frame->msg_bind_service_reply()); | 
|  | } | 
|  | if (frame->msg_case() == Frame::kMsgInvokeMethodReply) | 
|  | return OnInvokeMethodReply(frame->msg_invoke_method_reply()); | 
|  | if (frame->msg_case() == Frame::kMsgRequestError) | 
|  | return OnRequestError(); | 
|  | FAIL() << "Unexpected frame received from host " << frame->msg_case(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void SendFrame(const Frame& frame, int fd = -1) { | 
|  | std::string buf = BufferedFrameDeserializer::Serialize(frame); | 
|  | ASSERT_TRUE(sock_->Send(buf.data(), buf.size(), fd, | 
|  | base::UnixSocket::BlockingMode::kBlocking)); | 
|  | } | 
|  |  | 
|  | BufferedFrameDeserializer frame_deserializer_; | 
|  | std::unique_ptr<base::UnixSocket> sock_; | 
|  | std::map<uint64_t /* request_id */, int /* num_replies_received */> requests_; | 
|  | ServiceID last_bound_service_id_; | 
|  | }; | 
|  |  | 
|  | class HostImplTest : public ::testing::Test { | 
|  | public: | 
|  | void SetUp() override { | 
|  | DESTROY_TEST_SOCK(kSockName); | 
|  | task_runner_.reset(new base::TestTaskRunner()); | 
|  | Host* host = Host::CreateInstance(kSockName, task_runner_.get()).release(); | 
|  | ASSERT_NE(nullptr, host); | 
|  | host_.reset(static_cast<HostImpl*>(host)); | 
|  | cli_.reset(new FakeClient(task_runner_.get())); | 
|  | auto on_connect = task_runner_->CreateCheckpoint("on_connect"); | 
|  | EXPECT_CALL(*cli_, OnConnect()).WillOnce(Invoke(on_connect)); | 
|  | task_runner_->RunUntilCheckpoint("on_connect"); | 
|  | } | 
|  |  | 
|  | void TearDown() override { | 
|  | task_runner_->RunUntilIdle(); | 
|  | cli_.reset(); | 
|  | host_.reset(); | 
|  | task_runner_->RunUntilIdle(); | 
|  | task_runner_.reset(); | 
|  | DESTROY_TEST_SOCK(kSockName); | 
|  | } | 
|  |  | 
|  | // ::testing::StrictMock<MockEventListener> proxy_events_; | 
|  | std::unique_ptr<base::TestTaskRunner> task_runner_; | 
|  | std::unique_ptr<HostImpl> host_; | 
|  | std::unique_ptr<FakeClient> cli_; | 
|  | }; | 
|  |  | 
|  | TEST_F(HostImplTest, BindService) { | 
|  | // First bind the service when it doesn't exists yet and check that the | 
|  | // BindService() request fails. | 
|  | cli_->BindService("FakeService");  // FakeService does not exist yet. | 
|  | auto on_bind_failure = task_runner_->CreateCheckpoint("on_bind_failure"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)) | 
|  | .WillOnce(Invoke([on_bind_failure](const Frame::BindServiceReply& reply) { | 
|  | ASSERT_FALSE(reply.success()); | 
|  | on_bind_failure(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_bind_failure"); | 
|  |  | 
|  | // Now expose the service and bind it. | 
|  | ASSERT_TRUE(host_->ExposeService( | 
|  | std::unique_ptr<Service>(new FakeService("FakeService")))); | 
|  | auto on_bind_success = task_runner_->CreateCheckpoint("on_bind_success"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)) | 
|  | .WillOnce(Invoke([on_bind_success](const Frame::BindServiceReply& reply) { | 
|  | ASSERT_TRUE(reply.success()); | 
|  | on_bind_success(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_bind_success"); | 
|  | } | 
|  |  | 
|  | TEST_F(HostImplTest, InvokeNonExistingMethod) { | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | auto on_invoke_failure = task_runner_->CreateCheckpoint("on_invoke_failure"); | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 42, RequestProto()); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)) | 
|  | .WillOnce( | 
|  | Invoke([on_invoke_failure](const Frame::InvokeMethodReply& reply) { | 
|  | ASSERT_FALSE(reply.success()); | 
|  | ASSERT_FALSE(reply.has_more()); | 
|  | on_invoke_failure(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_invoke_failure"); | 
|  | } | 
|  |  | 
|  | TEST_F(HostImplTest, InvokeMethod) { | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | RequestProto req_args; | 
|  | req_args.set_data("foo"); | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, req_args); | 
|  | auto on_reply_sent = task_runner_->CreateCheckpoint("on_reply_sent"); | 
|  | EXPECT_CALL(*fake_service, OnFakeMethod1(_, _)) | 
|  | .WillOnce( | 
|  | Invoke([on_reply_sent](const RequestProto& req, DeferredBase* reply) { | 
|  | ASSERT_EQ("foo", req.data()); | 
|  | std::unique_ptr<ReplyProto> reply_args(new ReplyProto()); | 
|  | reply_args->set_data("bar"); | 
|  | reply->Resolve(AsyncResult<ProtoMessage>( | 
|  | std::unique_ptr<ProtoMessage>(reply_args.release()))); | 
|  | on_reply_sent(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_reply_sent"); | 
|  |  | 
|  | auto on_reply_received = task_runner_->CreateCheckpoint("on_reply_received"); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)) | 
|  | .WillOnce( | 
|  | Invoke([on_reply_received](const Frame::InvokeMethodReply& reply) { | 
|  | ASSERT_TRUE(reply.success()); | 
|  | ASSERT_FALSE(reply.has_more()); | 
|  | ReplyProto reply_args; | 
|  | reply_args.ParseFromString(reply.reply_proto()); | 
|  | ASSERT_EQ("bar", reply_args.data()); | 
|  | on_reply_received(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_reply_received"); | 
|  | } | 
|  |  | 
|  | TEST_F(HostImplTest, InvokeMethodDropReply) { | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | // OnFakeMethod1 will: | 
|  | // - Do nothing on the 1st call, when |drop_reply| == true. | 
|  | // - Reply on the the 2nd call, when |drop_reply| == false. | 
|  | EXPECT_CALL(*fake_service, OnFakeMethod1(_, _)) | 
|  | .Times(2) | 
|  | .WillRepeatedly(Invoke([](const RequestProto& req, DeferredBase* reply) { | 
|  | if (req.data() == "drop_reply") | 
|  | return; | 
|  | std::unique_ptr<ReplyProto> reply_args(new ReplyProto()); | 
|  | reply_args->set_data("the_reply"); | 
|  | reply->Resolve(AsyncResult<ProtoMessage>( | 
|  | std::unique_ptr<ProtoMessage>(reply_args.release()))); | 
|  | })); | 
|  |  | 
|  | auto on_reply_received = task_runner_->CreateCheckpoint("on_reply_received"); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)) | 
|  | .WillOnce( | 
|  | Invoke([on_reply_received](const Frame::InvokeMethodReply& reply) { | 
|  | ASSERT_TRUE(reply.success()); | 
|  | ReplyProto reply_args; | 
|  | reply_args.ParseFromString(reply.reply_proto()); | 
|  | ASSERT_EQ("the_reply", reply_args.data()); | 
|  | on_reply_received(); | 
|  | })); | 
|  |  | 
|  | // Invoke the method first with |drop_reply|=true, then |drop_reply|=false. | 
|  | RequestProto rp; | 
|  | rp.set_data("drop_reply"); | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, rp, true /*drop_reply*/); | 
|  | rp.set_data("do_reply"); | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, rp, false /*drop_reply*/); | 
|  |  | 
|  | task_runner_->RunUntilCheckpoint("on_reply_received"); | 
|  | } | 
|  |  | 
|  | TEST_F(HostImplTest, SendFileDescriptor) { | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | static constexpr char kFileContent[] = "shared file"; | 
|  | RequestProto req_args; | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, req_args); | 
|  | auto on_reply_sent = task_runner_->CreateCheckpoint("on_reply_sent"); | 
|  | base::TempFile tx_file = base::TempFile::CreateUnlinked(); | 
|  | ASSERT_EQ(base::WriteAll(tx_file.fd(), kFileContent, sizeof(kFileContent)), | 
|  | sizeof(kFileContent)); | 
|  | EXPECT_CALL(*fake_service, OnFakeMethod1(_, _)) | 
|  | .WillOnce(Invoke([on_reply_sent, &tx_file](const RequestProto&, | 
|  | DeferredBase* reply) { | 
|  | std::unique_ptr<ReplyProto> reply_args(new ReplyProto()); | 
|  | auto async_res = AsyncResult<ProtoMessage>( | 
|  | std::unique_ptr<ProtoMessage>(reply_args.release())); | 
|  | async_res.set_fd(tx_file.fd()); | 
|  | reply->Resolve(std::move(async_res)); | 
|  | on_reply_sent(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_reply_sent"); | 
|  | tx_file.ReleaseFD(); | 
|  |  | 
|  | auto on_fd_received = task_runner_->CreateCheckpoint("on_fd_received"); | 
|  | EXPECT_CALL(*cli_, OnFileDescriptorReceived(_)) | 
|  | .WillOnce(Invoke([on_fd_received](int fd) { | 
|  | char buf[sizeof(kFileContent)] = {}; | 
|  | ASSERT_EQ(0, lseek(fd, 0, SEEK_SET)); | 
|  | ASSERT_EQ(static_cast<int32_t>(sizeof(buf)), | 
|  | PERFETTO_EINTR(read(fd, buf, sizeof(buf)))); | 
|  | ASSERT_STREQ(kFileContent, buf); | 
|  | on_fd_received(); | 
|  | })); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)); | 
|  | task_runner_->RunUntilCheckpoint("on_fd_received"); | 
|  | } | 
|  |  | 
|  | TEST_F(HostImplTest, ReceiveFileDescriptor) { | 
|  | auto received = task_runner_->CreateCheckpoint("received"); | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | static constexpr char kFileContent[] = "shared file"; | 
|  | RequestProto req_args; | 
|  | base::TempFile tx_file = base::TempFile::CreateUnlinked(); | 
|  | ASSERT_EQ(base::WriteAll(tx_file.fd(), kFileContent, sizeof(kFileContent)), | 
|  | sizeof(kFileContent)); | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, req_args, false, | 
|  | tx_file.fd()); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)); | 
|  | base::ScopedFile rx_fd; | 
|  | EXPECT_CALL(*fake_service, OnFakeMethod1(_, _)) | 
|  | .WillOnce(Invoke([received, &fake_service, &rx_fd](const RequestProto&, | 
|  | DeferredBase*) { | 
|  | rx_fd = fake_service->TakeReceivedFD(); | 
|  | received(); | 
|  | })); | 
|  |  | 
|  | task_runner_->RunUntilCheckpoint("received"); | 
|  |  | 
|  | ASSERT_TRUE(rx_fd); | 
|  | char buf[sizeof(kFileContent)] = {}; | 
|  | ASSERT_EQ(0, lseek(*rx_fd, 0, SEEK_SET)); | 
|  | ASSERT_EQ(static_cast<int32_t>(sizeof(buf)), | 
|  | PERFETTO_EINTR(read(*rx_fd, buf, sizeof(buf)))); | 
|  | ASSERT_STREQ(kFileContent, buf); | 
|  | } | 
|  |  | 
|  | // Invoke a method and immediately after disconnect the client. | 
|  | TEST_F(HostImplTest, OnClientDisconnect) { | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | RequestProto req_args; | 
|  | req_args.set_data("foo"); | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, req_args); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)).Times(0); | 
|  | cli_.reset();  // Disconnect the client. | 
|  | auto on_host_method = task_runner_->CreateCheckpoint("on_host_method"); | 
|  | EXPECT_CALL(*fake_service, OnFakeMethod1(_, _)) | 
|  | .WillOnce( | 
|  | Invoke([on_host_method](const RequestProto& req, DeferredBase*) { | 
|  | ASSERT_EQ("foo", req.data()); | 
|  | on_host_method(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_host_method"); | 
|  | } | 
|  |  | 
|  | // Like InvokeMethod, but instead of resolving the Deferred reply within the | 
|  | // call stack, std::move()-s it outside an replies | 
|  | TEST_F(HostImplTest, MoveReplyObjectAndReplyAsynchronously) { | 
|  | FakeService* fake_service = new FakeService("FakeService"); | 
|  | ASSERT_TRUE(host_->ExposeService(std::unique_ptr<Service>(fake_service))); | 
|  | auto on_bind = task_runner_->CreateCheckpoint("on_bind"); | 
|  | cli_->BindService("FakeService"); | 
|  | EXPECT_CALL(*cli_, OnServiceBound(_)).WillOnce(InvokeWithoutArgs(on_bind)); | 
|  | task_runner_->RunUntilCheckpoint("on_bind"); | 
|  |  | 
|  | // Invokes the remote method and waits that the FakeService sees it. The reply | 
|  | // is not resolved but just moved into |moved_reply|. | 
|  | RequestProto req_args; | 
|  | cli_->InvokeMethod(cli_->last_bound_service_id_, 1, req_args); | 
|  | auto on_invoke = task_runner_->CreateCheckpoint("on_invoke"); | 
|  | DeferredBase moved_reply; | 
|  | EXPECT_CALL(*fake_service, OnFakeMethod1(_, _)) | 
|  | .WillOnce(Invoke( | 
|  | [on_invoke, &moved_reply](const RequestProto&, DeferredBase* reply) { | 
|  | moved_reply = std::move(*reply); | 
|  | on_invoke(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_invoke"); | 
|  |  | 
|  | // Check that the FakeClient doesn't see any reply yet. | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)).Times(0); | 
|  | task_runner_->RunUntilIdle(); | 
|  | ASSERT_TRUE(::testing::Mock::VerifyAndClearExpectations(cli_.get())); | 
|  |  | 
|  | // Resolve the reply asynchronously in a deferred task. | 
|  | task_runner_->PostTask([&moved_reply] { | 
|  | std::unique_ptr<ReplyProto> reply_args(new ReplyProto()); | 
|  | reply_args->set_data("bar"); | 
|  | moved_reply.Resolve(AsyncResult<ProtoMessage>( | 
|  | std::unique_ptr<ProtoMessage>(reply_args.release()))); | 
|  | }); | 
|  |  | 
|  | auto on_reply_received = task_runner_->CreateCheckpoint("on_reply_received"); | 
|  | EXPECT_CALL(*cli_, OnInvokeMethodReply(_)) | 
|  | .WillOnce( | 
|  | Invoke([on_reply_received](const Frame::InvokeMethodReply& reply) { | 
|  | ASSERT_TRUE(reply.success()); | 
|  | ASSERT_FALSE(reply.has_more()); | 
|  | ReplyProto reply_args; | 
|  | reply_args.ParseFromString(reply.reply_proto()); | 
|  | ASSERT_EQ("bar", reply_args.data()); | 
|  | on_reply_received(); | 
|  | })); | 
|  | task_runner_->RunUntilCheckpoint("on_reply_received"); | 
|  | } | 
|  |  | 
|  | // TODO(primiano): add the tests below in next CLs. | 
|  | // TEST(HostImplTest, ManyClients) {} | 
|  | // TEST(HostImplTest, OverlappingRequstsOutOfOrder) {} | 
|  | // TEST(HostImplTest, StreamingRequest) {} | 
|  | // TEST(HostImplTest, ManyDropReplyRequestsDontLeakMemory) {} | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace ipc | 
|  | }  // namespace perfetto |