blob: d43fd80842ab98a18fa1159df8e2bc580d4afdfe [file] [log] [blame]
Daniel Zheng0d307182023-01-31 21:07:53 +00001//
2// Copyright (C) 2023 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
16#include "task.h"
Daniel Zheng7627b542023-04-13 13:43:17 -070017
Daniel Zheng47d70a52023-02-14 17:44:40 +000018#include <iostream>
Daniel Zheng7627b542023-04-13 13:43:17 -070019
20#include <android-base/logging.h>
21#include <android-base/parseint.h>
22
Daniel Zheng71b3b432023-01-30 17:43:00 +000023#include "fastboot.h"
Daniel Zheng47d70a52023-02-14 17:44:40 +000024#include "filesystem.h"
25#include "super_flash_helper.h"
Daniel Zheng0d307182023-01-31 21:07:53 +000026
Daniel Zheng47d70a52023-02-14 17:44:40 +000027using namespace std::string_literals;
Daniel Zheng403657d2023-03-10 07:37:25 +000028FlashTask::FlashTask(const std::string& _slot, const std::string& _pname, const std::string& _fname,
29 const bool apply_vbmeta)
30 : pname_(_pname), fname_(_fname), slot_(_slot), apply_vbmeta_(apply_vbmeta) {}
Daniel Zheng0d307182023-01-31 21:07:53 +000031
32void FlashTask::Run() {
33 auto flash = [&](const std::string& partition) {
Daniel Zhengbc01da52023-02-23 03:25:52 +000034 if (should_flash_in_userspace(partition) && !is_userspace_fastboot()) {
Daniel Zheng0d307182023-01-31 21:07:53 +000035 die("The partition you are trying to flash is dynamic, and "
36 "should be flashed via fastbootd. Please run:\n"
37 "\n"
38 " fastboot reboot fastboot\n"
39 "\n"
40 "And try again. If you are intentionally trying to "
41 "overwrite a fixed partition, use --force.");
42 }
Daniel Zheng403657d2023-03-10 07:37:25 +000043 do_flash(partition.c_str(), fname_.c_str(), apply_vbmeta_);
Daniel Zheng0d307182023-01-31 21:07:53 +000044 };
45 do_for_partitions(pname_, slot_, flash, true);
46}
Daniel Zheng71b3b432023-01-30 17:43:00 +000047
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -070048RebootTask::RebootTask(const FlashingPlan* fp) : fp_(fp){};
49RebootTask::RebootTask(const FlashingPlan* fp, const std::string& reboot_target)
Daniel Zheng43987c92023-03-07 18:20:53 +000050 : reboot_target_(reboot_target), fp_(fp){};
Daniel Zheng71b3b432023-01-30 17:43:00 +000051
52void RebootTask::Run() {
53 if ((reboot_target_ == "userspace" || reboot_target_ == "fastboot")) {
54 if (!is_userspace_fastboot()) {
55 reboot_to_userspace_fastboot();
Daniel Zhengbc01da52023-02-23 03:25:52 +000056 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000057 }
58 } else if (reboot_target_ == "recovery") {
Daniel Zhengbc01da52023-02-23 03:25:52 +000059 fp_->fb->RebootTo("recovery");
60 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000061 } else if (reboot_target_ == "bootloader") {
Daniel Zhengbc01da52023-02-23 03:25:52 +000062 fp_->fb->RebootTo("bootloader");
63 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000064 } else if (reboot_target_ == "") {
Daniel Zhengbc01da52023-02-23 03:25:52 +000065 fp_->fb->Reboot();
66 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000067 } else {
68 syntax_error("unknown reboot target %s", reboot_target_.c_str());
69 }
70}
Daniel Zheng47d70a52023-02-14 17:44:40 +000071
72FlashSuperLayoutTask::FlashSuperLayoutTask(const std::string& super_name,
73 std::unique_ptr<SuperFlashHelper> helper,
David Anderson74c78072023-03-16 21:21:28 -070074 SparsePtr sparse_layout, uint64_t super_size)
Daniel Zheng47d70a52023-02-14 17:44:40 +000075 : super_name_(super_name),
76 helper_(std::move(helper)),
David Anderson74c78072023-03-16 21:21:28 -070077 sparse_layout_(std::move(sparse_layout)),
78 super_size_(super_size) {}
Daniel Zheng47d70a52023-02-14 17:44:40 +000079
80void FlashSuperLayoutTask::Run() {
David Anderson74c78072023-03-16 21:21:28 -070081 // Use the reported super partition size as the upper limit, rather than
82 // sparse_file_len, which (1) can fail and (2) is kind of expensive, since
83 // it will map in all of the embedded fds.
Daniel Zheng47d70a52023-02-14 17:44:40 +000084 std::vector<SparsePtr> files;
David Anderson74c78072023-03-16 21:21:28 -070085 if (int limit = get_sparse_limit(super_size_)) {
Daniel Zheng47d70a52023-02-14 17:44:40 +000086 files = resparse_file(sparse_layout_.get(), limit);
87 } else {
88 files.emplace_back(std::move(sparse_layout_));
89 }
90
91 // Send the data to the device.
92 flash_partition_files(super_name_, files);
93}
94
95std::unique_ptr<FlashSuperLayoutTask> FlashSuperLayoutTask::Initialize(
96 FlashingPlan* fp, std::vector<ImageEntry>& os_images) {
97 if (!supports_AB()) {
98 LOG(VERBOSE) << "Cannot optimize flashing super on non-AB device";
99 return nullptr;
100 }
Daniel Zheng1e6456b2023-04-10 09:35:48 -0700101 if (fp->slot_override == "all") {
Daniel Zheng47d70a52023-02-14 17:44:40 +0000102 LOG(VERBOSE) << "Cannot optimize flashing super for all slots";
103 return nullptr;
104 }
105
106 // Does this device use dynamic partitions at all?
107 unique_fd fd = fp->source->OpenFile("super_empty.img");
108
109 if (fd < 0) {
110 LOG(VERBOSE) << "could not open super_empty.img";
111 return nullptr;
112 }
113
114 std::string super_name;
115 // Try to find whether there is a super partition.
116 if (fp->fb->GetVar("super-partition-name", &super_name) != fastboot::SUCCESS) {
117 super_name = "super";
118 }
Daniel Zheng47d70a52023-02-14 17:44:40 +0000119
David Anderson74c78072023-03-16 21:21:28 -0700120 uint64_t partition_size;
121 std::string partition_size_str;
Daniel Zheng47d70a52023-02-14 17:44:40 +0000122 if (fp->fb->GetVar("partition-size:" + super_name, &partition_size_str) != fastboot::SUCCESS) {
123 LOG(VERBOSE) << "Cannot optimize super flashing: could not determine super partition";
124 return nullptr;
125 }
David Anderson74c78072023-03-16 21:21:28 -0700126 partition_size_str = fb_fix_numeric_var(partition_size_str);
127 if (!android::base::ParseUint(partition_size_str, &partition_size)) {
128 LOG(VERBOSE) << "Could not parse " << super_name << " size: " << partition_size_str;
129 return nullptr;
130 }
131
Daniel Zheng47d70a52023-02-14 17:44:40 +0000132 std::unique_ptr<SuperFlashHelper> helper = std::make_unique<SuperFlashHelper>(*fp->source);
133 if (!helper->Open(fd)) {
134 return nullptr;
135 }
136
137 for (const auto& entry : os_images) {
Kelvin Zhang2a4a45f2023-04-03 20:44:05 +0000138 auto partition = GetPartitionName(entry, fp->current_slot);
Daniel Zheng47d70a52023-02-14 17:44:40 +0000139 auto image = entry.first;
140
141 if (!helper->AddPartition(partition, image->img_name, image->optional_if_no_image)) {
142 return nullptr;
143 }
144 }
145
146 auto s = helper->GetSparseLayout();
147 if (!s) return nullptr;
148
149 // Remove images that we already flashed, just in case we have non-dynamic OS images.
150 auto remove_if_callback = [&](const ImageEntry& entry) -> bool {
Kelvin Zhang2a4a45f2023-04-03 20:44:05 +0000151 return helper->WillFlash(GetPartitionName(entry, fp->current_slot));
Daniel Zheng47d70a52023-02-14 17:44:40 +0000152 };
153 os_images.erase(std::remove_if(os_images.begin(), os_images.end(), remove_if_callback),
154 os_images.end());
David Anderson74c78072023-03-16 21:21:28 -0700155 return std::make_unique<FlashSuperLayoutTask>(super_name, std::move(helper), std::move(s),
156 partition_size);
Daniel Zheng47d70a52023-02-14 17:44:40 +0000157}
Daniel Zheng6bb8baa2023-03-03 07:14:23 +0000158
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700159UpdateSuperTask::UpdateSuperTask(const FlashingPlan* fp) : fp_(fp) {}
Daniel Zheng6bb8baa2023-03-03 07:14:23 +0000160
161void UpdateSuperTask::Run() {
162 unique_fd fd = fp_->source->OpenFile("super_empty.img");
163 if (fd < 0) {
164 return;
165 }
166 if (!is_userspace_fastboot()) {
167 reboot_to_userspace_fastboot();
168 }
169
170 std::string super_name;
171 if (fp_->fb->GetVar("super-partition-name", &super_name) != fastboot::RetCode::SUCCESS) {
172 super_name = "super";
173 }
174 fp_->fb->Download(super_name, fd, get_file_size(fd));
175
176 std::string command = "update-super:" + super_name;
177 if (fp_->wants_wipe) {
178 command += ":wipe";
179 }
180 fp_->fb->RawCommand(command, "Updating super partition");
181}
Daniel Zheng9f7bf7e2023-03-03 07:16:46 +0000182
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700183ResizeTask::ResizeTask(const FlashingPlan* fp, const std::string& pname, const std::string& size,
Daniel Zheng9f7bf7e2023-03-03 07:16:46 +0000184 const std::string& slot)
185 : fp_(fp), pname_(pname), size_(size), slot_(slot) {}
186
187void ResizeTask::Run() {
188 auto resize_partition = [this](const std::string& partition) -> void {
189 if (is_logical(partition)) {
190 fp_->fb->ResizePartition(partition, size_);
191 }
192 };
193 do_for_partitions(pname_, slot_, resize_partition, false);
194}
Daniel Zhengaa70f4c2023-03-07 18:59:51 +0000195
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700196DeleteTask::DeleteTask(const FlashingPlan* fp, const std::string& pname) : fp_(fp), pname_(pname){};
Daniel Zhengaa70f4c2023-03-07 18:59:51 +0000197
198void DeleteTask::Run() {
199 fp_->fb->DeletePartition(pname_);
Daniel Zheng15e77832023-03-13 17:09:43 +0000200}
201
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700202WipeTask::WipeTask(const FlashingPlan* fp, const std::string& pname) : fp_(fp), pname_(pname){};
Daniel Zheng15e77832023-03-13 17:09:43 +0000203
204void WipeTask::Run() {
205 std::string partition_type;
206 if (fp_->fb->GetVar("partition-type:" + pname_, &partition_type) != fastboot::SUCCESS) {
Daniel Zheng6f213b22023-03-28 22:47:04 +0000207 LOG(ERROR) << "wipe task partition not found: " << pname_;
Daniel Zheng15e77832023-03-13 17:09:43 +0000208 return;
209 }
210 if (partition_type.empty()) return;
Daniel Zheng6f213b22023-03-28 22:47:04 +0000211 if (fp_->fb->Erase(pname_) != fastboot::SUCCESS) {
212 LOG(ERROR) << "wipe task erase failed with partition: " << pname_;
213 return;
214 }
Daniel Zheng15e77832023-03-13 17:09:43 +0000215 fb_perform_format(pname_, 1, partition_type, "", fp_->fs_options);
216}