Reboots to fastbootd (userspace fastboot) take a long time, particularly due to the orange AVB screen and the likelihood of devices having uart enabled. For "flashall", there is rarely a need to actually go into userspace, because all of super is getting thrown away. We can just flash super in the bootloader. In the past we didn't do this because computing super.img is expensive - both in terms of time (due to reading dependent images) and in terms of space (it's easily over 5GB). But we don't actually need to fully compute super.img. We can build a sparse_file containing the metadata/headers, with additional references to each image file containing partition data. Liblp provides the API to do that, and here, we simply need to translate the layout to libsparse. On Pixel, this reduces flashall time by around 35-50 seconds, or around 20% of total time, depending on whether uart is in use. There are some caveats, in which case we'll fall back to normal fastbootd. This does not work on non-A/B devices, on retrofit dynamic partition devices (Pixel 3), and in some other edge-casey scenarios. If it fails, -v will add logging information about why. Bue: 266982466 Test: fastboot flashall on Pixel 5+ Change-Id: Ie040da597d739faa7f834202184cec8f8e412076
125 lines
4.3 KiB
C++
125 lines
4.3 KiB
C++
//
|
|
// Copyright (C) 2023 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 "super_flash_helper.h"
|
|
|
|
#include <android-base/logging.h>
|
|
|
|
#include "util.h"
|
|
|
|
using android::base::borrowed_fd;
|
|
using android::base::unique_fd;
|
|
using android::fs_mgr::SuperImageExtent;
|
|
|
|
SuperFlashHelper::SuperFlashHelper(const ImageSource& source) : source_(source) {}
|
|
|
|
bool SuperFlashHelper::Open(borrowed_fd fd) {
|
|
if (!builder_.Open(fd)) {
|
|
LOG(VERBOSE) << "device does not support optimized super flashing";
|
|
return false;
|
|
}
|
|
|
|
base_metadata_ = builder_.Export();
|
|
return !!base_metadata_;
|
|
}
|
|
|
|
bool SuperFlashHelper::IncludeInSuper(const std::string& partition) {
|
|
return should_flash_in_userspace(*base_metadata_.get(), partition);
|
|
}
|
|
|
|
bool SuperFlashHelper::AddPartition(const std::string& partition, const std::string& image_name,
|
|
bool optional) {
|
|
if (!IncludeInSuper(partition)) {
|
|
return true;
|
|
}
|
|
auto iter = image_fds_.find(image_name);
|
|
if (iter == image_fds_.end()) {
|
|
unique_fd fd = source_.OpenFile(image_name);
|
|
if (fd < 0) {
|
|
if (!optional) {
|
|
LOG(VERBOSE) << "could not find partition image: " << image_name;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
if (is_sparse_file(fd)) {
|
|
LOG(VERBOSE) << "cannot optimize dynamic partitions with sparse images";
|
|
return false;
|
|
}
|
|
iter = image_fds_.emplace(image_name, std::move(fd)).first;
|
|
}
|
|
|
|
if (!builder_.AddPartition(partition, image_name, get_file_size(iter->second))) {
|
|
return false;
|
|
}
|
|
|
|
will_flash_.emplace(partition);
|
|
return true;
|
|
}
|
|
|
|
SparsePtr SuperFlashHelper::GetSparseLayout() {
|
|
// Cache extents since the sparse ptr depends on data pointers.
|
|
if (extents_.empty()) {
|
|
extents_ = builder_.GetImageLayout();
|
|
if (extents_.empty()) {
|
|
LOG(VERBOSE) << "device does not support optimized super flashing";
|
|
return {nullptr, nullptr};
|
|
}
|
|
}
|
|
|
|
unsigned int block_size = base_metadata_->geometry.logical_block_size;
|
|
int64_t flashed_size = extents_.back().offset + extents_.back().size;
|
|
SparsePtr s(sparse_file_new(block_size, flashed_size), sparse_file_destroy);
|
|
|
|
for (const auto& extent : extents_) {
|
|
if (extent.offset / block_size > UINT_MAX) {
|
|
// Super image is too big to send via sparse files (>8TB).
|
|
LOG(VERBOSE) << "super image is too big to flash";
|
|
return {nullptr, nullptr};
|
|
}
|
|
unsigned int block = extent.offset / block_size;
|
|
|
|
int rv = 0;
|
|
switch (extent.type) {
|
|
case SuperImageExtent::Type::DONTCARE:
|
|
break;
|
|
case SuperImageExtent::Type::ZERO:
|
|
rv = sparse_file_add_fill(s.get(), 0, extent.size, block);
|
|
break;
|
|
case SuperImageExtent::Type::DATA:
|
|
rv = sparse_file_add_data(s.get(), extent.blob->data(), extent.size, block);
|
|
break;
|
|
case SuperImageExtent::Type::PARTITION: {
|
|
auto iter = image_fds_.find(extent.image_name);
|
|
if (iter == image_fds_.end()) {
|
|
LOG(FATAL) << "image added but not found: " << extent.image_name;
|
|
return {nullptr, nullptr};
|
|
}
|
|
rv = sparse_file_add_fd(s.get(), iter->second.get(), extent.image_offset,
|
|
extent.size, block);
|
|
break;
|
|
}
|
|
default:
|
|
LOG(VERBOSE) << "unrecognized extent type in super image layout";
|
|
return {nullptr, nullptr};
|
|
}
|
|
if (rv) {
|
|
LOG(VERBOSE) << "sparse failure building super image layout";
|
|
return {nullptr, nullptr};
|
|
}
|
|
}
|
|
return s;
|
|
}
|