250 lines
7.8 KiB
C++
250 lines
7.8 KiB
C++
// 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
|