| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifdef UNSAFE_BUFFERS_BUILD |
| // TODO(crbug.com/351564777): Remove this and convert code to safer constructs. |
| #pragma allow_unsafe_buffers |
| #endif |
| |
| #include"ipc/ipc_message.h" |
| |
| #include<stddef.h> |
| #include<stdint.h> |
| #include<string.h> |
| |
| #include<limits> |
| #include<memory> |
| #include<utility> |
| |
| #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" |
| #include"ipc/ipc_message_start.h" |
| |
| #define IPC_MESSAGE_STARTTestMsgStart |
| |
| 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"); |
| std::u16string v3(u"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; |
| std::u16string 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,Value){ |
| auto expect_value_equals=[](const base::Value& input){ |
| IPC::Message msg(1,2, IPC::Message::PRIORITY_NORMAL); |
| IPC::WriteParam(&msg, input); |
| |
| base::Value output; |
| base::PickleIterator iter(msg); |
| EXPECT_TRUE(IPC::ReadParam(&msg,&iter,&output))<< input; |
| EXPECT_EQ(input, output); |
| }; |
| |
| expect_value_equals(base::Value("foo")); |
| expect_value_equals(base::Value(42)); |
| expect_value_equals(base::Value(0.07)); |
| expect_value_equals(base::Value(true)); |
| expect_value_equals(base::Value(base::Value::BlobStorage({'a','b','c'}))); |
| |
| { |
| base::Value::Dict dict; |
| dict.Set("key1",42); |
| dict.Set("key2","hi"); |
| expect_value_equals(base::Value(std::move(dict))); |
| } |
| { |
| base::Value::Listlist; |
| list.Append(42); |
| list.Append("hello"); |
| expect_value_equals(base::Value(std::move(list))); |
| } |
| |
| // Also test the corrupt case. |
| IPC::Message bad_msg(1,2, IPC::Message::PRIORITY_NORMAL); |
| bad_msg.WriteInt(99); |
| base::PickleIterator iter(bad_msg); |
| base::Value output; |
| EXPECT_FALSE(IPC::ReadParam(&bad_msg,&iter,&output)); |
| } |
| |
| TEST(IPCMessageTest,ValueDict){ |
| base::Value::Dict input; |
| input.Set("null", base::Value()); |
| input.Set("bool",true); |
| input.Set("int",42); |
| input.Set("int.with.dot",43); |
| |
| base::Value::Dict subdict; |
| subdict.Set("str","forty two"); |
| subdict.Set("bool",false); |
| |
| base::Value::List sublist; |
| sublist.Append(42.42); |
| sublist.Append("forty"); |
| sublist.Append("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::Value::Dict output; |
| base::PickleIterator iter(msg); |
| EXPECT_TRUE(IPC::ReadParam(&msg,&iter,&output)); |
| |
| EXPECT_EQ(input, 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<char> message_data(message.size()+7); |
| memcpy(message_data.data(), message.data(), message.size()); |
| |
| constchar* data_start= message_data.data(); |
| constchar* 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); |
| |
| constchar* data_start=reinterpret_cast<constchar*>(message.data()); |
| constchar* 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<uint32_t>(-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<size_t>::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<int32_t>::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{ |
| |
| classIPCMessageParameterTest:public testing::Test{ |
| public: |
| IPCMessageParameterTest(): extra_param_("extra_param"), called_(false){} |
| |
| boolOnMessageReceived(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; |
| } |
| |
| voidOnEmpty(std::string* extra_param){ |
| EXPECT_EQ(extra_param,&extra_param_); |
| called_=true; |
| } |
| |
| voidOnInt(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 BUILDFLAG(IS_ANDROID) |
| #define MAYBE_OneIntegerWithParam DISABLED_OneIntegerWithParam |
| #else |
| #define MAYBE_OneIntegerWithParamOneIntegerWithParam |
| #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 |