// 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