// Copyright (c) 2012 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 "build/build_config.h" #if defined(OS_WIN) #include #endif #include #include "base/message_loop.h" #include "base/pickle.h" #include "base/threading/thread.h" #include "ipc/ipc_message.h" #include "ipc/ipc_test_base.h" namespace { const size_t kLongMessageStringNumBytes = 50000; static void Send(IPC::Sender* sender, const char* text) { static int message_index = 0; IPC::Message* message = new IPC::Message(0, 2, IPC::Message::PRIORITY_NORMAL); message->WriteInt(message_index++); message->WriteString(std::string(text)); // Make sure we can handle large messages. char junk[kLongMessageStringNumBytes]; memset(junk, 'a', sizeof(junk)-1); junk[sizeof(junk)-1] = 0; message->WriteString(std::string(junk)); // DEBUG: printf("[%u] sending message [%s]\n", GetCurrentProcessId(), text); sender->Send(message); } // A generic listener that expects messages of a certain type (see // OnMessageReceived()), and either sends a generic response or quits after the // 50th message (or on channel error). class GenericChannelListener : public IPC::Listener { public: GenericChannelListener() : sender_(NULL), messages_left_(50) {} virtual ~GenericChannelListener() {} virtual bool OnMessageReceived(const IPC::Message& message) OVERRIDE { PickleIterator iter(message); int ignored; EXPECT_TRUE(iter.ReadInt(&ignored)); std::string data; EXPECT_TRUE(iter.ReadString(&data)); std::string big_string; EXPECT_TRUE(iter.ReadString(&big_string)); EXPECT_EQ(kLongMessageStringNumBytes - 1, big_string.length()); SendNextMessage(); return true; } virtual void OnChannelError() OVERRIDE { // There is a race when closing the channel so the last message may be lost. EXPECT_LE(messages_left_, 1); base::MessageLoop::current()->Quit(); } void Init(IPC::Sender* s) { sender_ = s; } protected: void SendNextMessage() { if (--messages_left_ <= 0) base::MessageLoop::current()->Quit(); else Send(sender_, "Foo"); } private: IPC::Sender* sender_; int messages_left_; }; class IPCChannelTest : public IPCTestBase { }; // TODO(viettrungluu): Move to a separate IPCMessageTest. TEST_F(IPCChannelTest, BasicMessageTest) { int v1 = 10; std::string v2("foobar"); std::wstring v3(L"hello world"); IPC::Message m(0, 1, IPC::Message::PRIORITY_NORMAL); EXPECT_TRUE(m.WriteInt(v1)); EXPECT_TRUE(m.WriteString(v2)); EXPECT_TRUE(m.WriteWString(v3)); PickleIterator iter(m); int vi; std::string vs; std::wstring vw; EXPECT_TRUE(m.ReadInt(&iter, &vi)); EXPECT_EQ(v1, vi); EXPECT_TRUE(m.ReadString(&iter, &vs)); EXPECT_EQ(v2, vs); EXPECT_TRUE(m.ReadWString(&iter, &vw)); EXPECT_EQ(v3, vw); // should fail EXPECT_FALSE(m.ReadInt(&iter, &vi)); EXPECT_FALSE(m.ReadString(&iter, &vs)); EXPECT_FALSE(m.ReadWString(&iter, &vw)); } TEST_F(IPCChannelTest, ChannelTest) { Init("GenericClient"); // Set up IPC channel and start client. GenericChannelListener listener; CreateChannel(&listener); listener.Init(sender()); ASSERT_TRUE(ConnectChannel()); ASSERT_TRUE(StartClient()); Send(sender(), "hello from parent"); // Run message loop. base::MessageLoop::current()->Run(); // Close the channel so the client's OnChannelError() gets fired. channel()->Close(); EXPECT_TRUE(WaitForClientShutdown()); DestroyChannel(); } // TODO(viettrungluu): Move to a separate IPCChannelWinTest. #if defined(OS_WIN) TEST_F(IPCChannelTest, ChannelTestExistingPipe) { Init("GenericClient"); // Create pipe manually using the standard Chromium name and set up IPC // channel. GenericChannelListener listener; std::string name("\\\\.\\pipe\\chrome."); name.append(GetChannelName("GenericClient")); HANDLE pipe = CreateNamedPipeA(name.c_str(), PIPE_ACCESS_DUPLEX | FILE_FLAG_OVERLAPPED | FILE_FLAG_FIRST_PIPE_INSTANCE, PIPE_TYPE_BYTE | PIPE_READMODE_BYTE, 1, 4096, 4096, 5000, NULL); CreateChannelFromChannelHandle(IPC::ChannelHandle(pipe), &listener); CloseHandle(pipe); // The channel duplicates the handle. listener.Init(sender()); // Connect to channel and start client. ASSERT_TRUE(ConnectChannel()); ASSERT_TRUE(StartClient()); Send(sender(), "hello from parent"); // Run message loop. base::MessageLoop::current()->Run(); // Close the channel so the client's OnChannelError() gets fired. channel()->Close(); EXPECT_TRUE(WaitForClientShutdown()); DestroyChannel(); } #endif // defined (OS_WIN) TEST_F(IPCChannelTest, ChannelProxyTest) { Init("GenericClient"); base::Thread thread("ChannelProxyTestServer"); base::Thread::Options options; options.message_loop_type = base::MessageLoop::TYPE_IO; thread.StartWithOptions(options); // Set up IPC channel proxy. GenericChannelListener listener; CreateChannelProxy(&listener, thread.message_loop_proxy().get()); listener.Init(sender()); ASSERT_TRUE(StartClient()); Send(sender(), "hello from parent"); // Run message loop. base::MessageLoop::current()->Run(); EXPECT_TRUE(WaitForClientShutdown()); // Destroy the channel proxy before shutting down the thread. DestroyChannelProxy(); thread.Stop(); } class ChannelListenerWithOnConnectedSend : public GenericChannelListener { public: ChannelListenerWithOnConnectedSend() {} virtual ~ChannelListenerWithOnConnectedSend() {} virtual void OnChannelConnected(int32 peer_pid) OVERRIDE { SendNextMessage(); } }; #if defined(OS_WIN) // Acting flakey in Windows. http://crbug.com/129595 #define MAYBE_SendMessageInChannelConnected DISABLED_SendMessageInChannelConnected #else #define MAYBE_SendMessageInChannelConnected SendMessageInChannelConnected #endif // This tests the case of a listener sending back an event in its // OnChannelConnected handler. TEST_F(IPCChannelTest, MAYBE_SendMessageInChannelConnected) { Init("GenericClient"); // Set up IPC channel and start client. ChannelListenerWithOnConnectedSend listener; CreateChannel(&listener); listener.Init(sender()); ASSERT_TRUE(ConnectChannel()); ASSERT_TRUE(StartClient()); Send(sender(), "hello from parent"); // Run message loop. base::MessageLoop::current()->Run(); // Close the channel so the client's OnChannelError() gets fired. channel()->Close(); EXPECT_TRUE(WaitForClientShutdown()); DestroyChannel(); } MULTIPROCESS_IPC_TEST_CLIENT_MAIN(GenericClient) { base::MessageLoopForIO main_message_loop; GenericChannelListener listener; // Set up IPC channel. IPC::Channel channel(IPCTestBase::GetChannelName("GenericClient"), IPC::Channel::MODE_CLIENT, &listener); CHECK(channel.Connect()); listener.Init(&channel); Send(&channel, "hello from child"); base::MessageLoop::current()->Run(); return 0; } } // namespace