blob: bf64f0e1e42c66b707063f71412d9f562d3096ff [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 Zheng29a211c2023-03-10 07:23:49 +000026#include "util.h"
Daniel Zheng0d307182023-01-31 21:07:53 +000027
Daniel Zheng47d70a52023-02-14 17:44:40 +000028using namespace std::string_literals;
Daniel Zheng5a9905a2023-05-23 10:51:01 -070029FlashTask::FlashTask(const std::string& slot, const std::string& pname, const std::string& fname,
Daniel Zheng791a83b2023-05-23 10:42:39 -070030 const bool apply_vbmeta, const FlashingPlan* fp)
31 : pname_(pname), fname_(fname), slot_(slot), apply_vbmeta_(apply_vbmeta), fp_(fp) {}
Daniel Zheng0d307182023-01-31 21:07:53 +000032
33void FlashTask::Run() {
34 auto flash = [&](const std::string& partition) {
Daniel Zhengbc01da52023-02-23 03:25:52 +000035 if (should_flash_in_userspace(partition) && !is_userspace_fastboot()) {
Daniel Zheng0d307182023-01-31 21:07:53 +000036 die("The partition you are trying to flash is dynamic, and "
37 "should be flashed via fastbootd. Please run:\n"
38 "\n"
39 " fastboot reboot fastboot\n"
40 "\n"
41 "And try again. If you are intentionally trying to "
42 "overwrite a fixed partition, use --force.");
43 }
Daniel Zheng791a83b2023-05-23 10:42:39 -070044 do_flash(partition.c_str(), fname_.c_str(), apply_vbmeta_, fp_);
Daniel Zheng0d307182023-01-31 21:07:53 +000045 };
46 do_for_partitions(pname_, slot_, flash, true);
47}
Daniel Zheng71b3b432023-01-30 17:43:00 +000048
Daniel Zheng665a4602023-06-05 11:24:59 -070049std::string FlashTask::ToString() {
50 std::string apply_vbmeta_string = "";
51 if (apply_vbmeta_) {
52 apply_vbmeta_string = " --apply_vbmeta";
53 }
54 return "flash" + apply_vbmeta_string + " " + pname_ + " " + fname_;
55}
56
Daniel Zheng29a211c2023-03-10 07:23:49 +000057std::string FlashTask::GetPartitionAndSlot() {
58 auto slot = slot_;
59 if (slot.empty()) {
60 slot = get_current_slot();
61 }
62 if (slot.empty()) {
63 return pname_;
64 }
65 if (slot == "all") {
66 LOG(FATAL) << "Cannot retrieve a singular name when using all slots";
67 }
68 return pname_ + "_" + slot;
69}
70
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -070071RebootTask::RebootTask(const FlashingPlan* fp) : fp_(fp){};
72RebootTask::RebootTask(const FlashingPlan* fp, const std::string& reboot_target)
Daniel Zheng43987c92023-03-07 18:20:53 +000073 : reboot_target_(reboot_target), fp_(fp){};
Daniel Zheng71b3b432023-01-30 17:43:00 +000074
75void RebootTask::Run() {
Daniel Zheng77af6af2023-04-20 14:30:28 -070076 if (reboot_target_ == "fastboot") {
Daniel Zheng71b3b432023-01-30 17:43:00 +000077 if (!is_userspace_fastboot()) {
78 reboot_to_userspace_fastboot();
Daniel Zhengbc01da52023-02-23 03:25:52 +000079 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000080 }
81 } else if (reboot_target_ == "recovery") {
Daniel Zhengbc01da52023-02-23 03:25:52 +000082 fp_->fb->RebootTo("recovery");
83 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000084 } else if (reboot_target_ == "bootloader") {
Daniel Zhengbc01da52023-02-23 03:25:52 +000085 fp_->fb->RebootTo("bootloader");
86 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000087 } else if (reboot_target_ == "") {
Daniel Zhengbc01da52023-02-23 03:25:52 +000088 fp_->fb->Reboot();
89 fp_->fb->WaitForDisconnect();
Daniel Zheng71b3b432023-01-30 17:43:00 +000090 } else {
91 syntax_error("unknown reboot target %s", reboot_target_.c_str());
92 }
93}
Daniel Zheng47d70a52023-02-14 17:44:40 +000094
Daniel Zheng665a4602023-06-05 11:24:59 -070095std::string RebootTask::ToString() {
96 return "reboot " + reboot_target_;
97}
98
Daniel Zheng3e048572023-06-21 15:21:06 -070099OptimizedFlashSuperTask::OptimizedFlashSuperTask(const std::string& super_name,
100 std::unique_ptr<SuperFlashHelper> helper,
101 SparsePtr sparse_layout, uint64_t super_size,
102 const FlashingPlan* fp)
Daniel Zheng47d70a52023-02-14 17:44:40 +0000103 : super_name_(super_name),
104 helper_(std::move(helper)),
David Anderson74c78072023-03-16 21:21:28 -0700105 sparse_layout_(std::move(sparse_layout)),
Daniel Zheng5769b262023-06-21 14:41:11 -0700106 super_size_(super_size),
107 fp_(fp) {}
Daniel Zheng47d70a52023-02-14 17:44:40 +0000108
Daniel Zheng3e048572023-06-21 15:21:06 -0700109void OptimizedFlashSuperTask::Run() {
David Anderson74c78072023-03-16 21:21:28 -0700110 // Use the reported super partition size as the upper limit, rather than
111 // sparse_file_len, which (1) can fail and (2) is kind of expensive, since
112 // it will map in all of the embedded fds.
Daniel Zheng47d70a52023-02-14 17:44:40 +0000113 std::vector<SparsePtr> files;
Daniel Zheng5769b262023-06-21 14:41:11 -0700114 if (int limit = get_sparse_limit(super_size_, fp_)) {
Daniel Zheng47d70a52023-02-14 17:44:40 +0000115 files = resparse_file(sparse_layout_.get(), limit);
116 } else {
117 files.emplace_back(std::move(sparse_layout_));
118 }
119
120 // Send the data to the device.
121 flash_partition_files(super_name_, files);
122}
Daniel Zheng3e048572023-06-21 15:21:06 -0700123std::string OptimizedFlashSuperTask::ToString() {
Daniel Zheng665a4602023-06-05 11:24:59 -0700124 return "optimized-flash-super";
125}
Daniel Zheng47d70a52023-02-14 17:44:40 +0000126
Daniel Zheng3e048572023-06-21 15:21:06 -0700127std::unique_ptr<OptimizedFlashSuperTask> OptimizedFlashSuperTask::Initialize(
Daniel Zheng29a211c2023-03-10 07:23:49 +0000128 const FlashingPlan* fp, std::vector<ImageEntry>& os_images) {
Daniel Zhengd62002c2023-06-21 14:27:35 -0700129 if (!fp->should_optimize_flash_super) {
130 LOG(INFO) << "super optimization is disabled";
131 return nullptr;
132 }
Daniel Zheng47d70a52023-02-14 17:44:40 +0000133 if (!supports_AB()) {
134 LOG(VERBOSE) << "Cannot optimize flashing super on non-AB device";
135 return nullptr;
136 }
Daniel Zheng1e6456b2023-04-10 09:35:48 -0700137 if (fp->slot_override == "all") {
Daniel Zheng47d70a52023-02-14 17:44:40 +0000138 LOG(VERBOSE) << "Cannot optimize flashing super for all slots";
139 return nullptr;
140 }
141
142 // Does this device use dynamic partitions at all?
143 unique_fd fd = fp->source->OpenFile("super_empty.img");
144
145 if (fd < 0) {
146 LOG(VERBOSE) << "could not open super_empty.img";
147 return nullptr;
148 }
149
150 std::string super_name;
151 // Try to find whether there is a super partition.
152 if (fp->fb->GetVar("super-partition-name", &super_name) != fastboot::SUCCESS) {
153 super_name = "super";
154 }
Daniel Zheng47d70a52023-02-14 17:44:40 +0000155
David Anderson74c78072023-03-16 21:21:28 -0700156 uint64_t partition_size;
157 std::string partition_size_str;
Daniel Zheng47d70a52023-02-14 17:44:40 +0000158 if (fp->fb->GetVar("partition-size:" + super_name, &partition_size_str) != fastboot::SUCCESS) {
159 LOG(VERBOSE) << "Cannot optimize super flashing: could not determine super partition";
160 return nullptr;
161 }
David Anderson74c78072023-03-16 21:21:28 -0700162 partition_size_str = fb_fix_numeric_var(partition_size_str);
163 if (!android::base::ParseUint(partition_size_str, &partition_size)) {
164 LOG(VERBOSE) << "Could not parse " << super_name << " size: " << partition_size_str;
165 return nullptr;
166 }
167
Daniel Zheng47d70a52023-02-14 17:44:40 +0000168 std::unique_ptr<SuperFlashHelper> helper = std::make_unique<SuperFlashHelper>(*fp->source);
169 if (!helper->Open(fd)) {
170 return nullptr;
171 }
172
173 for (const auto& entry : os_images) {
Kelvin Zhang2a4a45f2023-04-03 20:44:05 +0000174 auto partition = GetPartitionName(entry, fp->current_slot);
Daniel Zheng47d70a52023-02-14 17:44:40 +0000175 auto image = entry.first;
176
177 if (!helper->AddPartition(partition, image->img_name, image->optional_if_no_image)) {
178 return nullptr;
179 }
180 }
181
182 auto s = helper->GetSparseLayout();
183 if (!s) return nullptr;
184
185 // Remove images that we already flashed, just in case we have non-dynamic OS images.
186 auto remove_if_callback = [&](const ImageEntry& entry) -> bool {
Kelvin Zhang2a4a45f2023-04-03 20:44:05 +0000187 return helper->WillFlash(GetPartitionName(entry, fp->current_slot));
Daniel Zheng47d70a52023-02-14 17:44:40 +0000188 };
189 os_images.erase(std::remove_if(os_images.begin(), os_images.end(), remove_if_callback),
190 os_images.end());
Daniel Zheng3e048572023-06-21 15:21:06 -0700191 return std::make_unique<OptimizedFlashSuperTask>(super_name, std::move(helper), std::move(s),
192 partition_size, fp);
Daniel Zheng47d70a52023-02-14 17:44:40 +0000193}
Daniel Zheng6bb8baa2023-03-03 07:14:23 +0000194
Daniel Zheng3e048572023-06-21 15:21:06 -0700195std::unique_ptr<OptimizedFlashSuperTask> OptimizedFlashSuperTask::InitializeFromTasks(
Daniel Zheng29a211c2023-03-10 07:23:49 +0000196 const FlashingPlan* fp, std::vector<std::unique_ptr<Task>>& tasks) {
Daniel Zhengd62002c2023-06-21 14:27:35 -0700197 if (!fp->should_optimize_flash_super) {
198 LOG(INFO) << "super optimization is disabled";
199 return nullptr;
200 }
Daniel Zheng29a211c2023-03-10 07:23:49 +0000201 if (!supports_AB()) {
202 LOG(VERBOSE) << "Cannot optimize flashing super on non-AB device";
203 return nullptr;
204 }
205 if (fp->slot_override == "all") {
206 LOG(VERBOSE) << "Cannot optimize flashing super for all slots";
207 return nullptr;
208 }
209
210 // Does this device use dynamic partitions at all?
211 unique_fd fd = fp->source->OpenFile("super_empty.img");
212
213 if (fd < 0) {
214 LOG(VERBOSE) << "could not open super_empty.img";
215 return nullptr;
216 }
217
218 std::string super_name;
219 // Try to find whether there is a super partition.
220 if (fp->fb->GetVar("super-partition-name", &super_name) != fastboot::SUCCESS) {
221 super_name = "super";
222 }
223 uint64_t partition_size;
224 std::string partition_size_str;
225 if (fp->fb->GetVar("partition-size:" + super_name, &partition_size_str) != fastboot::SUCCESS) {
226 LOG(VERBOSE) << "Cannot optimize super flashing: could not determine super partition";
227 return nullptr;
228 }
229 partition_size_str = fb_fix_numeric_var(partition_size_str);
230 if (!android::base::ParseUint(partition_size_str, &partition_size)) {
231 LOG(VERBOSE) << "Could not parse " << super_name << " size: " << partition_size_str;
232 return nullptr;
233 }
234
235 std::unique_ptr<SuperFlashHelper> helper = std::make_unique<SuperFlashHelper>(*fp->source);
236 if (!helper->Open(fd)) {
237 return nullptr;
238 }
239
240 for (const auto& task : tasks) {
241 if (auto flash_task = task->AsFlashTask()) {
Daniel Zheng2775df62023-05-25 15:14:59 -0700242 auto partition = flash_task->GetPartitionAndSlot();
243 if (!helper->AddPartition(partition, flash_task->GetImageName(), false)) {
244 return nullptr;
Daniel Zheng29a211c2023-03-10 07:23:49 +0000245 }
246 }
247 }
248
249 auto s = helper->GetSparseLayout();
250 if (!s) return nullptr;
251 // Remove images that we already flashed, just in case we have non-dynamic OS images.
252 auto remove_if_callback = [&](const auto& task) -> bool {
253 if (auto flash_task = task->AsFlashTask()) {
254 return helper->WillFlash(flash_task->GetPartitionAndSlot());
255 } else if (auto update_super_task = task->AsUpdateSuperTask()) {
256 return true;
257 } else if (auto reboot_task = task->AsRebootTask()) {
258 return true;
259 }
260 return false;
261 };
262 tasks.erase(std::remove_if(tasks.begin(), tasks.end(), remove_if_callback), tasks.end());
263
Daniel Zheng3e048572023-06-21 15:21:06 -0700264 return std::make_unique<OptimizedFlashSuperTask>(super_name, std::move(helper), std::move(s),
265 partition_size, fp);
Daniel Zheng29a211c2023-03-10 07:23:49 +0000266}
267
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700268UpdateSuperTask::UpdateSuperTask(const FlashingPlan* fp) : fp_(fp) {}
Daniel Zheng6bb8baa2023-03-03 07:14:23 +0000269
270void UpdateSuperTask::Run() {
271 unique_fd fd = fp_->source->OpenFile("super_empty.img");
272 if (fd < 0) {
273 return;
274 }
275 if (!is_userspace_fastboot()) {
276 reboot_to_userspace_fastboot();
277 }
278
279 std::string super_name;
280 if (fp_->fb->GetVar("super-partition-name", &super_name) != fastboot::RetCode::SUCCESS) {
281 super_name = "super";
282 }
283 fp_->fb->Download(super_name, fd, get_file_size(fd));
284
285 std::string command = "update-super:" + super_name;
286 if (fp_->wants_wipe) {
287 command += ":wipe";
288 }
289 fp_->fb->RawCommand(command, "Updating super partition");
290}
Daniel Zheng665a4602023-06-05 11:24:59 -0700291std::string UpdateSuperTask::ToString() {
292 return "update-super";
293}
Daniel Zheng9f7bf7e2023-03-03 07:16:46 +0000294
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700295ResizeTask::ResizeTask(const FlashingPlan* fp, const std::string& pname, const std::string& size,
Daniel Zheng9f7bf7e2023-03-03 07:16:46 +0000296 const std::string& slot)
297 : fp_(fp), pname_(pname), size_(size), slot_(slot) {}
298
299void ResizeTask::Run() {
300 auto resize_partition = [this](const std::string& partition) -> void {
301 if (is_logical(partition)) {
302 fp_->fb->ResizePartition(partition, size_);
303 }
304 };
305 do_for_partitions(pname_, slot_, resize_partition, false);
306}
Daniel Zhengaa70f4c2023-03-07 18:59:51 +0000307
Daniel Zheng665a4602023-06-05 11:24:59 -0700308std::string ResizeTask::ToString() {
309 return "resize " + pname_;
310}
311
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700312DeleteTask::DeleteTask(const FlashingPlan* fp, const std::string& pname) : fp_(fp), pname_(pname){};
Daniel Zhengaa70f4c2023-03-07 18:59:51 +0000313
314void DeleteTask::Run() {
315 fp_->fb->DeletePartition(pname_);
Daniel Zheng15e77832023-03-13 17:09:43 +0000316}
317
Daniel Zheng665a4602023-06-05 11:24:59 -0700318std::string DeleteTask::ToString() {
319 return "delete " + pname_;
320}
321
Daniel Zhenge6dbd4f2023-04-10 09:53:08 -0700322WipeTask::WipeTask(const FlashingPlan* fp, const std::string& pname) : fp_(fp), pname_(pname){};
Daniel Zheng15e77832023-03-13 17:09:43 +0000323
324void WipeTask::Run() {
325 std::string partition_type;
326 if (fp_->fb->GetVar("partition-type:" + pname_, &partition_type) != fastboot::SUCCESS) {
Daniel Zheng6f213b22023-03-28 22:47:04 +0000327 LOG(ERROR) << "wipe task partition not found: " << pname_;
Daniel Zheng15e77832023-03-13 17:09:43 +0000328 return;
329 }
330 if (partition_type.empty()) return;
Daniel Zheng6f213b22023-03-28 22:47:04 +0000331 if (fp_->fb->Erase(pname_) != fastboot::SUCCESS) {
332 LOG(ERROR) << "wipe task erase failed with partition: " << pname_;
333 return;
334 }
Daniel Zheng5769b262023-06-21 14:41:11 -0700335 fb_perform_format(pname_, 1, partition_type, "", fp_->fs_options, fp_);
Daniel Zheng15e77832023-03-13 17:09:43 +0000336}
Daniel Zheng665a4602023-06-05 11:24:59 -0700337
338std::string WipeTask::ToString() {
339 return "erase " + pname_;
340}