blob: fd65588b3d9dcb64c0af9af0e335ae13a029469d [file] [log] [blame]
/*
* Copyright (C) 2019 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 "apexd_verity.h"
#include <android-base/file.h>
#include <android-base/result.h>
#include <android-base/unique_fd.h>
#include <verity/hash_tree_builder.h>
#include <filesystem>
#include <iomanip>
#include <sstream>
#include <string>
#include <vector>
#include "apex_constants.h"
#include "apex_file.h"
#include "apexd_utils.h"
using android::base::Dirname;
using android::base::ErrnoError;
using android::base::Error;
using android::base::ReadFully;
using android::base::Result;
using android::base::unique_fd;
namespace android {
namespace apex {
namespace {
uint8_t HexToBin(char h) {
if (h >= 'A' && h <= 'H') return h - 'A' + 10;
if (h >= 'a' && h <= 'h') return h - 'a' + 10;
return h - '0';
}
std::vector<uint8_t> HexToBin(const std::string& hex) {
std::vector<uint8_t> bin;
bin.reserve(hex.size() / 2);
for (size_t i = 0; i + 1 < hex.size(); i += 2) {
uint8_t c = (HexToBin(hex[i]) << 4) + HexToBin(hex[i + 1]);
bin.push_back(c);
}
return bin;
}
} // namespace
Result<void> VerifyVerityRootDigest(const ApexFile& apex) {
CHECK(apex.GetImageOffset().has_value());
CHECK(!apex.IsCompressed());
unique_fd fd(
TEMP_FAILURE_RETRY(open(apex.GetPath().c_str(), O_RDONLY | O_CLOEXEC)));
if (fd.get() == -1) {
return ErrnoError() << "Failed to open " << apex.GetPath();
}
auto verity_data =
OR_RETURN(apex.VerifyApexVerity(apex.GetBundledPublicKey()));
auto block_size = verity_data.desc->hash_block_size;
auto image_size = verity_data.desc->image_size;
auto hash_fn = HashTreeBuilder::HashFunction(verity_data.hash_algorithm);
if (hash_fn == nullptr) {
return Error() << "Unsupported hash algorithm "
<< verity_data.hash_algorithm;
}
auto builder = std::make_unique<HashTreeBuilder>(block_size, hash_fn);
if (!builder->Initialize(image_size, HexToBin(verity_data.salt))) {
return Error() << "Invalid image size " << image_size;
}
if (lseek(fd, apex.GetImageOffset().value(), SEEK_SET) == -1) {
return ErrnoError() << "Failed to seek";
}
auto block_count = image_size / block_size;
auto buf = std::vector<uint8_t>(block_size);
while (block_count-- > 0) {
if (!ReadFully(fd, buf.data(), block_size)) {
return Error() << "Failed to read";
}
if (!builder->Update(buf.data(), block_size)) {
return Error() << "Failed to build hashtree: Update";
}
}
if (!builder->BuildHashTree()) {
return Error() << "Failed to build hashtree: incomplete data";
}
auto golden_digest = HexToBin(verity_data.root_digest);
auto digest = builder->root_hash();
// This returns zero-padded digest.
// resize() it to compare with golden digest,
digest.resize(golden_digest.size());
if (digest != golden_digest) {
return Error() << "root digest mismatch";
}
return {};
}
} // namespace apex
} // namespace android