// Copyright 2015 The Chromium OS 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 <brillo/streams/memory_containers.h>
#include <limits>
#include <memory>
#include <brillo/streams/mock_stream.h>
#include <brillo/streams/stream_errors.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
using testing::DoAll;
using testing::Invoke;
using testing::InSequence;
using testing::Return;
using testing::WithArgs;
using testing::_;
namespace brillo {
namespace {
class MockContiguousBuffer : public data_container::ContiguousBufferBase {
public:
MockContiguousBuffer() = default;
MOCK_METHOD2(Resize, bool(size_t, ErrorPtr*));
MOCK_CONST_METHOD0(GetSize, size_t());
MOCK_CONST_METHOD0(IsReadOnly, bool());
MOCK_CONST_METHOD2(GetReadOnlyBuffer, const void*(size_t, ErrorPtr*));
MOCK_METHOD2(GetBuffer, void*(size_t, ErrorPtr*));
MOCK_CONST_METHOD3(CopyMemoryBlock, void(void*, const void*, size_t));
private:
DISALLOW_COPY_AND_ASSIGN(MockContiguousBuffer);
};
} // anonymous namespace
class MemoryContainerTest : public testing::Test {
public:
inline static void* IntToPtr(int addr) {
return reinterpret_cast<void*>(addr);
}
inline static const void* IntToConstPtr(int addr) {
return reinterpret_cast<const void*>(addr);
}
// Dummy buffer pointer values used as external data source/destination for
// read/write operations.
void* const test_read_buffer_ = IntToPtr(12345);
const void* const test_write_buffer_ = IntToConstPtr(67890);
// Dummy buffer pointer values used for internal buffer owned by the
// memory buffer container class.
const void* const const_buffer_ = IntToConstPtr(123);
void* const buffer_ = IntToPtr(456);
MockContiguousBuffer container_;
};
TEST_F(MemoryContainerTest, Read_WithinBuffer) {
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, GetReadOnlyBuffer(10, _))
.WillOnce(Return(const_buffer_));
EXPECT_CALL(container_,
CopyMemoryBlock(test_read_buffer_, const_buffer_, 50)).Times(1);
}
size_t read = 0;
ErrorPtr error;
EXPECT_TRUE(container_.Read(test_read_buffer_, 50, 10, &read, &error));
EXPECT_EQ(50, read);
EXPECT_EQ(nullptr, error.get());
}
TEST_F(MemoryContainerTest, Read_PastEndOfBuffer) {
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, GetReadOnlyBuffer(80, _))
.WillOnce(Return(const_buffer_));
EXPECT_CALL(container_,
CopyMemoryBlock(test_read_buffer_, const_buffer_, 20)).Times(1);
}
size_t read = 0;
EXPECT_TRUE(container_.Read(test_read_buffer_, 50, 80, &read, nullptr));
EXPECT_EQ(20, read);
}
TEST_F(MemoryContainerTest, Read_OutsideBuffer) {
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
size_t read = 0;
EXPECT_TRUE(container_.Read(test_read_buffer_, 50, 100, &read, nullptr));
EXPECT_EQ(0, read);
}
TEST_F(MemoryContainerTest, Read_Error) {
auto OnReadError = [](ErrorPtr* error) {
Error::AddTo(error, FROM_HERE, "domain", "read_error", "read error");
};
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, GetReadOnlyBuffer(0, _))
.WillOnce(DoAll(WithArgs<1>(Invoke(OnReadError)), Return(nullptr)));
}
size_t read = 0;
ErrorPtr error;
EXPECT_FALSE(container_.Read(test_read_buffer_, 10, 0, &read, &error));
EXPECT_EQ(0, read);
EXPECT_NE(nullptr, error.get());
EXPECT_EQ("domain", error->GetDomain());
EXPECT_EQ("read_error", error->GetCode());
EXPECT_EQ("read error", error->GetMessage());
}
TEST_F(MemoryContainerTest, Write_WithinBuffer) {
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, GetBuffer(10, _))
.WillOnce(Return(buffer_));
EXPECT_CALL(container_,
CopyMemoryBlock(buffer_, test_write_buffer_, 50)).Times(1);
}
size_t written = 0;
ErrorPtr error;
EXPECT_TRUE(container_.Write(test_write_buffer_, 50, 10, &written, &error));
EXPECT_EQ(50, written);
EXPECT_EQ(nullptr, error.get());
}
TEST_F(MemoryContainerTest, Write_PastEndOfBuffer) {
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, Resize(130, _)).WillOnce(Return(true));
EXPECT_CALL(container_, GetBuffer(80, _))
.WillOnce(Return(buffer_));
EXPECT_CALL(container_,
CopyMemoryBlock(buffer_, test_write_buffer_, 50)).Times(1);
}
size_t written = 0;
EXPECT_TRUE(container_.Write(test_write_buffer_, 50, 80, &written, nullptr));
EXPECT_EQ(50, written);
}
TEST_F(MemoryContainerTest, Write_OutsideBuffer) {
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, Resize(160, _)).WillOnce(Return(true));
EXPECT_CALL(container_, GetBuffer(110, _))
.WillOnce(Return(buffer_));
EXPECT_CALL(container_,
CopyMemoryBlock(buffer_, test_write_buffer_, 50)).Times(1);
}
size_t written = 0;
EXPECT_TRUE(container_.Write(test_write_buffer_, 50, 110, &written, nullptr));
EXPECT_EQ(50, written);
}
TEST_F(MemoryContainerTest, Write_Error_Resize) {
auto OnWriteError = [](ErrorPtr* error) {
Error::AddTo(error, FROM_HERE, "domain", "write_error", "resize error");
};
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, Resize(160, _))
.WillOnce(DoAll(WithArgs<1>(Invoke(OnWriteError)), Return(false)));
}
size_t written = 0;
ErrorPtr error;
EXPECT_FALSE(container_.Write(test_write_buffer_, 50, 110, &written, &error));
EXPECT_EQ(0, written);
EXPECT_NE(nullptr, error.get());
EXPECT_EQ("domain", error->GetDomain());
EXPECT_EQ("write_error", error->GetCode());
EXPECT_EQ("resize error", error->GetMessage());
}
TEST_F(MemoryContainerTest, Write_Error) {
auto OnWriteError = [](ErrorPtr* error) {
Error::AddTo(error, FROM_HERE, "domain", "write_error", "write error");
};
{
InSequence s;
EXPECT_CALL(container_, GetSize()).WillOnce(Return(100));
EXPECT_CALL(container_, Resize(160, _)).WillOnce(Return(true));
EXPECT_CALL(container_, GetBuffer(110, _))
.WillOnce(DoAll(WithArgs<1>(Invoke(OnWriteError)), Return(nullptr)));
}
size_t written = 0;
ErrorPtr error;
EXPECT_FALSE(container_.Write(test_write_buffer_, 50, 110, &written, &error));
EXPECT_EQ(0, written);
EXPECT_NE(nullptr, error.get());
EXPECT_EQ("domain", error->GetDomain());
EXPECT_EQ("write_error", error->GetCode());
EXPECT_EQ("write error", error->GetMessage());
}
} // namespace brillo