blob: 98b93e66d16a7cf8184651edd5d0b9e37bdb7293 [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;
Kelvin Zhang3b323cf2020-08-05 10:56:02 -040060 if (is_dynamic != nullptr) {
61 *is_dynamic = false;
62 }
Alex Deymo763e7db2015-08-27 21:08:08 -070063 return true;
64 }
65
Tianjie51a5a392020-06-03 14:39:32 -070066 bool GetPartitionDevice(const std::string& partition_name,
67 BootControlInterface::Slot slot,
68 std::string* device) const override {
69 return GetPartitionDevice(partition_name, slot, false, device, nullptr);
70 }
71
Alex Deymo763e7db2015-08-27 21:08:08 -070072 bool IsSlotBootable(BootControlInterface::Slot slot) const override {
73 return slot < num_slots_ && is_bootable_[slot];
74 }
75
76 bool MarkSlotUnbootable(BootControlInterface::Slot slot) override {
77 if (slot >= num_slots_)
78 return false;
79 is_bootable_[slot] = false;
80 return true;
81 }
82
Alex Deymo31d95ac2015-09-17 11:56:18 -070083 bool SetActiveBootSlot(Slot slot) override { return true; }
84
Alex Deymoaa26f622015-09-16 18:21:27 -070085 bool MarkBootSuccessfulAsync(base::Callback<void(bool)> callback) override {
86 // We run the callback directly from here to avoid having to setup a message
87 // loop in the test environment.
Yifan Hongf1415942020-02-24 18:34:49 -080088 is_marked_successful_[GetCurrentSlot()] = true;
Alex Deymoaa26f622015-09-16 18:21:27 -070089 callback.Run(true);
90 return true;
91 }
92
Yifan Hongf1415942020-02-24 18:34:49 -080093 bool IsSlotMarkedSuccessful(Slot slot) const override {
94 return slot < num_slots_ && is_marked_successful_[slot];
Yifan Hong537802d2018-08-15 13:15:42 -070095 }
96
Alex Deymo763e7db2015-08-27 21:08:08 -070097 // Setters
98 void SetNumSlots(unsigned int num_slots) {
99 num_slots_ = num_slots;
100 is_bootable_.resize(num_slots_, false);
Yifan Hongf1415942020-02-24 18:34:49 -0800101 is_marked_successful_.resize(num_slots_, false);
Alex Deymo763e7db2015-08-27 21:08:08 -0700102 devices_.resize(num_slots_);
103 }
104
Amin Hassanib2689592019-01-13 17:04:28 -0800105 void SetCurrentSlot(BootControlInterface::Slot slot) { current_slot_ = slot; }
Alex Deymo763e7db2015-08-27 21:08:08 -0700106
Chih-Hung Hsieh5c6bb1d2016-07-27 13:33:15 -0700107 void SetPartitionDevice(const std::string& partition_name,
Alex Deymo763e7db2015-08-27 21:08:08 -0700108 BootControlInterface::Slot slot,
Chih-Hung Hsieh5c6bb1d2016-07-27 13:33:15 -0700109 const std::string& device) {
Alex Deymo763e7db2015-08-27 21:08:08 -0700110 DCHECK(slot < num_slots_);
111 devices_[slot][partition_name] = device;
112 }
113
114 void SetSlotBootable(BootControlInterface::Slot slot, bool bootable) {
115 DCHECK(slot < num_slots_);
116 is_bootable_[slot] = bootable;
117 }
118
Amin Hassanie53b39b2020-09-16 11:19:28 -0700119 DynamicPartitionControlInterface* GetDynamicPartitionControl() override {
Yifan Hongdaac7322019-11-07 10:48:26 -0800120 return dynamic_partition_control_.get();
121 }
122
Alex Deymo763e7db2015-08-27 21:08:08 -0700123 private:
124 BootControlInterface::Slot num_slots_{2};
125 BootControlInterface::Slot current_slot_{0};
126
127 std::vector<bool> is_bootable_;
Yifan Hongf1415942020-02-24 18:34:49 -0800128 std::vector<bool> is_marked_successful_;
Alex Deymo763e7db2015-08-27 21:08:08 -0700129 std::vector<std::map<std::string, std::string>> devices_;
130
Yifan Hongdaac7322019-11-07 10:48:26 -0800131 std::unique_ptr<DynamicPartitionControlInterface> dynamic_partition_control_;
132
Alex Deymo763e7db2015-08-27 21:08:08 -0700133 DISALLOW_COPY_AND_ASSIGN(FakeBootControl);
134};
135
136} // namespace chromeos_update_engine
137
Alex Deymo39910dc2015-11-09 17:04:30 -0800138#endif // UPDATE_ENGINE_COMMON_FAKE_BOOT_CONTROL_H_