blob: 3b20fc21a5bf6ee2888199c2e692d76efcc7e302 [file] [log] [blame]
Alex Deymob17327c2015-09-04 10:29:00 -07001//
2// Copyright (C) 2015 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
Amin Hassaniec7bc112020-10-29 16:47:58 -070017#include "update_engine/aosp/boot_control_android.h"
Alex Deymob17327c2015-09-04 10:29:00 -070018
Sen Jiangd944faa2018-08-22 18:46:39 -070019#include <memory>
20#include <utility>
Yifan Hongd4db07e2018-10-18 17:46:27 -070021#include <vector>
Sen Jiangd944faa2018-08-22 18:46:39 -070022
Alex Deymoaa26f622015-09-16 18:21:27 -070023#include <base/bind.h>
Alex Deymoaa26f622015-09-16 18:21:27 -070024#include <base/logging.h>
Sen Jiangd944faa2018-08-22 18:46:39 -070025#include <bootloader_message/bootloader_message.h>
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070026#include <brillo/message_loops/message_loop.h>
Alex Deymob17327c2015-09-04 10:29:00 -070027
Amin Hassaniec7bc112020-10-29 16:47:58 -070028#include "update_engine/aosp/dynamic_partition_control_android.h"
Alex Deymo39910dc2015-11-09 17:04:30 -080029#include "update_engine/common/utils.h"
Alex Deymob17327c2015-09-04 10:29:00 -070030
31using std::string;
32
Connor O'Briencee6ad92016-11-21 13:53:52 -080033using android::hardware::Return;
34using android::hardware::boot::V1_0::BoolResult;
35using android::hardware::boot::V1_0::CommandResult;
36using android::hardware::boot::V1_0::IBootControl;
Yifan Hong537802d2018-08-15 13:15:42 -070037using Slot = chromeos_update_engine::BootControlInterface::Slot;
Connor O'Briencee6ad92016-11-21 13:53:52 -080038
39namespace {
Yifan Hong537802d2018-08-15 13:15:42 -070040
Connor O'Briencee6ad92016-11-21 13:53:52 -080041auto StoreResultCallback(CommandResult* dest) {
42 return [dest](const CommandResult& result) { *dest = result; };
43}
44} // namespace
Alex Deymo44348e02016-07-29 16:22:26 -070045
Alex Deymob17327c2015-09-04 10:29:00 -070046namespace chromeos_update_engine {
47
48namespace boot_control {
49
50// Factory defined in boot_control.h.
51std::unique_ptr<BootControlInterface> CreateBootControl() {
Yifan Hong537802d2018-08-15 13:15:42 -070052 auto boot_control = std::make_unique<BootControlAndroid>();
David Zeuthen753fadc2015-09-15 16:34:09 -040053 if (!boot_control->Init()) {
54 return nullptr;
55 }
Alex Vakulenkoce8c8ee2016-04-08 08:59:26 -070056 return std::move(boot_control);
Alex Deymob17327c2015-09-04 10:29:00 -070057}
58
59} // namespace boot_control
60
David Zeuthen753fadc2015-09-15 16:34:09 -040061bool BootControlAndroid::Init() {
Chris Phoenixafde8e82017-01-17 23:14:58 -080062 module_ = IBootControl::getService();
Connor O'Briencee6ad92016-11-21 13:53:52 -080063 if (module_ == nullptr) {
Steven Moreland927e00d2017-01-04 12:58:40 -080064 LOG(ERROR) << "Error getting bootctrl HIDL module.";
David Zeuthen753fadc2015-09-15 16:34:09 -040065 return false;
66 }
67
Steven Moreland927e00d2017-01-04 12:58:40 -080068 LOG(INFO) << "Loaded boot control hidl hal.";
David Zeuthen753fadc2015-09-15 16:34:09 -040069
Yifan Hong537802d2018-08-15 13:15:42 -070070 dynamic_control_ = std::make_unique<DynamicPartitionControlAndroid>();
71
David Zeuthen753fadc2015-09-15 16:34:09 -040072 return true;
73}
Alex Deymob17327c2015-09-04 10:29:00 -070074
75unsigned int BootControlAndroid::GetNumSlots() const {
Connor O'Briencee6ad92016-11-21 13:53:52 -080076 return module_->getNumberSlots();
Alex Deymob17327c2015-09-04 10:29:00 -070077}
78
79BootControlInterface::Slot BootControlAndroid::GetCurrentSlot() const {
Connor O'Briencee6ad92016-11-21 13:53:52 -080080 return module_->getCurrentSlot();
Alex Deymob17327c2015-09-04 10:29:00 -070081}
82
Tianjie51a5a392020-06-03 14:39:32 -070083bool BootControlAndroid::GetPartitionDevice(const std::string& partition_name,
84 BootControlInterface::Slot slot,
85 bool not_in_payload,
86 std::string* device,
87 bool* is_dynamic) const {
88 return dynamic_control_->GetPartitionDevice(partition_name,
89 slot,
90 GetCurrentSlot(),
91 not_in_payload,
92 device,
93 is_dynamic);
94}
Yifan Hongae04e192018-10-29 11:00:28 -070095
Yifan Hong537802d2018-08-15 13:15:42 -070096bool BootControlAndroid::GetPartitionDevice(const string& partition_name,
Tianjie51a5a392020-06-03 14:39:32 -070097 BootControlInterface::Slot slot,
Yifan Hong537802d2018-08-15 13:15:42 -070098 string* device) const {
Tianjie51a5a392020-06-03 14:39:32 -070099 return GetPartitionDevice(
100 partition_name, slot, false /* not_in_payload */, device, nullptr);
Alex Deymob17327c2015-09-04 10:29:00 -0700101}
102
103bool BootControlAndroid::IsSlotBootable(Slot slot) const {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800104 Return<BoolResult> ret = module_->isSlotBootable(slot);
Yifan Hong7b514b42016-12-21 13:02:00 -0800105 if (!ret.isOk()) {
Alex Deymo31d95ac2015-09-17 11:56:18 -0700106 LOG(ERROR) << "Unable to determine if slot " << SlotName(slot)
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800107 << " is bootable: " << ret.description();
David Zeuthen753fadc2015-09-15 16:34:09 -0400108 return false;
109 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800110 if (ret == BoolResult::INVALID_SLOT) {
111 LOG(ERROR) << "Invalid slot: " << SlotName(slot);
112 return false;
113 }
114 return ret == BoolResult::TRUE;
Alex Deymob17327c2015-09-04 10:29:00 -0700115}
116
117bool BootControlAndroid::MarkSlotUnbootable(Slot slot) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800118 CommandResult result;
119 auto ret = module_->setSlotAsUnbootable(slot, StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800120 if (!ret.isOk()) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800121 LOG(ERROR) << "Unable to call MarkSlotUnbootable for slot "
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800122 << SlotName(slot) << ": " << ret.description();
David Zeuthen753fadc2015-09-15 16:34:09 -0400123 return false;
124 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800125 if (!result.success) {
126 LOG(ERROR) << "Unable to mark slot " << SlotName(slot)
127 << " as unbootable: " << result.errMsg.c_str();
128 }
129 return result.success;
Alex Deymob17327c2015-09-04 10:29:00 -0700130}
131
Alex Deymo31d95ac2015-09-17 11:56:18 -0700132bool BootControlAndroid::SetActiveBootSlot(Slot slot) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800133 CommandResult result;
134 auto ret = module_->setActiveBootSlot(slot, StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800135 if (!ret.isOk()) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800136 LOG(ERROR) << "Unable to call SetActiveBootSlot for slot " << SlotName(slot)
Yifan Hong7b514b42016-12-21 13:02:00 -0800137 << ": " << ret.description();
Connor O'Briencee6ad92016-11-21 13:53:52 -0800138 return false;
Alex Deymo29dcbf32016-10-06 13:33:20 -0700139 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800140 if (!result.success) {
141 LOG(ERROR) << "Unable to set the active slot to slot " << SlotName(slot)
142 << ": " << result.errMsg.c_str();
143 }
144 return result.success;
Alex Deymo31d95ac2015-09-17 11:56:18 -0700145}
146
Alex Deymoaa26f622015-09-16 18:21:27 -0700147bool BootControlAndroid::MarkBootSuccessfulAsync(
148 base::Callback<void(bool)> callback) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800149 CommandResult result;
150 auto ret = module_->markBootSuccessful(StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800151 if (!ret.isOk()) {
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800152 LOG(ERROR) << "Unable to call MarkBootSuccessful: " << ret.description();
Connor O'Briencee6ad92016-11-21 13:53:52 -0800153 return false;
154 }
155 if (!result.success) {
156 LOG(ERROR) << "Unable to mark boot successful: " << result.errMsg.c_str();
Alex Deymoaa26f622015-09-16 18:21:27 -0700157 }
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700158 return brillo::MessageLoop::current()->PostTask(
Connor O'Briencee6ad92016-11-21 13:53:52 -0800159 FROM_HERE, base::Bind(callback, result.success)) !=
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700160 brillo::MessageLoop::kTaskIdNull;
Alex Deymoaa26f622015-09-16 18:21:27 -0700161}
162
Yifan Hongf1415942020-02-24 18:34:49 -0800163bool BootControlAndroid::IsSlotMarkedSuccessful(
164 BootControlInterface::Slot slot) const {
165 Return<BoolResult> ret = module_->isSlotMarkedSuccessful(slot);
166 CommandResult result;
167 if (!ret.isOk()) {
168 LOG(ERROR) << "Unable to determine if slot " << SlotName(slot)
169 << " is marked successful: " << ret.description();
170 return false;
171 }
172 if (ret == BoolResult::INVALID_SLOT) {
173 LOG(ERROR) << "Invalid slot: " << SlotName(slot);
174 return false;
175 }
176 return ret == BoolResult::TRUE;
177}
178
Yifan Hongdaac7322019-11-07 10:48:26 -0800179DynamicPartitionControlInterface*
180BootControlAndroid::GetDynamicPartitionControl() {
181 return dynamic_control_.get();
182}
183
Kelvin Zhang91d95fa2020-11-05 13:52:00 -0500184std::optional<PartitionDevice> BootControlAndroid::GetPartitionDevice(
185 const std::string& partition_name,
186 uint32_t slot,
187 uint32_t current_slot,
188 bool not_in_payload) const {
189 return dynamic_control_->GetPartitionDevice(
190 partition_name, slot, current_slot, not_in_payload);
191}
Alex Deymob17327c2015-09-04 10:29:00 -0700192} // namespace chromeos_update_engine