| // |
| // 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 "task.h" |
| |
| #include <iostream> |
| |
| #include <android-base/logging.h> |
| #include <android-base/parseint.h> |
| |
| #include "fastboot.h" |
| #include "filesystem.h" |
| #include "super_flash_helper.h" |
| #include "util.h" |
| |
| using namespace std::string_literals; |
| FlashTask::FlashTask(const std::string& slot, const std::string& pname, const std::string& fname, |
| const bool apply_vbmeta) |
| : pname_(pname), fname_(fname), slot_(slot), apply_vbmeta_(apply_vbmeta) {} |
| |
| void FlashTask::Run() { |
| auto flash = [&](const std::string& partition) { |
| if (should_flash_in_userspace(partition) && !is_userspace_fastboot()) { |
| die("The partition you are trying to flash is dynamic, and " |
| "should be flashed via fastbootd. Please run:\n" |
| "\n" |
| " fastboot reboot fastboot\n" |
| "\n" |
| "And try again. If you are intentionally trying to " |
| "overwrite a fixed partition, use --force."); |
| } |
| do_flash(partition.c_str(), fname_.c_str(), apply_vbmeta_); |
| }; |
| do_for_partitions(pname_, slot_, flash, true); |
| } |
| |
| std::string FlashTask::GetPartitionAndSlot() { |
| auto slot = slot_; |
| if (slot.empty()) { |
| slot = get_current_slot(); |
| } |
| if (slot.empty()) { |
| return pname_; |
| } |
| if (slot == "all") { |
| LOG(FATAL) << "Cannot retrieve a singular name when using all slots"; |
| } |
| return pname_ + "_" + slot; |
| } |
| |
| RebootTask::RebootTask(const FlashingPlan* fp) : fp_(fp){}; |
| RebootTask::RebootTask(const FlashingPlan* fp, const std::string& reboot_target) |
| : reboot_target_(reboot_target), fp_(fp){}; |
| |
| void RebootTask::Run() { |
| if (reboot_target_ == "fastboot") { |
| if (!is_userspace_fastboot()) { |
| reboot_to_userspace_fastboot(); |
| fp_->fb->WaitForDisconnect(); |
| } |
| } else if (reboot_target_ == "recovery") { |
| fp_->fb->RebootTo("recovery"); |
| fp_->fb->WaitForDisconnect(); |
| } else if (reboot_target_ == "bootloader") { |
| fp_->fb->RebootTo("bootloader"); |
| fp_->fb->WaitForDisconnect(); |
| } else if (reboot_target_ == "") { |
| fp_->fb->Reboot(); |
| fp_->fb->WaitForDisconnect(); |
| } else { |
| syntax_error("unknown reboot target %s", reboot_target_.c_str()); |
| } |
| } |
| |
| FlashSuperLayoutTask::FlashSuperLayoutTask(const std::string& super_name, |
| std::unique_ptr<SuperFlashHelper> helper, |
| SparsePtr sparse_layout, uint64_t super_size) |
| : super_name_(super_name), |
| helper_(std::move(helper)), |
| sparse_layout_(std::move(sparse_layout)), |
| super_size_(super_size) {} |
| |
| void FlashSuperLayoutTask::Run() { |
| // Use the reported super partition size as the upper limit, rather than |
| // sparse_file_len, which (1) can fail and (2) is kind of expensive, since |
| // it will map in all of the embedded fds. |
| std::vector<SparsePtr> files; |
| if (int limit = get_sparse_limit(super_size_)) { |
| files = resparse_file(sparse_layout_.get(), limit); |
| } else { |
| files.emplace_back(std::move(sparse_layout_)); |
| } |
| |
| // Send the data to the device. |
| flash_partition_files(super_name_, files); |
| } |
| |
| std::unique_ptr<FlashSuperLayoutTask> FlashSuperLayoutTask::Initialize( |
| const FlashingPlan* fp, std::vector<ImageEntry>& os_images) { |
| if (!supports_AB()) { |
| LOG(VERBOSE) << "Cannot optimize flashing super on non-AB device"; |
| return nullptr; |
| } |
| if (fp->slot_override == "all") { |
| LOG(VERBOSE) << "Cannot optimize flashing super for all slots"; |
| return nullptr; |
| } |
| |
| // Does this device use dynamic partitions at all? |
| unique_fd fd = fp->source->OpenFile("super_empty.img"); |
| |
| if (fd < 0) { |
| LOG(VERBOSE) << "could not open super_empty.img"; |
| return nullptr; |
| } |
| |
| std::string super_name; |
| // Try to find whether there is a super partition. |
| if (fp->fb->GetVar("super-partition-name", &super_name) != fastboot::SUCCESS) { |
| super_name = "super"; |
| } |
| |
| uint64_t partition_size; |
| std::string partition_size_str; |
| if (fp->fb->GetVar("partition-size:" + super_name, &partition_size_str) != fastboot::SUCCESS) { |
| LOG(VERBOSE) << "Cannot optimize super flashing: could not determine super partition"; |
| return nullptr; |
| } |
| partition_size_str = fb_fix_numeric_var(partition_size_str); |
| if (!android::base::ParseUint(partition_size_str, &partition_size)) { |
| LOG(VERBOSE) << "Could not parse " << super_name << " size: " << partition_size_str; |
| return nullptr; |
| } |
| |
| std::unique_ptr<SuperFlashHelper> helper = std::make_unique<SuperFlashHelper>(*fp->source); |
| if (!helper->Open(fd)) { |
| return nullptr; |
| } |
| |
| for (const auto& entry : os_images) { |
| auto partition = GetPartitionName(entry, fp->current_slot); |
| auto image = entry.first; |
| |
| if (!helper->AddPartition(partition, image->img_name, image->optional_if_no_image)) { |
| return nullptr; |
| } |
| } |
| |
| auto s = helper->GetSparseLayout(); |
| if (!s) return nullptr; |
| |
| // Remove images that we already flashed, just in case we have non-dynamic OS images. |
| auto remove_if_callback = [&](const ImageEntry& entry) -> bool { |
| return helper->WillFlash(GetPartitionName(entry, fp->current_slot)); |
| }; |
| os_images.erase(std::remove_if(os_images.begin(), os_images.end(), remove_if_callback), |
| os_images.end()); |
| return std::make_unique<FlashSuperLayoutTask>(super_name, std::move(helper), std::move(s), |
| partition_size); |
| } |
| |
| std::unique_ptr<FlashSuperLayoutTask> FlashSuperLayoutTask::InitializeFromTasks( |
| const FlashingPlan* fp, std::vector<std::unique_ptr<Task>>& tasks) { |
| if (!supports_AB()) { |
| LOG(VERBOSE) << "Cannot optimize flashing super on non-AB device"; |
| return nullptr; |
| } |
| if (fp->slot_override == "all") { |
| LOG(VERBOSE) << "Cannot optimize flashing super for all slots"; |
| return nullptr; |
| } |
| |
| // Does this device use dynamic partitions at all? |
| unique_fd fd = fp->source->OpenFile("super_empty.img"); |
| |
| if (fd < 0) { |
| LOG(VERBOSE) << "could not open super_empty.img"; |
| return nullptr; |
| } |
| |
| std::string super_name; |
| // Try to find whether there is a super partition. |
| if (fp->fb->GetVar("super-partition-name", &super_name) != fastboot::SUCCESS) { |
| super_name = "super"; |
| } |
| uint64_t partition_size; |
| std::string partition_size_str; |
| if (fp->fb->GetVar("partition-size:" + super_name, &partition_size_str) != fastboot::SUCCESS) { |
| LOG(VERBOSE) << "Cannot optimize super flashing: could not determine super partition"; |
| return nullptr; |
| } |
| partition_size_str = fb_fix_numeric_var(partition_size_str); |
| if (!android::base::ParseUint(partition_size_str, &partition_size)) { |
| LOG(VERBOSE) << "Could not parse " << super_name << " size: " << partition_size_str; |
| return nullptr; |
| } |
| |
| std::unique_ptr<SuperFlashHelper> helper = std::make_unique<SuperFlashHelper>(*fp->source); |
| if (!helper->Open(fd)) { |
| return nullptr; |
| } |
| |
| for (const auto& task : tasks) { |
| if (auto flash_task = task->AsFlashTask()) { |
| if (should_flash_in_userspace(flash_task->GetPartitionAndSlot())) { |
| auto partition = flash_task->GetPartitionAndSlot(); |
| if (!helper->AddPartition(partition, flash_task->GetImageName(), false)) { |
| return nullptr; |
| } |
| } |
| } |
| } |
| |
| auto s = helper->GetSparseLayout(); |
| if (!s) return nullptr; |
| // Remove images that we already flashed, just in case we have non-dynamic OS images. |
| auto remove_if_callback = [&](const auto& task) -> bool { |
| if (auto flash_task = task->AsFlashTask()) { |
| return helper->WillFlash(flash_task->GetPartitionAndSlot()); |
| } else if (auto update_super_task = task->AsUpdateSuperTask()) { |
| return true; |
| } else if (auto reboot_task = task->AsRebootTask()) { |
| return true; |
| } |
| return false; |
| }; |
| tasks.erase(std::remove_if(tasks.begin(), tasks.end(), remove_if_callback), tasks.end()); |
| |
| return std::make_unique<FlashSuperLayoutTask>(super_name, std::move(helper), std::move(s), |
| partition_size); |
| } |
| |
| UpdateSuperTask::UpdateSuperTask(const FlashingPlan* fp) : fp_(fp) {} |
| |
| void UpdateSuperTask::Run() { |
| unique_fd fd = fp_->source->OpenFile("super_empty.img"); |
| if (fd < 0) { |
| return; |
| } |
| if (!is_userspace_fastboot()) { |
| reboot_to_userspace_fastboot(); |
| } |
| |
| std::string super_name; |
| if (fp_->fb->GetVar("super-partition-name", &super_name) != fastboot::RetCode::SUCCESS) { |
| super_name = "super"; |
| } |
| fp_->fb->Download(super_name, fd, get_file_size(fd)); |
| |
| std::string command = "update-super:" + super_name; |
| if (fp_->wants_wipe) { |
| command += ":wipe"; |
| } |
| fp_->fb->RawCommand(command, "Updating super partition"); |
| } |
| |
| ResizeTask::ResizeTask(const FlashingPlan* fp, const std::string& pname, const std::string& size, |
| const std::string& slot) |
| : fp_(fp), pname_(pname), size_(size), slot_(slot) {} |
| |
| void ResizeTask::Run() { |
| auto resize_partition = [this](const std::string& partition) -> void { |
| if (is_logical(partition)) { |
| fp_->fb->ResizePartition(partition, size_); |
| } |
| }; |
| do_for_partitions(pname_, slot_, resize_partition, false); |
| } |
| |
| DeleteTask::DeleteTask(const FlashingPlan* fp, const std::string& pname) : fp_(fp), pname_(pname){}; |
| |
| void DeleteTask::Run() { |
| fp_->fb->DeletePartition(pname_); |
| } |
| |
| WipeTask::WipeTask(const FlashingPlan* fp, const std::string& pname) : fp_(fp), pname_(pname){}; |
| |
| void WipeTask::Run() { |
| std::string partition_type; |
| if (fp_->fb->GetVar("partition-type:" + pname_, &partition_type) != fastboot::SUCCESS) { |
| LOG(ERROR) << "wipe task partition not found: " << pname_; |
| return; |
| } |
| if (partition_type.empty()) return; |
| if (fp_->fb->Erase(pname_) != fastboot::SUCCESS) { |
| LOG(ERROR) << "wipe task erase failed with partition: " << pname_; |
| return; |
| } |
| fb_perform_format(pname_, 1, partition_type, "", fp_->fs_options); |
| } |