311e6ca7b6
Adds a fuse data provider that parses the metadata from a block map, reads the data from the given ranges of the block device; and provides the data to the fuse. Bug: 127071893 Test: unit tests pass, install a package from block map Change-Id: Ie9925ee9144e98642505b3f5e1a4a186d2b21ed0
199 lines
7.1 KiB
C++
199 lines
7.1 KiB
C++
/*
|
|
* Copyright 2016 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include <string>
|
|
|
|
#include <android-base/file.h>
|
|
#include <android-base/strings.h>
|
|
#include <gtest/gtest.h>
|
|
|
|
#include "otautil/rangeset.h"
|
|
#include "otautil/sysutil.h"
|
|
|
|
TEST(SysUtilTest, InvalidArgs) {
|
|
MemMapping mapping;
|
|
|
|
// Invalid argument.
|
|
ASSERT_FALSE(mapping.MapFile(""));
|
|
}
|
|
|
|
TEST(SysUtilTest, ParseBlockMapFile_smoke) {
|
|
std::vector<std::string> content = {
|
|
"/dev/abc", "49652 4096", "3", "1000 1008", "2100 2102", "30 33",
|
|
};
|
|
|
|
TemporaryFile temp_file;
|
|
ASSERT_TRUE(android::base::WriteStringToFile(android::base::Join(content, '\n'), temp_file.path));
|
|
|
|
auto block_map_data = BlockMapData::ParseBlockMapFile(temp_file.path);
|
|
ASSERT_EQ("/dev/abc", block_map_data.path());
|
|
ASSERT_EQ(49652, block_map_data.file_size());
|
|
ASSERT_EQ(4096, block_map_data.block_size());
|
|
ASSERT_EQ(RangeSet(std::vector<Range>{
|
|
{ 1000, 1008 },
|
|
{ 2100, 2102 },
|
|
{ 30, 33 },
|
|
}),
|
|
block_map_data.block_ranges());
|
|
}
|
|
|
|
TEST(SysUtilTest, ParseBlockMapFile_invalid_line_count) {
|
|
std::vector<std::string> content = {
|
|
"/dev/abc", "49652 4096", "2", "1000 1008", "2100 2102", "30 33",
|
|
};
|
|
|
|
TemporaryFile temp_file;
|
|
ASSERT_TRUE(android::base::WriteStringToFile(android::base::Join(content, '\n'), temp_file.path));
|
|
|
|
auto block_map_data1 = BlockMapData::ParseBlockMapFile(temp_file.path);
|
|
ASSERT_FALSE(block_map_data1);
|
|
}
|
|
|
|
TEST(SysUtilTest, ParseBlockMapFile_invalid_size) {
|
|
std::vector<std::string> content = {
|
|
"/dev/abc",
|
|
"42949672950 4294967295",
|
|
"1",
|
|
"0 10",
|
|
};
|
|
|
|
TemporaryFile temp_file;
|
|
ASSERT_TRUE(android::base::WriteStringToFile(android::base::Join(content, '\n'), temp_file.path));
|
|
|
|
auto block_map_data = BlockMapData::ParseBlockMapFile(temp_file.path);
|
|
ASSERT_EQ("/dev/abc", block_map_data.path());
|
|
ASSERT_EQ(42949672950, block_map_data.file_size());
|
|
ASSERT_EQ(4294967295, block_map_data.block_size());
|
|
|
|
content[1] = "42949672950 4294967296";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(android::base::Join(content, '\n'), temp_file.path));
|
|
auto large_block_size = BlockMapData::ParseBlockMapFile(temp_file.path);
|
|
ASSERT_FALSE(large_block_size);
|
|
|
|
content[1] = "4294967296 1";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(android::base::Join(content, '\n'), temp_file.path));
|
|
auto too_many_blocks = BlockMapData::ParseBlockMapFile(temp_file.path);
|
|
ASSERT_FALSE(too_many_blocks);
|
|
}
|
|
|
|
TEST(SysUtilTest, MapFileRegularFile) {
|
|
TemporaryFile temp_file1;
|
|
std::string content = "abc";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(content, temp_file1.path));
|
|
|
|
// MemMapping::MapFile() should map the file to one range.
|
|
MemMapping mapping;
|
|
ASSERT_TRUE(mapping.MapFile(temp_file1.path));
|
|
ASSERT_NE(nullptr, mapping.addr);
|
|
ASSERT_EQ(content.size(), mapping.length);
|
|
ASSERT_EQ(1U, mapping.ranges());
|
|
}
|
|
|
|
TEST(SysUtilTest, MapFileBlockMap) {
|
|
// Create a file that has 10 blocks.
|
|
TemporaryFile package;
|
|
std::string content;
|
|
constexpr size_t file_size = 4096 * 10;
|
|
content.reserve(file_size);
|
|
ASSERT_TRUE(android::base::WriteStringToFile(content, package.path));
|
|
|
|
TemporaryFile block_map_file;
|
|
std::string filename = std::string("@") + block_map_file.path;
|
|
MemMapping mapping;
|
|
|
|
// One range.
|
|
std::string block_map_content = std::string(package.path) + "\n40960 4096\n1\n0 10\n";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(block_map_content, block_map_file.path));
|
|
|
|
ASSERT_TRUE(mapping.MapFile(filename));
|
|
ASSERT_EQ(file_size, mapping.length);
|
|
ASSERT_EQ(1U, mapping.ranges());
|
|
|
|
// It's okay to not have the trailing '\n'.
|
|
block_map_content = std::string(package.path) + "\n40960 4096\n1\n0 10";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(block_map_content, block_map_file.path));
|
|
|
|
ASSERT_TRUE(mapping.MapFile(filename));
|
|
ASSERT_EQ(file_size, mapping.length);
|
|
ASSERT_EQ(1U, mapping.ranges());
|
|
|
|
// Or having multiple trailing '\n's.
|
|
block_map_content = std::string(package.path) + "\n40960 4096\n1\n0 10\n\n\n";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(block_map_content, block_map_file.path));
|
|
|
|
ASSERT_TRUE(mapping.MapFile(filename));
|
|
ASSERT_EQ(file_size, mapping.length);
|
|
ASSERT_EQ(1U, mapping.ranges());
|
|
|
|
// Multiple ranges.
|
|
block_map_content = std::string(package.path) + "\n40960 4096\n3\n0 3\n3 5\n5 10\n";
|
|
ASSERT_TRUE(android::base::WriteStringToFile(block_map_content, block_map_file.path));
|
|
|
|
ASSERT_TRUE(mapping.MapFile(filename));
|
|
ASSERT_EQ(file_size, mapping.length);
|
|
ASSERT_EQ(3U, mapping.ranges());
|
|
}
|
|
|
|
TEST(SysUtilTest, MapFileBlockMapInvalidBlockMap) {
|
|
MemMapping mapping;
|
|
TemporaryFile temp_file;
|
|
std::string filename = std::string("@") + temp_file.path;
|
|
|
|
// Block map file is too short.
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n4096 4096\n0\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
// Block map file has unexpected number of lines.
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n4096 4096\n1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n4096 4096\n2\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
// Invalid size/blksize/range_count.
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\nabc 4096\n1\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n4096 4096\n\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
// size/blksize/range_count don't match.
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n0 4096\n1\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n4096 0\n1\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/somefile\n4096 4096\n0\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
|
|
// Invalid block dev path.
|
|
ASSERT_TRUE(android::base::WriteStringToFile("/doesntexist\n4096 4096\n1\n0 1\n", temp_file.path));
|
|
ASSERT_FALSE(mapping.MapFile(filename));
|
|
}
|
|
|
|
TEST(SysUtilTest, StringVectorToNullTerminatedArray) {
|
|
std::vector<std::string> args{ "foo", "bar", "baz" };
|
|
auto args_with_nullptr = StringVectorToNullTerminatedArray(args);
|
|
ASSERT_EQ(4, args_with_nullptr.size());
|
|
ASSERT_STREQ("foo", args_with_nullptr[0]);
|
|
ASSERT_STREQ("bar", args_with_nullptr[1]);
|
|
ASSERT_STREQ("baz", args_with_nullptr[2]);
|
|
ASSERT_EQ(nullptr, args_with_nullptr[3]);
|
|
}
|