1 // 2 // Copyright (C) 2015 The Android Open Source Project 3 // 4 // Licensed under the Apache License, Version 2.0 (the "License"); 5 // you may not use this file except in compliance with the License. 6 // You may obtain a copy of the License at 7 // 8 // http://www.apache.org/licenses/LICENSE-2.0 9 // 10 // Unless required by applicable law or agreed to in writing, software 11 // distributed under the License is distributed on an "AS IS" BASIS, 12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 // See the License for the specific language governing permissions and 14 // limitations under the License. 15 // 16 17 #include "update_engine/payload_generator/ext2_filesystem.h" 18 19 #include <unistd.h> 20 21 #include <map> 22 #include <set> 23 #include <string> 24 #include <vector> 25 26 #include <base/format_macros.h> 27 #include <base/logging.h> 28 #include <base/strings/string_number_conversions.h> 29 #include <base/strings/string_util.h> 30 #include <base/strings/stringprintf.h> 31 #include <gtest/gtest.h> 32 33 #include "update_engine/common/test_utils.h" 34 #include "update_engine/common/utils.h" 35 #include "update_engine/payload_generator/extent_utils.h" 36 37 using std::map; 38 using std::set; 39 using std::string; 40 using std::unique_ptr; 41 using std::vector; 42 43 namespace chromeos_update_engine { 44 45 namespace { 46 47 uint64_t kDefaultFilesystemSize = 4 * 1024 * 1024; 48 size_t kDefaultFilesystemBlockCount = 1024; 49 size_t kDefaultFilesystemBlockSize = 4096; 50 51 // Checks that all the blocks in |extents| are in the range [0, total_blocks). 52 void ExpectBlocksInRange(const vector<Extent>& extents, uint64_t total_blocks) { 53 for (const Extent& extent : extents) { 54 EXPECT_LE(0U, extent.start_block()); 55 EXPECT_LE(extent.start_block() + extent.num_blocks(), total_blocks); 56 } 57 } 58 59 } // namespace 60 61 class Ext2FilesystemTest : public ::testing::Test {}; 62 63 TEST_F(Ext2FilesystemTest, InvalidFilesystem) { 64 test_utils::ScopedTempFile fs_filename_{"Ext2FilesystemTest-XXXXXX"}; 65 ASSERT_EQ(0, truncate(fs_filename_.path().c_str(), kDefaultFilesystemSize)); 66 unique_ptr<Ext2Filesystem> fs = 67 Ext2Filesystem::CreateFromFile(fs_filename_.path()); 68 ASSERT_EQ(nullptr, fs.get()); 69 70 fs = Ext2Filesystem::CreateFromFile("/path/to/invalid/file"); 71 ASSERT_EQ(nullptr, fs.get()); 72 } 73 74 TEST_F(Ext2FilesystemTest, EmptyFilesystem) { 75 base::FilePath path = 76 test_utils::GetBuildArtifactsPath().Append("gen/disk_ext2_4k_empty.img"); 77 unique_ptr<Ext2Filesystem> fs = Ext2Filesystem::CreateFromFile(path.value()); 78 79 ASSERT_NE(nullptr, fs.get()); 80 EXPECT_EQ(kDefaultFilesystemBlockCount, fs->GetBlockCount()); 81 EXPECT_EQ(kDefaultFilesystemBlockSize, fs->GetBlockSize()); 82 83 vector<FilesystemInterface::File> files; 84 EXPECT_TRUE(fs->GetFiles(&files)); 85 86 map<string, FilesystemInterface::File> map_files; 87 for (const auto& file : files) { 88 EXPECT_EQ(map_files.end(), map_files.find(file.name)) 89 << "File " << file.name << " repeated in the list."; 90 map_files[file.name] = file; 91 ExpectBlocksInRange(file.extents, fs->GetBlockCount()); 92 } 93 EXPECT_EQ(2U, map_files["/"].file_stat.st_ino); 94 EXPECT_FALSE(map_files["<free-space>"].extents.empty()); 95 } 96 97 // This test parses the sample images generated during build time with the 98 // "generate_image.sh" script. The expected conditions of each file in these 99 // images is encoded in the file name, as defined in the mentioned script. 100 TEST_F(Ext2FilesystemTest, ParseGeneratedImages) { 101 const vector<string> kGeneratedImages = { 102 "disk_ext2_1k.img", 103 "disk_ext2_4k.img" }; 104 base::FilePath build_path = test_utils::GetBuildArtifactsPath().Append("gen"); 105 for (const string& fs_name : kGeneratedImages) { 106 LOG(INFO) << "Testing " << fs_name; 107 unique_ptr<Ext2Filesystem> fs = Ext2Filesystem::CreateFromFile( 108 build_path.Append(fs_name).value()); 109 ASSERT_NE(nullptr, fs.get()); 110 111 vector<FilesystemInterface::File> files; 112 map<string, FilesystemInterface::File> map_files; 113 set<string> filenames; 114 EXPECT_TRUE(fs->GetFiles(&files)); 115 for (const auto& file : files) { 116 // Check no repeated files. We should parse hard-links with two different 117 // names. 118 EXPECT_EQ(map_files.end(), map_files.find(file.name)) 119 << "File " << file.name << " repeated in the list."; 120 map_files[file.name] = file; 121 filenames.insert(file.name); 122 ExpectBlocksInRange(file.extents, fs->GetBlockCount()); 123 } 124 125 // Check that all the files are parsed, and the /removed file should not 126 // be included in the list. 127 set<string> kExpectedFiles = { 128 "/", 129 "/dir1", 130 "/dir1/file", 131 "/dir1/dir2", 132 "/dir1/dir2/file", 133 "/dir1/dir2/dir1", 134 "/empty-file", 135 "/link-hard-regular-16k", 136 "/link-long_symlink", 137 "/link-short_symlink", 138 "/lost+found", 139 "/regular-small", 140 "/regular-16k", 141 "/regular-32k-zeros", 142 "/regular-with_net_cap", 143 "/sparse_empty-10k", 144 "/sparse_empty-2blocks", 145 "/sparse-10000blocks", 146 "/sparse-16k-last_block", 147 "/sparse-16k-first_block", 148 "/sparse-16k-holes", 149 "<inode-blocks>", 150 "<free-space>", 151 "<group-descriptors>", 152 }; 153 EXPECT_EQ(kExpectedFiles, filenames); 154 155 FilesystemInterface::File file; 156 157 // Small symlinks don't actually have data blocks. 158 EXPECT_TRUE(map_files["/link-short_symlink"].extents.empty()); 159 EXPECT_EQ(1U, BlocksInExtents(map_files["/link-long_symlink"].extents)); 160 161 // Hard-links report the same list of blocks. 162 EXPECT_EQ(map_files["/link-hard-regular-16k"].extents, 163 map_files["/regular-16k"].extents); 164 EXPECT_FALSE(map_files["/regular-16k"].extents.empty()); 165 166 // The number of blocks in these files doesn't depend on the 167 // block size. 168 EXPECT_TRUE(map_files["/empty-file"].extents.empty()); 169 EXPECT_EQ(1U, BlocksInExtents(map_files["/regular-small"].extents)); 170 EXPECT_EQ(1U, BlocksInExtents(map_files["/regular-with_net_cap"].extents)); 171 EXPECT_TRUE(map_files["/sparse_empty-10k"].extents.empty()); 172 EXPECT_TRUE(map_files["/sparse_empty-2blocks"].extents.empty()); 173 EXPECT_EQ(1U, BlocksInExtents(map_files["/sparse-16k-last_block"].extents)); 174 EXPECT_EQ(1U, 175 BlocksInExtents(map_files["/sparse-16k-first_block"].extents)); 176 EXPECT_EQ(2U, BlocksInExtents(map_files["/sparse-16k-holes"].extents)); 177 } 178 } 179 180 TEST_F(Ext2FilesystemTest, LoadSettingsFailsTest) { 181 base::FilePath path = test_utils::GetBuildArtifactsPath().Append( 182 "gen/disk_ext2_1k.img"); 183 unique_ptr<Ext2Filesystem> fs = Ext2Filesystem::CreateFromFile(path.value()); 184 ASSERT_NE(nullptr, fs.get()); 185 186 brillo::KeyValueStore store; 187 // disk_ext2_1k.img doesn't have the /etc/update_engine.conf file. 188 EXPECT_FALSE(fs->LoadSettings(&store)); 189 } 190 191 TEST_F(Ext2FilesystemTest, LoadSettingsWorksTest) { 192 base::FilePath path = 193 test_utils::GetBuildArtifactsPath().Append("gen/disk_ext2_unittest.img"); 194 unique_ptr<Ext2Filesystem> fs = Ext2Filesystem::CreateFromFile(path.value()); 195 ASSERT_NE(nullptr, fs.get()); 196 197 brillo::KeyValueStore store; 198 EXPECT_TRUE(fs->LoadSettings(&store)); 199 string minor_version; 200 EXPECT_TRUE(store.GetString("PAYLOAD_MINOR_VERSION", &minor_version)); 201 EXPECT_EQ("1234", minor_version); 202 } 203 204 } // namespace chromeos_update_engine 205