// Copyright 2015 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" #include <stddef.h> #include <stdint.h> #include <limits> #include <memory> #include <set> #include "base/run_loop.h" #include "ipc/ipc_channel_reader.h" #include "testing/gtest/include/gtest/gtest.h" namespace IPC { namespace internal { namespace { class MockChannelReader : public ChannelReader { public: MockChannelReader() : ChannelReader(nullptr), last_dispatched_message_(nullptr) {} ReadState ReadData(char* buffer, int buffer_len, int* bytes_read) override { if (data_.empty()) return READ_PENDING; size_t read_len = std::min(static_cast<size_t>(buffer_len), data_.size()); memcpy(buffer, data_.data(), read_len); *bytes_read = static_cast<int>(read_len); data_.erase(0, read_len); return READ_SUCCEEDED; } bool ShouldDispatchInputMessage(Message* msg) override { return true; } bool GetAttachments(Message* msg) override { return true; } bool DidEmptyInputBuffers() override { return true; } void HandleInternalMessage(const Message& msg) override {} void DispatchMessage(Message* m) override { last_dispatched_message_ = m; } Message* get_last_dispatched_message() { return last_dispatched_message_; } void AppendData(const void* data, size_t size) { data_.append(static_cast<const char*>(data), size); } void AppendMessageData(const Message& message) { AppendData(message.data(), message.size()); } private: Message* last_dispatched_message_; std::string data_; }; class ExposedMessage: public Message { public: using Message::Header; using Message::header; }; // Payload that makes messages large const size_t LargePayloadSize = Channel::kMaximumReadBufferSize * 3 / 2; } // namespace // We can determine message size from its header (and hence resize the buffer) // only when attachment broker is not used, see IPC::Message::FindNext(). TEST(ChannelReaderTest, ResizeOverflowBuffer) { MockChannelReader reader; ExposedMessage::Header header = {}; header.payload_size = 128 * 1024; EXPECT_LT(reader.input_overflow_buf_.capacity(), header.payload_size); EXPECT_TRUE(reader.TranslateInputData( reinterpret_cast<const char*>(&header), sizeof(header))); // Once message header is available we resize overflow buffer to // fit the entire message. EXPECT_GE(reader.input_overflow_buf_.capacity(), header.payload_size); } TEST(ChannelReaderTest, InvalidMessageSize) { MockChannelReader reader; ExposedMessage::Header header = {}; size_t capacity_before = reader.input_overflow_buf_.capacity(); // Message is slightly larger than maximum allowed size header.payload_size = Channel::kMaximumMessageSize + 1; EXPECT_FALSE(reader.TranslateInputData( reinterpret_cast<const char*>(&header), sizeof(header))); EXPECT_LE(reader.input_overflow_buf_.capacity(), capacity_before); // Payload size is negative, overflow is detected by Pickle::PeekNext() header.payload_size = static_cast<uint32_t>(-1); EXPECT_FALSE(reader.TranslateInputData( reinterpret_cast<const char*>(&header), sizeof(header))); EXPECT_LE(reader.input_overflow_buf_.capacity(), capacity_before); // Payload size is maximum int32_t value header.payload_size = std::numeric_limits<int32_t>::max(); EXPECT_FALSE(reader.TranslateInputData( reinterpret_cast<const char*>(&header), sizeof(header))); EXPECT_LE(reader.input_overflow_buf_.capacity(), capacity_before); } TEST(ChannelReaderTest, TrimBuffer) { // ChannelReader uses std::string as a buffer, and calls reserve() // to trim it to kMaximumReadBufferSize. However, an implementation // is free to actually reserve a larger amount. size_t trimmed_buffer_size; { std::string buf; buf.reserve(Channel::kMaximumReadBufferSize); trimmed_buffer_size = buf.capacity(); } // Buffer is trimmed after message is processed. { MockChannelReader reader; Message message; message.WriteString(std::string(LargePayloadSize, 'X')); // Sanity check EXPECT_TRUE(message.size() > trimmed_buffer_size); // Initially buffer is small EXPECT_LE(reader.input_overflow_buf_.capacity(), trimmed_buffer_size); // Write and process large message reader.AppendMessageData(message); EXPECT_EQ(ChannelReader::DISPATCH_FINISHED, reader.ProcessIncomingMessages()); // After processing large message buffer is trimmed EXPECT_EQ(reader.input_overflow_buf_.capacity(), trimmed_buffer_size); } // Buffer is trimmed only after entire message is processed. { MockChannelReader reader; ExposedMessage message; message.WriteString(std::string(LargePayloadSize, 'X')); // Write and process message header reader.AppendData(message.header(), sizeof(ExposedMessage::Header)); EXPECT_EQ(ChannelReader::DISPATCH_FINISHED, reader.ProcessIncomingMessages()); // We determined message size for the message from its header, so // we resized the buffer to fit. EXPECT_GE(reader.input_overflow_buf_.capacity(), message.size()); // Write and process payload reader.AppendData(message.payload(), message.payload_size()); EXPECT_EQ(ChannelReader::DISPATCH_FINISHED, reader.ProcessIncomingMessages()); // But once we process the message, we trim the buffer EXPECT_EQ(reader.input_overflow_buf_.capacity(), trimmed_buffer_size); } // Buffer is not trimmed if the next message is also large. { MockChannelReader reader; // Write large message Message message1; message1.WriteString(std::string(LargePayloadSize * 2, 'X')); reader.AppendMessageData(message1); // Write header for the next large message ExposedMessage message2; message2.WriteString(std::string(LargePayloadSize, 'Y')); reader.AppendData(message2.header(), sizeof(ExposedMessage::Header)); // Process messages EXPECT_EQ(ChannelReader::DISPATCH_FINISHED, reader.ProcessIncomingMessages()); // We determined message size for the second (partial) message, so // we resized the buffer to fit. EXPECT_GE(reader.input_overflow_buf_.capacity(), message1.size()); } // Buffer resized appropriately if next message is larger than the first. // (Similar to the test above except for the order of messages.) { MockChannelReader reader; // Write large message Message message1; message1.WriteString(std::string(LargePayloadSize, 'Y')); reader.AppendMessageData(message1); // Write header for the next even larger message ExposedMessage message2; message2.WriteString(std::string(LargePayloadSize * 2, 'X')); reader.AppendData(message2.header(), sizeof(ExposedMessage::Header)); // Process messages EXPECT_EQ(ChannelReader::DISPATCH_FINISHED, reader.ProcessIncomingMessages()); // We determined message size for the second (partial) message, and // resized the buffer to fit it. EXPECT_GE(reader.input_overflow_buf_.capacity(), message2.size()); } // Buffer is not trimmed if we've just resized it to accommodate large // incoming message. { MockChannelReader reader; // Write small message Message message1; message1.WriteString(std::string(11, 'X')); reader.AppendMessageData(message1); // Write header for the next large message ExposedMessage message2; message2.WriteString(std::string(LargePayloadSize, 'Y')); reader.AppendData(message2.header(), sizeof(ExposedMessage::Header)); EXPECT_EQ(ChannelReader::DISPATCH_FINISHED, reader.ProcessIncomingMessages()); // We determined message size for the second (partial) message, so // we resized the buffer to fit. EXPECT_GE(reader.input_overflow_buf_.capacity(), message2.size()); } } } // namespace internal } // namespace IPC