// 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 "ipc/ipc_message.h" #include #include #include #include #include #include #include "base/memory/ptr_util.h" #include "base/strings/utf_string_conversions.h" #include "base/values.h" #include "build/build_config.h" #include "ipc/ipc_message_utils.h" #include "testing/gtest/include/gtest/gtest.h" // IPC messages for testing ---------------------------------------------------- #define IPC_MESSAGE_IMPL #include "ipc/ipc_message_macros.h" #define IPC_MESSAGE_START TestMsgStart IPC_MESSAGE_CONTROL0(TestMsgClassEmpty) IPC_MESSAGE_CONTROL1(TestMsgClassI, int) IPC_SYNC_MESSAGE_CONTROL1_1(TestMsgClassIS, int, std::string) namespace IPC { TEST(IPCMessageTest, BasicMessageTest) { int v1 = 10; std::string v2("foobar"); base::string16 v3(base::ASCIIToUTF16("hello world")); IPC::Message m(0, 1, IPC::Message::PRIORITY_NORMAL); m.WriteInt(v1); m.WriteString(v2); m.WriteString16(v3); base::PickleIterator iter(m); int vi; std::string vs; base::string16 vs16; EXPECT_TRUE(iter.ReadInt(&vi)); EXPECT_EQ(v1, vi); EXPECT_TRUE(iter.ReadString(&vs)); EXPECT_EQ(v2, vs); EXPECT_TRUE(iter.ReadString16(&vs16)); EXPECT_EQ(v3, vs16); // should fail EXPECT_FALSE(iter.ReadInt(&vi)); EXPECT_FALSE(iter.ReadString(&vs)); EXPECT_FALSE(iter.ReadString16(&vs16)); } TEST(IPCMessageTest, ListValue) { base::ListValue input; input.AppendDouble(42.42); input.AppendString("forty"); input.Append(std::make_unique()); IPC::Message msg(1, 2, IPC::Message::PRIORITY_NORMAL); IPC::WriteParam(&msg, input); base::ListValue output; base::PickleIterator iter(msg); EXPECT_TRUE(IPC::ReadParam(&msg, &iter, &output)); EXPECT_TRUE(input.Equals(&output)); // Also test the corrupt case. IPC::Message bad_msg(1, 2, IPC::Message::PRIORITY_NORMAL); bad_msg.WriteInt(99); iter = base::PickleIterator(bad_msg); EXPECT_FALSE(IPC::ReadParam(&bad_msg, &iter, &output)); } TEST(IPCMessageTest, DictionaryValue) { base::DictionaryValue input; input.Set("null", std::make_unique()); input.SetBoolean("bool", true); input.SetInteger("int", 42); input.SetKey("int.with.dot", base::Value(43)); auto subdict = std::make_unique(); subdict->SetString("str", "forty two"); subdict->SetBoolean("bool", false); auto sublist = std::make_unique(); sublist->AppendDouble(42.42); sublist->AppendString("forty"); sublist->AppendString("two"); subdict->Set("list", std::move(sublist)); input.Set("dict", std::move(subdict)); IPC::Message msg(1, 2, IPC::Message::PRIORITY_NORMAL); IPC::WriteParam(&msg, input); base::DictionaryValue output; base::PickleIterator iter(msg); EXPECT_TRUE(IPC::ReadParam(&msg, &iter, &output)); EXPECT_TRUE(input.Equals(&output)); // Also test the corrupt case. IPC::Message bad_msg(1, 2, IPC::Message::PRIORITY_NORMAL); bad_msg.WriteInt(99); iter = base::PickleIterator(bad_msg); EXPECT_FALSE(IPC::ReadParam(&bad_msg, &iter, &output)); } TEST(IPCMessageTest, FindNext) { IPC::Message message; message.WriteString("Goooooooogle"); message.WriteInt(111); std::vector message_data(message.size() + 7); memcpy(message_data.data(), message.data(), message.size()); const char* data_start = message_data.data(); const char* data_end = data_start + message.size(); IPC::Message::NextMessageInfo next; // Data range contains the entire message plus some extra bytes IPC::Message::FindNext(data_start, data_end + 1, &next); EXPECT_TRUE(next.message_found); EXPECT_EQ(next.message_size, message.size()); EXPECT_EQ(next.pickle_end, data_end); EXPECT_EQ(next.message_end, data_end); // Data range exactly contains the entire message IPC::Message::FindNext(data_start, data_end, &next); EXPECT_TRUE(next.message_found); EXPECT_EQ(next.message_size, message.size()); EXPECT_EQ(next.pickle_end, data_end); EXPECT_EQ(next.message_end, data_end); // Data range doesn't contain the entire message // (but contains the message header) IPC::Message::FindNext(data_start, data_end - 1, &next); EXPECT_FALSE(next.message_found); EXPECT_EQ(next.message_size, message.size()); // Data range doesn't contain the message header // (but contains the pickle header) IPC::Message::FindNext(data_start, data_start + sizeof(IPC::Message::Header) - 1, &next); EXPECT_FALSE(next.message_found); EXPECT_EQ(next.message_size, 0u); // Data range doesn't contain the pickle header IPC::Message::FindNext(data_start, data_start + sizeof(base::Pickle::Header) - 1, &next); EXPECT_FALSE(next.message_found); EXPECT_EQ(next.message_size, 0u); } TEST(IPCMessageTest, FindNextOverflow) { IPC::Message message; message.WriteString("Data"); message.WriteInt(777); const char* data_start = reinterpret_cast(message.data()); const char* data_end = data_start + message.size(); IPC::Message::NextMessageInfo next; // Payload size is negative (defeats 'start + size > end' check) message.header()->payload_size = static_cast(-1); IPC::Message::FindNext(data_start, data_end, &next); EXPECT_FALSE(next.message_found); if (sizeof(size_t) > sizeof(uint32_t)) { // No overflow, just insane message size EXPECT_EQ(next.message_size, message.header()->payload_size + sizeof(IPC::Message::Header)); } else { // Actual overflow, reported as max size_t EXPECT_EQ(next.message_size, std::numeric_limits::max()); } // Payload size is max positive integer (defeats size < 0 check, while // still potentially causing overflow down the road). message.header()->payload_size = std::numeric_limits::max(); IPC::Message::FindNext(data_start, data_end, &next); EXPECT_FALSE(next.message_found); EXPECT_EQ(next.message_size, message.header()->payload_size + sizeof(IPC::Message::Header)); } namespace { class IPCMessageParameterTest : public testing::Test { public: IPCMessageParameterTest() : extra_param_("extra_param"), called_(false) {} bool OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP_WITH_PARAM(IPCMessageParameterTest, message, &extra_param_) IPC_MESSAGE_HANDLER(TestMsgClassEmpty, OnEmpty) IPC_MESSAGE_HANDLER(TestMsgClassI, OnInt) //IPC_MESSAGE_HANDLER(TestMsgClassIS, OnSync) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void OnEmpty(std::string* extra_param) { EXPECT_EQ(extra_param, &extra_param_); called_ = true; } void OnInt(std::string* extra_param, int foo) { EXPECT_EQ(extra_param, &extra_param_); EXPECT_EQ(foo, 42); called_ = true; } /* TODO: handle sync IPCs void OnSync(std::string* extra_param, int foo, std::string* out) { EXPECT_EQ(extra_param, &extra_param_); EXPECT_EQ(foo, 42); called_ = true; *out = std::string("out"); } bool Send(IPC::Message* reply) { delete reply; return true; }*/ std::string extra_param_; bool called_; }; } // namespace TEST_F(IPCMessageParameterTest, EmptyDispatcherWithParam) { TestMsgClassEmpty message; EXPECT_TRUE(OnMessageReceived(message)); EXPECT_TRUE(called_); } #if defined(OS_ANDROID) #define MAYBE_OneIntegerWithParam DISABLED_OneIntegerWithParam #else #define MAYBE_OneIntegerWithParam OneIntegerWithParam #endif TEST_F(IPCMessageParameterTest, MAYBE_OneIntegerWithParam) { TestMsgClassI message(42); EXPECT_TRUE(OnMessageReceived(message)); EXPECT_TRUE(called_); } /* TODO: handle sync IPCs TEST_F(IPCMessageParameterTest, Sync) { std::string output; TestMsgClassIS message(42, &output); EXPECT_TRUE(OnMessageReceived(message)); EXPECT_TRUE(called_); EXPECT_EQ(output, std::string("out")); }*/ } // namespace IPC