blob: ec2ca0f88146938049a9944ddcbd32a43dbbe0b2 [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
Yifan Hongae04e192018-10-29 11:00:28 -070085
Yifan Hong537802d2018-08-15 13:15:42 -070086bool BootControlAndroid::GetPartitionDevice(const string& partition_name,
87 Slot slot,
88 string* device) const {
Yifan Hong3a1a5612019-11-05 16:34:32 -080089 return dynamic_control_->GetPartitionDevice(
90 partition_name, slot, GetCurrentSlot(), device);
Alex Deymob17327c2015-09-04 10:29:00 -070091}
92
93bool BootControlAndroid::IsSlotBootable(Slot slot) const {
Connor O'Briencee6ad92016-11-21 13:53:52 -080094 Return<BoolResult> ret = module_->isSlotBootable(slot);
Yifan Hong7b514b42016-12-21 13:02:00 -080095 if (!ret.isOk()) {
Alex Deymo31d95ac2015-09-17 11:56:18 -070096 LOG(ERROR) << "Unable to determine if slot " << SlotName(slot)
Amin Hassani7cc8bb02019-01-14 16:29:47 -080097 << " is bootable: " << ret.description();
David Zeuthen753fadc2015-09-15 16:34:09 -040098 return false;
99 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800100 if (ret == BoolResult::INVALID_SLOT) {
101 LOG(ERROR) << "Invalid slot: " << SlotName(slot);
102 return false;
103 }
104 return ret == BoolResult::TRUE;
Alex Deymob17327c2015-09-04 10:29:00 -0700105}
106
107bool BootControlAndroid::MarkSlotUnbootable(Slot slot) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800108 CommandResult result;
109 auto ret = module_->setSlotAsUnbootable(slot, StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800110 if (!ret.isOk()) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800111 LOG(ERROR) << "Unable to call MarkSlotUnbootable for slot "
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800112 << SlotName(slot) << ": " << ret.description();
David Zeuthen753fadc2015-09-15 16:34:09 -0400113 return false;
114 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800115 if (!result.success) {
116 LOG(ERROR) << "Unable to mark slot " << SlotName(slot)
117 << " as unbootable: " << result.errMsg.c_str();
118 }
119 return result.success;
Alex Deymob17327c2015-09-04 10:29:00 -0700120}
121
Alex Deymo31d95ac2015-09-17 11:56:18 -0700122bool BootControlAndroid::SetActiveBootSlot(Slot slot) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800123 CommandResult result;
124 auto ret = module_->setActiveBootSlot(slot, StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800125 if (!ret.isOk()) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800126 LOG(ERROR) << "Unable to call SetActiveBootSlot for slot " << SlotName(slot)
Yifan Hong7b514b42016-12-21 13:02:00 -0800127 << ": " << ret.description();
Connor O'Briencee6ad92016-11-21 13:53:52 -0800128 return false;
Alex Deymo29dcbf32016-10-06 13:33:20 -0700129 }
Connor O'Briencee6ad92016-11-21 13:53:52 -0800130 if (!result.success) {
131 LOG(ERROR) << "Unable to set the active slot to slot " << SlotName(slot)
132 << ": " << result.errMsg.c_str();
133 }
134 return result.success;
Alex Deymo31d95ac2015-09-17 11:56:18 -0700135}
136
Alex Deymoaa26f622015-09-16 18:21:27 -0700137bool BootControlAndroid::MarkBootSuccessfulAsync(
138 base::Callback<void(bool)> callback) {
Connor O'Briencee6ad92016-11-21 13:53:52 -0800139 CommandResult result;
140 auto ret = module_->markBootSuccessful(StoreResultCallback(&result));
Yifan Hong7b514b42016-12-21 13:02:00 -0800141 if (!ret.isOk()) {
Amin Hassani7cc8bb02019-01-14 16:29:47 -0800142 LOG(ERROR) << "Unable to call MarkBootSuccessful: " << ret.description();
Connor O'Briencee6ad92016-11-21 13:53:52 -0800143 return false;
144 }
145 if (!result.success) {
146 LOG(ERROR) << "Unable to mark boot successful: " << result.errMsg.c_str();
Alex Deymoaa26f622015-09-16 18:21:27 -0700147 }
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700148 return brillo::MessageLoop::current()->PostTask(
Connor O'Briencee6ad92016-11-21 13:53:52 -0800149 FROM_HERE, base::Bind(callback, result.success)) !=
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700150 brillo::MessageLoop::kTaskIdNull;
Alex Deymoaa26f622015-09-16 18:21:27 -0700151}
152
Yifan Hong0f9dbb12020-02-24 18:34:49 -0800153bool BootControlAndroid::IsSlotMarkedSuccessful(
154 BootControlInterface::Slot slot) const {
155 Return<BoolResult> ret = module_->isSlotMarkedSuccessful(slot);
156 CommandResult result;
157 if (!ret.isOk()) {
158 LOG(ERROR) << "Unable to determine if slot " << SlotName(slot)
159 << " is marked successful: " << ret.description();
160 return false;
161 }
162 if (ret == BoolResult::INVALID_SLOT) {
163 LOG(ERROR) << "Invalid slot: " << SlotName(slot);
164 return false;
165 }
166 return ret == BoolResult::TRUE;
167}
168
Yifan Hongdaac7322019-11-07 10:48:26 -0800169DynamicPartitionControlInterface*
170BootControlAndroid::GetDynamicPartitionControl() {
171 return dynamic_control_.get();
172}
173
Alex Deymob17327c2015-09-04 10:29:00 -0700174} // namespace chromeos_update_engine