blob: 82d4827d6c86955d74e5b4dd4f6c0728f9b74b57 [file] [log] [blame]
Alex Deymo763e7db2015-08-27 21:08:08 -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 Deymo39910dc2015-11-09 17:04:30 -080017#ifndef UPDATE_ENGINE_COMMON_FAKE_BOOT_CONTROL_H_
18#define UPDATE_ENGINE_COMMON_FAKE_BOOT_CONTROL_H_
Alex Deymo763e7db2015-08-27 21:08:08 -070019
20#include <map>
Yifan Hongdaac7322019-11-07 10:48:26 -080021#include <memory>
Alex Deymo763e7db2015-08-27 21:08:08 -070022#include <string>
23#include <vector>
24
25#include <base/time/time.h>
26
Alex Deymo39910dc2015-11-09 17:04:30 -080027#include "update_engine/common/boot_control_interface.h"
Yifan Hongdaac7322019-11-07 10:48:26 -080028#include "update_engine/common/dynamic_partition_control_stub.h"
Alex Deymo763e7db2015-08-27 21:08:08 -070029
30namespace chromeos_update_engine {
31
32// Implements a fake bootloader control interface used for testing.
33class FakeBootControl : public BootControlInterface {
34 public:
35 FakeBootControl() {
36 SetNumSlots(num_slots_);
37 // The current slot should be bootable.
38 is_bootable_[current_slot_] = true;
Yifan Hongdaac7322019-11-07 10:48:26 -080039
40 dynamic_partition_control_.reset(new DynamicPartitionControlStub());
Alex Deymo763e7db2015-08-27 21:08:08 -070041 }
42
Kelvin Zhang67b43782025-02-10 11:06:22 -080043 void SetDynamicPartitionControl(
44 std::unique_ptr<DynamicPartitionControlInterface> dynamic_control) {
45 dynamic_partition_control_ = std::move(dynamic_control);
46 }
47
Alex Deymo763e7db2015-08-27 21:08:08 -070048 // BootControlInterface overrides.
49 unsigned int GetNumSlots() const override { return num_slots_; }
50 BootControlInterface::Slot GetCurrentSlot() const override {
51 return current_slot_;
52 }
53
54 bool GetPartitionDevice(const std::string& partition_name,
55 BootControlInterface::Slot slot,
Tianjie51a5a392020-06-03 14:39:32 -070056 bool not_in_payload,
57 std::string* device,
58 bool* is_dynamic) const override {
Kelvin Zhang91d95fa2020-11-05 13:52:00 -050059 auto dev =
60 GetPartitionDevice(partition_name, slot, current_slot_, not_in_payload);
61 if (!dev.has_value()) {
Alex Deymo763e7db2015-08-27 21:08:08 -070062 return false;
Kelvin Zhang91d95fa2020-11-05 13:52:00 -050063 }
64 if (is_dynamic) {
65 *is_dynamic = dev->is_dynamic;
66 }
67 if (device) {
68 *device = dev->rw_device_path;
Kelvin Zhang3b323cf2020-08-05 10:56:02 -040069 }
Alex Deymo763e7db2015-08-27 21:08:08 -070070 return true;
71 }
72
Tianjie51a5a392020-06-03 14:39:32 -070073 bool GetPartitionDevice(const std::string& partition_name,
74 BootControlInterface::Slot slot,
75 std::string* device) const override {
76 return GetPartitionDevice(partition_name, slot, false, device, nullptr);
77 }
78
Alex Deymo763e7db2015-08-27 21:08:08 -070079 bool IsSlotBootable(BootControlInterface::Slot slot) const override {
80 return slot < num_slots_ && is_bootable_[slot];
81 }
82
83 bool MarkSlotUnbootable(BootControlInterface::Slot slot) override {
84 if (slot >= num_slots_)
85 return false;
86 is_bootable_[slot] = false;
87 return true;
88 }
89
Alex Deymo31d95ac2015-09-17 11:56:18 -070090 bool SetActiveBootSlot(Slot slot) override { return true; }
Kelvin Zhangcb419e62021-06-16 13:56:47 -040091 Slot GetActiveBootSlot() override { return kInvalidSlot; }
Alex Deymo31d95ac2015-09-17 11:56:18 -070092
Alex Deymoaa26f622015-09-16 18:21:27 -070093 bool MarkBootSuccessfulAsync(base::Callback<void(bool)> callback) override {
94 // We run the callback directly from here to avoid having to setup a message
95 // loop in the test environment.
Yifan Hongf1415942020-02-24 18:34:49 -080096 is_marked_successful_[GetCurrentSlot()] = true;
Alex Deymoaa26f622015-09-16 18:21:27 -070097 callback.Run(true);
98 return true;
99 }
100
Yifan Hongf1415942020-02-24 18:34:49 -0800101 bool IsSlotMarkedSuccessful(Slot slot) const override {
102 return slot < num_slots_ && is_marked_successful_[slot];
Yifan Hong537802d2018-08-15 13:15:42 -0700103 }
104
Alex Deymo763e7db2015-08-27 21:08:08 -0700105 // Setters
106 void SetNumSlots(unsigned int num_slots) {
107 num_slots_ = num_slots;
108 is_bootable_.resize(num_slots_, false);
Yifan Hongf1415942020-02-24 18:34:49 -0800109 is_marked_successful_.resize(num_slots_, false);
Alex Deymo763e7db2015-08-27 21:08:08 -0700110 devices_.resize(num_slots_);
111 }
112
Amin Hassanib2689592019-01-13 17:04:28 -0800113 void SetCurrentSlot(BootControlInterface::Slot slot) { current_slot_ = slot; }
Alex Deymo763e7db2015-08-27 21:08:08 -0700114
Chih-Hung Hsieh5c6bb1d2016-07-27 13:33:15 -0700115 void SetPartitionDevice(const std::string& partition_name,
Alex Deymo763e7db2015-08-27 21:08:08 -0700116 BootControlInterface::Slot slot,
Chih-Hung Hsieh5c6bb1d2016-07-27 13:33:15 -0700117 const std::string& device) {
Alex Deymo763e7db2015-08-27 21:08:08 -0700118 DCHECK(slot < num_slots_);
119 devices_[slot][partition_name] = device;
120 }
121
122 void SetSlotBootable(BootControlInterface::Slot slot, bool bootable) {
123 DCHECK(slot < num_slots_);
124 is_bootable_[slot] = bootable;
125 }
126
Amin Hassanie53b39b2020-09-16 11:19:28 -0700127 DynamicPartitionControlInterface* GetDynamicPartitionControl() override {
Yifan Hongdaac7322019-11-07 10:48:26 -0800128 return dynamic_partition_control_.get();
129 }
130
Kelvin Zhang91d95fa2020-11-05 13:52:00 -0500131 std::optional<PartitionDevice> GetPartitionDevice(
132 const std::string& partition_name,
133 uint32_t slot,
134 uint32_t current_slot,
135 bool not_in_payload = false) const override {
136 if (slot >= devices_.size()) {
137 return {};
138 }
139 auto device_path = devices_[slot].find(partition_name);
140 if (device_path == devices_[slot].end()) {
141 return {};
142 }
143 PartitionDevice device;
144 device.is_dynamic = false;
145 device.rw_device_path = device_path->second;
Kelvin Zhanga9b5d8c2021-05-05 09:17:46 -0400146 device.readonly_device_path = device.rw_device_path;
Kelvin Zhang91d95fa2020-11-05 13:52:00 -0500147 return device;
148 }
149
Alex Deymo763e7db2015-08-27 21:08:08 -0700150 private:
151 BootControlInterface::Slot num_slots_{2};
152 BootControlInterface::Slot current_slot_{0};
153
154 std::vector<bool> is_bootable_;
Yifan Hongf1415942020-02-24 18:34:49 -0800155 std::vector<bool> is_marked_successful_;
Alex Deymo763e7db2015-08-27 21:08:08 -0700156 std::vector<std::map<std::string, std::string>> devices_;
157
Yifan Hongdaac7322019-11-07 10:48:26 -0800158 std::unique_ptr<DynamicPartitionControlInterface> dynamic_partition_control_;
159
Alex Deymo763e7db2015-08-27 21:08:08 -0700160 DISALLOW_COPY_AND_ASSIGN(FakeBootControl);
161};
162
163} // namespace chromeos_update_engine
164
Alex Deymo39910dc2015-11-09 17:04:30 -0800165#endif // UPDATE_ENGINE_COMMON_FAKE_BOOT_CONTROL_H_