//
// Copyright (C) 2018 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 "update_engine/payload_generator/payload_generation_config.h"
#include
#include
#include
#include
#include
#include
#include
#include "update_engine/common/utils.h"
#include "update_engine/payload_consumer/verity_writer_android.h"
#include "update_engine/payload_generator/extent_ranges.h"
namespace chromeos_update_engine {
namespace {
bool AvbDescriptorCallback(const AvbDescriptor* descriptor, void* user_data) {
PartitionConfig* part = static_cast(user_data);
AvbDescriptor desc;
TEST_AND_RETURN_FALSE(
avb_descriptor_validate_and_byteswap(descriptor, &desc));
if (desc.tag != AVB_DESCRIPTOR_TAG_HASHTREE)
return true;
AvbHashtreeDescriptor hashtree;
TEST_AND_RETURN_FALSE(avb_hashtree_descriptor_validate_and_byteswap(
reinterpret_cast(descriptor), &hashtree));
// We only support version 1 right now, will need to introduce a new
// payload minor version to support new dm verity version.
TEST_AND_RETURN_FALSE(hashtree.dm_verity_version == 1);
part->verity.hash_tree_algorithm =
reinterpret_cast(hashtree.hash_algorithm);
const uint8_t* salt = reinterpret_cast(descriptor) +
sizeof(AvbHashtreeDescriptor) +
hashtree.partition_name_len;
part->verity.hash_tree_salt.assign(salt, salt + hashtree.salt_len);
TEST_AND_RETURN_FALSE(hashtree.data_block_size ==
part->fs_interface->GetBlockSize());
part->verity.hash_tree_data_extent =
ExtentForBytes(hashtree.data_block_size, 0, hashtree.image_size);
TEST_AND_RETURN_FALSE(hashtree.hash_block_size ==
part->fs_interface->GetBlockSize());
part->verity.hash_tree_extent = ExtentForBytes(
hashtree.hash_block_size, hashtree.tree_offset, hashtree.tree_size);
part->verity.fec_data_extent =
ExtentForBytes(hashtree.data_block_size, 0, hashtree.fec_offset);
part->verity.fec_extent = ExtentForBytes(
hashtree.data_block_size, hashtree.fec_offset, hashtree.fec_size);
part->verity.fec_roots = hashtree.fec_num_roots;
return true;
}
// Generate hash tree and FEC based on the verity config and verify that it
// matches the hash tree and FEC stored in the image.
bool VerifyVerityConfig(const PartitionConfig& part) {
const size_t block_size = part.fs_interface->GetBlockSize();
if (part.verity.hash_tree_extent.num_blocks() != 0) {
auto hash_function =
HashTreeBuilder::HashFunction(part.verity.hash_tree_algorithm);
TEST_AND_RETURN_FALSE(hash_function != nullptr);
HashTreeBuilder hash_tree_builder(block_size, hash_function);
uint64_t data_size =
part.verity.hash_tree_data_extent.num_blocks() * block_size;
uint64_t tree_size = hash_tree_builder.CalculateSize(data_size);
TEST_AND_RETURN_FALSE(
tree_size == part.verity.hash_tree_extent.num_blocks() * block_size);
TEST_AND_RETURN_FALSE(
hash_tree_builder.Initialize(data_size, part.verity.hash_tree_salt));
brillo::Blob buffer;
for (uint64_t offset = part.verity.hash_tree_data_extent.start_block() *
block_size,
data_end = offset + data_size;
offset < data_end;) {
constexpr uint64_t kBufferSize = 1024 * 1024;
size_t bytes_to_read = std::min(kBufferSize, data_end - offset);
TEST_AND_RETURN_FALSE(
utils::ReadFileChunk(part.path, offset, bytes_to_read, &buffer));
TEST_AND_RETURN_FALSE(
hash_tree_builder.Update(buffer.data(), buffer.size()));
offset += buffer.size();
buffer.clear();
}
TEST_AND_RETURN_FALSE(hash_tree_builder.BuildHashTree());
TEST_AND_RETURN_FALSE(utils::ReadFileChunk(
part.path,
part.verity.hash_tree_extent.start_block() * block_size,
tree_size,
&buffer));
TEST_AND_RETURN_FALSE(hash_tree_builder.CheckHashTree(buffer));
}
if (part.verity.fec_extent.num_blocks() != 0) {
TEST_AND_RETURN_FALSE(VerityWriterAndroid::EncodeFEC(
part.path,
part.verity.fec_data_extent.start_block() * block_size,
part.verity.fec_data_extent.num_blocks() * block_size,
part.verity.fec_extent.start_block() * block_size,
part.verity.fec_extent.num_blocks() * block_size,
part.verity.fec_roots,
block_size,
true /* verify_mode */));
}
return true;
}
} // namespace
bool ImageConfig::LoadVerityConfig() {
for (PartitionConfig& part : partitions) {
// Parse AVB devices.
if (part.size > sizeof(AvbFooter)) {
uint64_t footer_offset = part.size - sizeof(AvbFooter);
brillo::Blob buffer;
TEST_AND_RETURN_FALSE(utils::ReadFileChunk(
part.path, footer_offset, sizeof(AvbFooter), &buffer));
if (memcmp(buffer.data(), AVB_FOOTER_MAGIC, AVB_FOOTER_MAGIC_LEN) == 0) {
LOG(INFO) << "Parsing verity config from AVB footer for " << part.name;
AvbFooter footer;
TEST_AND_RETURN_FALSE(avb_footer_validate_and_byteswap(
reinterpret_cast(buffer.data()), &footer));
buffer.clear();
TEST_AND_RETURN_FALSE(
footer.vbmeta_offset + sizeof(AvbVBMetaImageHeader) <= part.size);
TEST_AND_RETURN_FALSE(utils::ReadFileChunk(
part.path, footer.vbmeta_offset, footer.vbmeta_size, &buffer));
TEST_AND_RETURN_FALSE(avb_descriptor_foreach(
buffer.data(), buffer.size(), AvbDescriptorCallback, &part));
}
}
// Parse VB1.0 devices with FEC metadata, devices with hash tree without
// FEC will be skipped for now.
if (part.verity.IsEmpty() && part.size > FEC_BLOCKSIZE) {
brillo::Blob fec_metadata;
TEST_AND_RETURN_FALSE(utils::ReadFileChunk(part.path,
part.size - FEC_BLOCKSIZE,
sizeof(fec_header),
&fec_metadata));
const fec_header* header =
reinterpret_cast(fec_metadata.data());
if (header->magic == FEC_MAGIC) {
LOG(INFO)
<< "Parsing verity config from Verified Boot 1.0 metadata for "
<< part.name;
const size_t block_size = part.fs_interface->GetBlockSize();
// FEC_VERITY_DISABLE skips verifying verity hash tree, because we will
// verify it ourselves later.
fec::io fh(part.path, O_RDONLY, FEC_VERITY_DISABLE);
TEST_AND_RETURN_FALSE(fh);
fec_verity_metadata verity_data;
if (fh.get_verity_metadata(verity_data)) {
auto verity_table = base::SplitString(verity_data.table,
" ",
base::KEEP_WHITESPACE,
base::SPLIT_WANT_ALL);
TEST_AND_RETURN_FALSE(verity_table.size() == 10);
size_t data_block_size = 0;
TEST_AND_RETURN_FALSE(
base::StringToSizeT(verity_table[3], &data_block_size));
TEST_AND_RETURN_FALSE(block_size == data_block_size);
size_t hash_block_size = 0;
TEST_AND_RETURN_FALSE(
base::StringToSizeT(verity_table[4], &hash_block_size));
TEST_AND_RETURN_FALSE(block_size == hash_block_size);
uint64_t num_data_blocks = 0;
TEST_AND_RETURN_FALSE(
base::StringToUint64(verity_table[5], &num_data_blocks));
part.verity.hash_tree_data_extent =
ExtentForRange(0, num_data_blocks);
uint64_t hash_start_block = 0;
TEST_AND_RETURN_FALSE(
base::StringToUint64(verity_table[6], &hash_start_block));
part.verity.hash_tree_algorithm = verity_table[7];
TEST_AND_RETURN_FALSE(base::HexStringToBytes(
verity_table[9], &part.verity.hash_tree_salt));
auto hash_function =
HashTreeBuilder::HashFunction(part.verity.hash_tree_algorithm);
TEST_AND_RETURN_FALSE(hash_function != nullptr);
HashTreeBuilder hash_tree_builder(block_size, hash_function);
uint64_t tree_size =
hash_tree_builder.CalculateSize(num_data_blocks * block_size);
part.verity.hash_tree_extent =
ExtentForRange(hash_start_block, tree_size / block_size);
}
fec_ecc_metadata ecc_data;
if (fh.get_ecc_metadata(ecc_data) && ecc_data.valid) {
TEST_AND_RETURN_FALSE(block_size == FEC_BLOCKSIZE);
part.verity.fec_data_extent = ExtentForRange(0, ecc_data.blocks);
part.verity.fec_extent =
ExtentForBytes(block_size, ecc_data.start, header->fec_size);
part.verity.fec_roots = ecc_data.roots;
}
}
}
if (!part.verity.IsEmpty()) {
TEST_AND_RETURN_FALSE(VerifyVerityConfig(part));
}
}
return true;
}
} // namespace chromeos_update_engine