|  | // | 
|  | // Copyright (C) 2021 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 <unistd.h> | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <mutex> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include <base/format_macros.h> | 
|  | #include <base/logging.h> | 
|  | #include <base/strings/string_number_conversions.h> | 
|  | #include <base/strings/string_util.h> | 
|  | #include <base/strings/stringprintf.h> | 
|  | #include <gtest/gtest.h> | 
|  | #include <erofs/internal.h> | 
|  | #include <erofs/io.h> | 
|  |  | 
|  | #include "update_engine/common/test_utils.h" | 
|  | #include "update_engine/common/utils.h" | 
|  | #include "update_engine/lz4diff/lz4diff_compress.h" | 
|  | #include "update_engine/payload_generator/delta_diff_generator.h" | 
|  | #include "update_engine/payload_generator/erofs_filesystem.h" | 
|  | #include "update_engine/payload_generator/extent_utils.h" | 
|  |  | 
|  | using std::string; | 
|  | using std::vector; | 
|  |  | 
|  | namespace chromeos_update_engine { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | static void ExtractErofsImage(const char* erofs_image, | 
|  | const char* inode_path, | 
|  | Blob* output) { | 
|  | // EROFS has plenty of global variable usage. Protect calls to EROFS APIs with | 
|  | // global mutex. | 
|  | // TODO(b/202784930) Replace erofs-utils with a cleaner and more C++ friendly | 
|  | // library. (Or turn erofs-utils into one) | 
|  | static std::mutex mutex; | 
|  | std::lock_guard lock(mutex); | 
|  | auto err = dev_open_ro(erofs_image); | 
|  | ASSERT_EQ(err, 0); | 
|  | DEFER { dev_close(); }; | 
|  |  | 
|  | err = erofs_read_superblock(); | 
|  | ASSERT_EQ(err, 0); | 
|  | struct erofs_inode inode; | 
|  | err = erofs_ilookup(inode_path, &inode); | 
|  | ASSERT_EQ(err, 0); | 
|  | output->resize(inode.i_size); | 
|  | err = erofs_pread(&inode, | 
|  | reinterpret_cast<char*>(output->data()), | 
|  | output->size(), | 
|  | 0 /* offset */); | 
|  | ASSERT_EQ(err, 0); | 
|  | } | 
|  |  | 
|  | class Lz4diffCompressTest : public ::testing::Test {}; | 
|  |  | 
|  | using test_utils::GetBuildArtifactsPath; | 
|  |  | 
|  | // This test parses the sample images generated during build time with the | 
|  | // "generate_image.sh" script. The expected conditions of each file in these | 
|  | // images is encoded in the file name, as defined in the mentioned script. | 
|  | TEST_F(Lz4diffCompressTest, ExtractElfBinary) { | 
|  | const auto build_path = GetBuildArtifactsPath("gen/erofs.img"); | 
|  | auto fs = ErofsFilesystem::CreateFromFile(build_path); | 
|  | ASSERT_NE(fs, nullptr); | 
|  | ASSERT_EQ(kBlockSize, fs->GetBlockSize()); | 
|  |  | 
|  | vector<ErofsFilesystem::File> files; | 
|  | ASSERT_TRUE(fs->GetFiles(&files)); | 
|  |  | 
|  | const auto it = | 
|  | std::find_if(files.begin(), files.end(), [](const auto& file) { | 
|  | return file.name == "/delta_generator"; | 
|  | }); | 
|  | ASSERT_NE(it, files.end()) | 
|  | << "There should be a delta_generator entry in gen/erofs.img. Is the " | 
|  | "generate_test_erofs_imgages.sh script implemented wrong?"; | 
|  |  | 
|  | const auto delta_generator = *it; | 
|  | Blob expected_blob; | 
|  | ASSERT_NO_FATAL_FAILURE(ExtractErofsImage( | 
|  | build_path.c_str(), "/delta_generator", &expected_blob)); | 
|  | Blob compressed_blob; | 
|  | ASSERT_TRUE(utils::ReadExtents( | 
|  | build_path, delta_generator.extents, &compressed_blob, kBlockSize)); | 
|  | auto decompressed_blob = TryDecompressBlob( | 
|  | compressed_blob, | 
|  | delta_generator.compressed_file_info.blocks, | 
|  | delta_generator.compressed_file_info.zero_padding_enabled); | 
|  | ASSERT_GT(decompressed_blob.size(), 0UL); | 
|  | ASSERT_GE(decompressed_blob.size(), | 
|  | static_cast<size_t>(delta_generator.file_stat.st_size)); | 
|  | decompressed_blob.resize(delta_generator.file_stat.st_size); | 
|  | ASSERT_EQ(decompressed_blob, expected_blob); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | }  // namespace chromeos_update_engine |