blob: adbacd670a3daa5e3824d8fa4a504753753abd7b [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
43 // BootControlInterface overrides.
44 unsigned int GetNumSlots() const override { return num_slots_; }
45 BootControlInterface::Slot GetCurrentSlot() const override {
46 return current_slot_;
47 }
48
49 bool GetPartitionDevice(const std::string& partition_name,
50 BootControlInterface::Slot slot,
Tianjie51a5a392020-06-03 14:39:32 -070051 bool not_in_payload,
52 std::string* device,
53 bool* is_dynamic) const override {
Alex Deymo763e7db2015-08-27 21:08:08 -070054 if (slot >= num_slots_)
55 return false;
56 auto part_it = devices_[slot].find(partition_name);
57 if (part_it == devices_[slot].end())
58 return false;
59 *device = part_it->second;
60 return true;
61 }
62
Tianjie51a5a392020-06-03 14:39:32 -070063 bool GetPartitionDevice(const std::string& partition_name,
64 BootControlInterface::Slot slot,
65 std::string* device) const override {
66 return GetPartitionDevice(partition_name, slot, false, device, nullptr);
67 }
68
Alex Deymo763e7db2015-08-27 21:08:08 -070069 bool IsSlotBootable(BootControlInterface::Slot slot) const override {
70 return slot < num_slots_ && is_bootable_[slot];
71 }
72
73 bool MarkSlotUnbootable(BootControlInterface::Slot slot) override {
74 if (slot >= num_slots_)
75 return false;
76 is_bootable_[slot] = false;
77 return true;
78 }
79
Alex Deymo31d95ac2015-09-17 11:56:18 -070080 bool SetActiveBootSlot(Slot slot) override { return true; }
81
Alex Deymoaa26f622015-09-16 18:21:27 -070082 bool MarkBootSuccessfulAsync(base::Callback<void(bool)> callback) override {
83 // We run the callback directly from here to avoid having to setup a message
84 // loop in the test environment.
Yifan Hongf1415942020-02-24 18:34:49 -080085 is_marked_successful_[GetCurrentSlot()] = true;
Alex Deymoaa26f622015-09-16 18:21:27 -070086 callback.Run(true);
87 return true;
88 }
89
Yifan Hongf1415942020-02-24 18:34:49 -080090 bool IsSlotMarkedSuccessful(Slot slot) const override {
91 return slot < num_slots_ && is_marked_successful_[slot];
92 }
93
Alex Deymo763e7db2015-08-27 21:08:08 -070094 // Setters
95 void SetNumSlots(unsigned int num_slots) {
96 num_slots_ = num_slots;
97 is_bootable_.resize(num_slots_, false);
Yifan Hongf1415942020-02-24 18:34:49 -080098 is_marked_successful_.resize(num_slots_, false);
Alex Deymo763e7db2015-08-27 21:08:08 -070099 devices_.resize(num_slots_);
100 }
101
Amin Hassanib2689592019-01-13 17:04:28 -0800102 void SetCurrentSlot(BootControlInterface::Slot slot) { current_slot_ = slot; }
Alex Deymo763e7db2015-08-27 21:08:08 -0700103
Chih-Hung Hsieh5c6bb1d2016-07-27 13:33:15 -0700104 void SetPartitionDevice(const std::string& partition_name,
Alex Deymo763e7db2015-08-27 21:08:08 -0700105 BootControlInterface::Slot slot,
Chih-Hung Hsieh5c6bb1d2016-07-27 13:33:15 -0700106 const std::string& device) {
Alex Deymo763e7db2015-08-27 21:08:08 -0700107 DCHECK(slot < num_slots_);
108 devices_[slot][partition_name] = device;
109 }
110
111 void SetSlotBootable(BootControlInterface::Slot slot, bool bootable) {
112 DCHECK(slot < num_slots_);
113 is_bootable_[slot] = bootable;
114 }
115
Yifan Hongdaac7322019-11-07 10:48:26 -0800116 DynamicPartitionControlInterface* GetDynamicPartitionControl() {
117 return dynamic_partition_control_.get();
118 }
119
Alex Deymo763e7db2015-08-27 21:08:08 -0700120 private:
121 BootControlInterface::Slot num_slots_{2};
122 BootControlInterface::Slot current_slot_{0};
123
124 std::vector<bool> is_bootable_;
Yifan Hongf1415942020-02-24 18:34:49 -0800125 std::vector<bool> is_marked_successful_;
Alex Deymo763e7db2015-08-27 21:08:08 -0700126 std::vector<std::map<std::string, std::string>> devices_;
127
Yifan Hongdaac7322019-11-07 10:48:26 -0800128 std::unique_ptr<DynamicPartitionControlInterface> dynamic_partition_control_;
129
Alex Deymo763e7db2015-08-27 21:08:08 -0700130 DISALLOW_COPY_AND_ASSIGN(FakeBootControl);
131};
132
133} // namespace chromeos_update_engine
134
Alex Deymo39910dc2015-11-09 17:04:30 -0800135#endif // UPDATE_ENGINE_COMMON_FAKE_BOOT_CONTROL_H_