blob: a1b254d34c4e544a573737c6db56e6733ee0c54d [file] [log] [blame]
// Copyright (c) 2009 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 "update_engine/extent_writer.h"
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <algorithm>
#include <string>
#include <vector>
#include <gtest/gtest.h>
#include "update_engine/payload_constants.h"
#include "update_engine/test_utils.h"
#include "update_engine/utils.h"
using std::min;
using std::string;
using std::vector;
namespace chromeos_update_engine {
COMPILE_ASSERT(sizeof(off_t) == 8, off_t_not_64_bit);
namespace {
const char kPathTemplate[] = "./ExtentWriterTest-file.XXXXXX";
const size_t kBlockSize = 4096;
}
class ExtentWriterTest : public ::testing::Test {
protected:
virtual void SetUp() {
memcpy(path_, kPathTemplate, sizeof(kPathTemplate));
fd_ = mkstemp(path_);
ASSERT_GE(fd_, 0);
}
virtual void TearDown() {
close(fd_);
unlink(path_);
}
int fd() { return fd_; }
const char* path() { return path_; }
// Writes data to an extent writer in 'chunk_size' chunks with
// the first chunk of size first_chunk_size. It calculates what the
// resultant file should look like and ensure that the extent writer
// wrote the file correctly.
void WriteAlignedExtents(size_t chunk_size, size_t first_chunk_size);
void TestZeroPad(bool aligned_size);
private:
int fd_;
char path_[sizeof(kPathTemplate)];
};
TEST_F(ExtentWriterTest, SimpleTest) {
vector<Extent> extents;
Extent extent;
extent.set_start_block(1);
extent.set_num_blocks(1);
extents.push_back(extent);
const string bytes = "1234";
DirectExtentWriter direct_writer;
EXPECT_TRUE(direct_writer.Init(fd(), extents, kBlockSize));
EXPECT_TRUE(direct_writer.Write(bytes.data(), bytes.size()));
EXPECT_TRUE(direct_writer.End());
EXPECT_EQ(kBlockSize + bytes.size(), utils::FileSize(fd()));
vector<char> result_file;
EXPECT_TRUE(utils::ReadFile(path(), &result_file));
vector<char> expected_file(kBlockSize);
expected_file.insert(expected_file.end(),
bytes.data(), bytes.data() + bytes.size());
ExpectVectorsEq(expected_file, result_file);
}
TEST_F(ExtentWriterTest, ZeroLengthTest) {
vector<Extent> extents;
Extent extent;
extent.set_start_block(1);
extent.set_num_blocks(1);
extents.push_back(extent);
DirectExtentWriter direct_writer;
EXPECT_TRUE(direct_writer.Init(fd(), extents, kBlockSize));
EXPECT_TRUE(direct_writer.Write(nullptr, 0));
EXPECT_TRUE(direct_writer.End());
}
TEST_F(ExtentWriterTest, OverflowExtentTest) {
WriteAlignedExtents(kBlockSize * 3, kBlockSize * 3);
}
TEST_F(ExtentWriterTest, UnalignedWriteTest) {
WriteAlignedExtents(7, 7);
}
TEST_F(ExtentWriterTest, LargeUnalignedWriteTest) {
WriteAlignedExtents(kBlockSize * 2, kBlockSize / 2);
}
void ExtentWriterTest::WriteAlignedExtents(size_t chunk_size,
size_t first_chunk_size) {
vector<Extent> extents;
Extent extent;
extent.set_start_block(1);
extent.set_num_blocks(1);
extents.push_back(extent);
extent.set_start_block(0);
extent.set_num_blocks(1);
extents.push_back(extent);
extent.set_start_block(2);
extent.set_num_blocks(1);
extents.push_back(extent);
vector<char> data(kBlockSize * 3);
FillWithData(&data);
DirectExtentWriter direct_writer;
EXPECT_TRUE(direct_writer.Init(fd(), extents, kBlockSize));
size_t bytes_written = 0;
while (bytes_written < data.size()) {
size_t bytes_to_write = min(data.size() - bytes_written, chunk_size);
if (bytes_written == 0) {
bytes_to_write = min(data.size() - bytes_written, first_chunk_size);
}
EXPECT_TRUE(direct_writer.Write(&data[bytes_written], bytes_to_write));
bytes_written += bytes_to_write;
}
EXPECT_TRUE(direct_writer.End());
EXPECT_EQ(data.size(), utils::FileSize(fd()));
vector<char> result_file;
EXPECT_TRUE(utils::ReadFile(path(), &result_file));
vector<char> expected_file;
expected_file.insert(expected_file.end(),
data.begin() + kBlockSize,
data.begin() + kBlockSize * 2);
expected_file.insert(expected_file.end(),
data.begin(), data.begin() + kBlockSize);
expected_file.insert(expected_file.end(),
data.begin() + kBlockSize * 2, data.end());
ExpectVectorsEq(expected_file, result_file);
}
TEST_F(ExtentWriterTest, ZeroPadNullTest) {
TestZeroPad(true);
}
TEST_F(ExtentWriterTest, ZeroPadFillTest) {
TestZeroPad(false);
}
void ExtentWriterTest::TestZeroPad(bool aligned_size) {
vector<Extent> extents;
Extent extent;
extent.set_start_block(1);
extent.set_num_blocks(1);
extents.push_back(extent);
extent.set_start_block(0);
extent.set_num_blocks(1);
extents.push_back(extent);
vector<char> data(kBlockSize * 2);
FillWithData(&data);
DirectExtentWriter direct_writer;
ZeroPadExtentWriter zero_pad_writer(&direct_writer);
EXPECT_TRUE(zero_pad_writer.Init(fd(), extents, kBlockSize));
size_t bytes_to_write = data.size();
const size_t missing_bytes = (aligned_size ? 0 : 9);
bytes_to_write -= missing_bytes;
lseek64(fd(), kBlockSize - missing_bytes, SEEK_SET);
EXPECT_EQ(3, write(fd(), "xxx", 3));
ASSERT_TRUE(zero_pad_writer.Write(&data[0], bytes_to_write));
EXPECT_TRUE(zero_pad_writer.End());
EXPECT_EQ(data.size(), utils::FileSize(fd()));
vector<char> result_file;
EXPECT_TRUE(utils::ReadFile(path(), &result_file));
vector<char> expected_file;
expected_file.insert(expected_file.end(),
data.begin() + kBlockSize,
data.begin() + kBlockSize * 2);
expected_file.insert(expected_file.end(),
data.begin(), data.begin() + kBlockSize);
if (missing_bytes) {
memset(&expected_file[kBlockSize - missing_bytes], 0, missing_bytes);
}
ExpectVectorsEq(expected_file, result_file);
}
TEST_F(ExtentWriterTest, SparseFileTest) {
vector<Extent> extents;
Extent extent;
extent.set_start_block(1);
extent.set_num_blocks(1);
extents.push_back(extent);
extent.set_start_block(kSparseHole);
extent.set_num_blocks(2);
extents.push_back(extent);
extent.set_start_block(0);
extent.set_num_blocks(1);
extents.push_back(extent);
const int block_count = 4;
const int on_disk_count = 2;
vector<char> data(17);
FillWithData(&data);
DirectExtentWriter direct_writer;
EXPECT_TRUE(direct_writer.Init(fd(), extents, kBlockSize));
size_t bytes_written = 0;
while (bytes_written < (block_count * kBlockSize)) {
size_t bytes_to_write = min(block_count * kBlockSize - bytes_written,
data.size());
EXPECT_TRUE(direct_writer.Write(&data[0], bytes_to_write));
bytes_written += bytes_to_write;
}
EXPECT_TRUE(direct_writer.End());
// check file size, then data inside
ASSERT_EQ(2 * kBlockSize, utils::FileSize(path()));
vector<char> resultant_data;
EXPECT_TRUE(utils::ReadFile(path(), &resultant_data));
// Create expected data
vector<char> expected_data(on_disk_count * kBlockSize);
vector<char> big(block_count * kBlockSize);
for (vector<char>::size_type i = 0; i < big.size(); i++) {
big[i] = data[i % data.size()];
}
memcpy(&expected_data[kBlockSize], &big[0], kBlockSize);
memcpy(&expected_data[0], &big[3 * kBlockSize], kBlockSize);
ExpectVectorsEq(expected_data, resultant_data);
}
} // namespace chromeos_update_engine