blob: dee5fa84ea690f091eebdd659e4dbd860cdbd775 [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
Alex Deymo1b03f9f2015-12-09 00:38:36 -080017#include "update_engine/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
Alex Deymo39910dc2015-11-09 17:04:30 -080028#include "update_engine/common/utils.h"
Yifan Hong537802d2018-08-15 13:15:42 -070029#include "update_engine/dynamic_partition_control_android.h"
Alex Deymob17327c2015-09-04 10:29:00 -070030
31using std::string;
32
Yifan Hong537802d2018-08-15 13:15:42 -070033using android::dm::DmDeviceState;
Yifan Hong537802d2018-08-15 13:15:42 -070034using android::hardware::hidl_string;
Connor O'Briencee6ad92016-11-21 13:53:52 -080035using android::hardware::Return;
36using android::hardware::boot::V1_0::BoolResult;
37using android::hardware::boot::V1_0::CommandResult;
38using android::hardware::boot::V1_0::IBootControl;
Yifan Hong537802d2018-08-15 13:15:42 -070039using Slot = chromeos_update_engine::BootControlInterface::Slot;
Connor O'Briencee6ad92016-11-21 13:53:52 -080040
41namespace {
Yifan Hong537802d2018-08-15 13:15:42 -070042
Connor O'Briencee6ad92016-11-21 13:53:52 -080043auto StoreResultCallback(CommandResult* dest) {
44 return [dest](const CommandResult& result) { *dest = result; };
45}
46} // namespace
Alex Deymo44348e02016-07-29 16:22:26 -070047
Alex Deymob17327c2015-09-04 10:29:00 -070048namespace chromeos_update_engine {
49
50namespace boot_control {
51
52// Factory defined in boot_control.h.
53std::unique_ptr<BootControlInterface> CreateBootControl() {
Yifan Hong537802d2018-08-15 13:15:42 -070054 auto boot_control = std::make_unique<BootControlAndroid>();
David Zeuthen753fadc2015-09-15 16:34:09 -040055 if (!boot_control->Init()) {
56 return nullptr;
57 }
Alex Vakulenkoce8c8ee2016-04-08 08:59:26 -070058 return std::move(boot_control);
Alex Deymob17327c2015-09-04 10:29:00 -070059}
60
61} // namespace boot_control
62
David Zeuthen753fadc2015-09-15 16:34:09 -040063bool BootControlAndroid::Init() {
Chris Phoenixafde8e82017-01-17 23:14:58 -080064 module_ = IBootControl::getService();
Connor O'Briencee6ad92016-11-21 13:53:52 -080065 if (module_ == nullptr) {
Steven Moreland927e00d2017-01-04 12:58:40 -080066 LOG(ERROR) << "Error getting bootctrl HIDL module.";
David Zeuthen753fadc2015-09-15 16:34:09 -040067 return false;
68 }
69
Steven Moreland927e00d2017-01-04 12:58:40 -080070 LOG(INFO) << "Loaded boot control hidl hal.";
David Zeuthen753fadc2015-09-15 16:34:09 -040071
Yifan Hong537802d2018-08-15 13:15:42 -070072 dynamic_control_ = std::make_unique<DynamicPartitionControlAndroid>();
73
David Zeuthen753fadc2015-09-15 16:34:09 -040074 return true;
75}
Alex Deymob17327c2015-09-04 10:29:00 -070076
77unsigned int BootControlAndroid::GetNumSlots() const {
Connor O'Briencee6ad92016-11-21 13:53:52 -080078 return module_->getNumberSlots();
Alex Deymob17327c2015-09-04 10:29:00 -070079}
80
81BootControlInterface::Slot BootControlAndroid::GetCurrentSlot() const {
Connor O'Briencee6ad92016-11-21 13:53:52 -080082 return module_->getCurrentSlot();
Alex Deymob17327c2015-09-04 10:29:00 -070083}
84
Tianjie51a5a392020-06-03 14:39:32 -070085bool BootControlAndroid::GetPartitionDevice(const std::string& partition_name,
86 BootControlInterface::Slot slot,
87 bool not_in_payload,
88 std::string* device,
89 bool* is_dynamic) const {
90 return dynamic_control_->GetPartitionDevice(partition_name,
91 slot,
92 GetCurrentSlot(),
93 not_in_payload,
94 device,
95 is_dynamic);
96}
Yifan Hongae04e192018-10-29 11:00:28 -070097
Yifan Hong537802d2018-08-15 13:15:42 -070098bool BootControlAndroid::GetPartitionDevice(const string& partition_name,
Tianjie51a5a392020-06-03 14:39:32 -070099 BootControlInterface::Slot slot,
Yifan Hong537802d2018-08-15 13:15:42 -0700100 string* device) const {
Tianjie51a5a392020-06-03 14:39:32 -0700101 return GetPartitionDevice(
102 partition_name, slot, false /* not_in_payload */, device, nullptr);
Alex Deymob17327c2015-09-04 10:29:00 -0700103}
104
105bool BootControlAndroid::IsSlotBootable(Slot slot) const {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800106 Return<BoolResult> ret = module_->isSlotBootable(slot);
Yifan Hong7b514b42016-12-21 13:02:00 -0800107 if (!ret.isOk()) {
Alex Deymo31d95ac2015-09-17 11:56:18 -0700108 LOG(ERROR) << "Unable to determine if slot " << SlotName(slot)
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800109 << " is bootable: " << ret.description();
David Zeuthen753fadc2015-09-15 16:34:09 -0400110 return false;
111 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800112 if (ret == BoolResult::INVALID_SLOT) {
113 LOG(ERROR) << "Invalid slot: " << SlotName(slot);
114 return false;
115 }
116 return ret == BoolResult::TRUE;
Alex Deymob17327c2015-09-04 10:29:00 -0700117}
118
119bool BootControlAndroid::MarkSlotUnbootable(Slot slot) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800120 CommandResult result;
121 auto ret = module_->setSlotAsUnbootable(slot, StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800122 if (!ret.isOk()) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800123 LOG(ERROR) << "Unable to call MarkSlotUnbootable for slot "
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800124 << SlotName(slot) << ": " << ret.description();
David Zeuthen753fadc2015-09-15 16:34:09 -0400125 return false;
126 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800127 if (!result.success) {
128 LOG(ERROR) << "Unable to mark slot " << SlotName(slot)
129 << " as unbootable: " << result.errMsg.c_str();
130 }
131 return result.success;
Alex Deymob17327c2015-09-04 10:29:00 -0700132}
133
Alex Deymo31d95ac2015-09-17 11:56:18 -0700134bool BootControlAndroid::SetActiveBootSlot(Slot slot) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800135 CommandResult result;
136 auto ret = module_->setActiveBootSlot(slot, StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800137 if (!ret.isOk()) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800138 LOG(ERROR) << "Unable to call SetActiveBootSlot for slot " << SlotName(slot)
Yifan Hong7b514b42016-12-21 13:02:00 -0800139 << ": " << ret.description();
Connor O'Briencee6ad92016-11-21 13:53:52 -0800140 return false;
Alex Deymo29dcbf32016-10-06 13:33:20 -0700141 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800142 if (!result.success) {
143 LOG(ERROR) << "Unable to set the active slot to slot " << SlotName(slot)
144 << ": " << result.errMsg.c_str();
145 }
146 return result.success;
Alex Deymo31d95ac2015-09-17 11:56:18 -0700147}
148
Alex Deymoaa26f622015-09-16 18:21:27 -0700149bool BootControlAndroid::MarkBootSuccessfulAsync(
150 base::Callback<void(bool)> callback) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800151 CommandResult result;
152 auto ret = module_->markBootSuccessful(StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800153 if (!ret.isOk()) {
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800154 LOG(ERROR) << "Unable to call MarkBootSuccessful: " << ret.description();
Connor O'Briencee6ad92016-11-21 13:53:52 -0800155 return false;
156 }
157 if (!result.success) {
158 LOG(ERROR) << "Unable to mark boot successful: " << result.errMsg.c_str();
Alex Deymoaa26f622015-09-16 18:21:27 -0700159 }
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700160 return brillo::MessageLoop::current()->PostTask(
Connor O'Briencee6ad92016-11-21 13:53:52 -0800161 FROM_HERE, base::Bind(callback, result.success)) !=
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700162 brillo::MessageLoop::kTaskIdNull;
Alex Deymoaa26f622015-09-16 18:21:27 -0700163}
164
Yifan Hongf1415942020-02-24 18:34:49 -0800165bool BootControlAndroid::IsSlotMarkedSuccessful(
166 BootControlInterface::Slot slot) const {
167 Return<BoolResult> ret = module_->isSlotMarkedSuccessful(slot);
168 CommandResult result;
169 if (!ret.isOk()) {
170 LOG(ERROR) << "Unable to determine if slot " << SlotName(slot)
171 << " is marked successful: " << ret.description();
172 return false;
173 }
174 if (ret == BoolResult::INVALID_SLOT) {
175 LOG(ERROR) << "Invalid slot: " << SlotName(slot);
176 return false;
177 }
178 return ret == BoolResult::TRUE;
179}
180
Yifan Hongdaac7322019-11-07 10:48:26 -0800181DynamicPartitionControlInterface*
182BootControlAndroid::GetDynamicPartitionControl() {
183 return dynamic_control_.get();
184}
185
Alex Deymob17327c2015-09-04 10:29:00 -0700186} // namespace chromeos_update_engine