| Gunnar Mills | b0ce996 | 2018-09-07 13:39:10 -0500 | [diff] [blame] | 1 | #include "config.h" | 
|  | 2 |  | 
|  | 3 | #include "item_updater.hpp" | 
|  | 4 |  | 
|  | 5 | #include "images.hpp" | 
|  | 6 | #include "serialize.hpp" | 
|  | 7 | #include "version.hpp" | 
| Chanh Nguyen | 1fd6ddd | 2021-01-06 11:09:09 +0700 | [diff] [blame] | 8 | #include "xyz/openbmc_project/Software/ExtendedVersion/server.hpp" | 
| Gunnar Mills | b0ce996 | 2018-09-07 13:39:10 -0500 | [diff] [blame] | 9 | #include "xyz/openbmc_project/Software/Version/server.hpp" | 
|  | 10 |  | 
| Adriana Kobylak | d5b8f75 | 2019-05-01 11:52:35 -0500 | [diff] [blame] | 11 | #include <phosphor-logging/elog-errors.hpp> | 
| Gunnar Mills | b0ce996 | 2018-09-07 13:39:10 -0500 | [diff] [blame] | 12 | #include <phosphor-logging/elog.hpp> | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 13 | #include <phosphor-logging/lg2.hpp> | 
| Adriana Kobylak | 58aa750 | 2020-06-08 11:12:11 -0500 | [diff] [blame] | 14 | #include <xyz/openbmc_project/Common/error.hpp> | 
|  | 15 | #include <xyz/openbmc_project/Software/Image/error.hpp> | 
|  | 16 |  | 
|  | 17 | #include <filesystem> | 
|  | 18 | #include <fstream> | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 19 | #include <queue> | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 20 | #include <set> | 
| Gunnar Mills | ec1b41c | 2017-05-02 12:20:36 -0500 | [diff] [blame] | 21 | #include <string> | 
| George Liu | 44b9fef | 2023-02-07 14:31:32 +0800 | [diff] [blame] | 22 | #include <system_error> | 
| Gunnar Mills | ec1b41c | 2017-05-02 12:20:36 -0500 | [diff] [blame] | 23 |  | 
|  | 24 | namespace phosphor | 
|  | 25 | { | 
|  | 26 | namespace software | 
|  | 27 | { | 
|  | 28 | namespace updater | 
|  | 29 | { | 
|  | 30 |  | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 31 | // When you see server:: you know we're referencing our base class | 
| Patrick Williams | 1e9a5f1 | 2023-08-23 16:53:06 -0500 | [diff] [blame] | 32 | namespace server = sdbusplus::server::xyz::openbmc_project::software; | 
|  | 33 | namespace control = sdbusplus::server::xyz::openbmc_project::control; | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 34 |  | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 35 | PHOSPHOR_LOG2_USING; | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 36 | using namespace phosphor::logging; | 
| Adriana Kobylak | ce82de5 | 2024-01-16 13:56:38 -0600 | [diff] [blame] | 37 | using namespace sdbusplus::error::xyz::openbmc_project::software::image; | 
| Jayanth Othayoth | 2ab9b10 | 2018-02-21 05:27:47 -0600 | [diff] [blame] | 38 | using namespace phosphor::software::image; | 
| Adriana Kobylak | c98d912 | 2020-05-05 10:36:01 -0500 | [diff] [blame] | 39 | namespace fs = std::filesystem; | 
| Adriana Kobylak | ce82de5 | 2024-01-16 13:56:38 -0600 | [diff] [blame] | 40 | using NotAllowed = sdbusplus::error::xyz::openbmc_project::common::NotAllowed; | 
| Saqib Khan | 35e83f3 | 2017-05-22 11:37:32 -0500 | [diff] [blame] | 41 |  | 
| Patrick Williams | bf2bb2b | 2022-07-22 19:26:52 -0500 | [diff] [blame] | 42 | void ItemUpdater::createActivation(sdbusplus::message_t& msg) | 
| Gunnar Mills | ec1b41c | 2017-05-02 12:20:36 -0500 | [diff] [blame] | 43 | { | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 44 | using SVersion = server::Version; | 
|  | 45 | using VersionPurpose = SVersion::VersionPurpose; | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 46 |  | 
| Patrick Williams | bc1facd | 2020-06-03 05:58:27 -0500 | [diff] [blame] | 47 | sdbusplus::message::object_path objPath; | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 48 | auto purpose = VersionPurpose::Unknown; | 
| Chanh Nguyen | 1fd6ddd | 2021-01-06 11:09:09 +0700 | [diff] [blame] | 49 | std::string extendedVersion; | 
| Saqib Khan | 705f1bf | 2017-06-09 23:58:38 -0500 | [diff] [blame] | 50 | std::string version; | 
| Justin Ledford | 054bb0b | 2022-03-15 15:46:58 -0700 | [diff] [blame] | 51 | std::map<std::string, | 
|  | 52 | std::map<std::string, | 
|  | 53 | std::variant<std::string, std::vector<std::string>>>> | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 54 | interfaces; | 
| Patrick Williams | e75d10f | 2017-05-30 16:56:32 -0500 | [diff] [blame] | 55 | msg.read(objPath, interfaces); | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 56 | std::string path(std::move(objPath)); | 
| Saqib Khan | 19177d3 | 2017-06-20 08:11:49 -0500 | [diff] [blame] | 57 | std::string filePath; | 
| Justin Ledford | 054bb0b | 2022-03-15 15:46:58 -0700 | [diff] [blame] | 58 | std::vector<std::string> compatibleNames; | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 59 |  | 
|  | 60 | for (const auto& intf : interfaces) | 
|  | 61 | { | 
| Saqib Khan | 705f1bf | 2017-06-09 23:58:38 -0500 | [diff] [blame] | 62 | if (intf.first == VERSION_IFACE) | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 63 | { | 
| Saqib Khan | 705f1bf | 2017-06-09 23:58:38 -0500 | [diff] [blame] | 64 | for (const auto& property : intf.second) | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 65 | { | 
| Saqib Khan | 705f1bf | 2017-06-09 23:58:38 -0500 | [diff] [blame] | 66 | if (property.first == "Purpose") | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 67 | { | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 68 | auto value = SVersion::convertVersionPurposeFromString( | 
| Patrick Williams | e883fb8 | 2020-05-13 11:38:55 -0500 | [diff] [blame] | 69 | std::get<std::string>(property.second)); | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 70 | if (value == VersionPurpose::BMC || | 
| Vijay Khemka | e9f6c84 | 2020-01-14 14:32:39 -0800 | [diff] [blame] | 71 | #ifdef HOST_BIOS_UPGRADE | 
|  | 72 | value == VersionPurpose::Host || | 
|  | 73 | #endif | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 74 | value == VersionPurpose::System) | 
|  | 75 | { | 
|  | 76 | purpose = value; | 
|  | 77 | } | 
| Saqib Khan | 705f1bf | 2017-06-09 23:58:38 -0500 | [diff] [blame] | 78 | } | 
|  | 79 | else if (property.first == "Version") | 
|  | 80 | { | 
| Patrick Williams | e883fb8 | 2020-05-13 11:38:55 -0500 | [diff] [blame] | 81 | version = std::get<std::string>(property.second); | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 82 | } | 
|  | 83 | } | 
|  | 84 | } | 
| Saqib Khan | 19177d3 | 2017-06-20 08:11:49 -0500 | [diff] [blame] | 85 | else if (intf.first == FILEPATH_IFACE) | 
|  | 86 | { | 
|  | 87 | for (const auto& property : intf.second) | 
|  | 88 | { | 
|  | 89 | if (property.first == "Path") | 
|  | 90 | { | 
| Patrick Williams | e883fb8 | 2020-05-13 11:38:55 -0500 | [diff] [blame] | 91 | filePath = std::get<std::string>(property.second); | 
| Saqib Khan | 19177d3 | 2017-06-20 08:11:49 -0500 | [diff] [blame] | 92 | } | 
|  | 93 | } | 
|  | 94 | } | 
| Chanh Nguyen | 1fd6ddd | 2021-01-06 11:09:09 +0700 | [diff] [blame] | 95 | else if (intf.first == EXTENDED_VERSION_IFACE) | 
|  | 96 | { | 
|  | 97 | for (const auto& property : intf.second) | 
|  | 98 | { | 
|  | 99 | if (property.first == "ExtendedVersion") | 
|  | 100 | { | 
|  | 101 | extendedVersion = std::get<std::string>(property.second); | 
|  | 102 | } | 
|  | 103 | } | 
|  | 104 | } | 
| Justin Ledford | 054bb0b | 2022-03-15 15:46:58 -0700 | [diff] [blame] | 105 | else if (intf.first == COMPATIBLE_IFACE) | 
|  | 106 | { | 
|  | 107 | for (const auto& property : intf.second) | 
|  | 108 | { | 
|  | 109 | if (property.first == "Names") | 
|  | 110 | { | 
|  | 111 | compatibleNames = | 
|  | 112 | std::get<std::vector<std::string>>(property.second); | 
|  | 113 | } | 
|  | 114 | } | 
|  | 115 | } | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 116 | } | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 117 | if (version.empty() || filePath.empty() || | 
| Saqib Khan | 84a0e69 | 2017-06-28 17:27:01 -0500 | [diff] [blame] | 118 | purpose == VersionPurpose::Unknown) | 
| Saqib Khan | 705f1bf | 2017-06-09 23:58:38 -0500 | [diff] [blame] | 119 | { | 
|  | 120 | return; | 
|  | 121 | } | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 122 |  | 
|  | 123 | // Version id is the last item in the path | 
| Pavithra Barithaya | 27d4938 | 2024-06-24 01:52:42 -0500 | [diff] [blame] | 124 | auto pos = path.rfind('/'); | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 125 | if (pos == std::string::npos) | 
|  | 126 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 127 | error("No version id found in object path: {PATH}", "PATH", path); | 
| Patrick Williams | e75d10f | 2017-05-30 16:56:32 -0500 | [diff] [blame] | 128 | return; | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 129 | } | 
|  | 130 |  | 
|  | 131 | auto versionId = path.substr(pos + 1); | 
|  | 132 |  | 
| Patrick Williams | e75d10f | 2017-05-30 16:56:32 -0500 | [diff] [blame] | 133 | if (activations.find(versionId) == activations.end()) | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 134 | { | 
| Jagpal Singh Gill | bb024eb | 2024-04-07 23:34:00 -0700 | [diff] [blame] | 135 | verifyAndCreateObjects(versionId, path, version, purpose, | 
|  | 136 | extendedVersion, filePath, compatibleNames); | 
| Gunnar Mills | 2ce7da2 | 2017-05-04 15:37:56 -0500 | [diff] [blame] | 137 | } | 
| Patrick Williams | e75d10f | 2017-05-30 16:56:32 -0500 | [diff] [blame] | 138 | return; | 
| Gunnar Mills | ec1b41c | 2017-05-02 12:20:36 -0500 | [diff] [blame] | 139 | } | 
|  | 140 |  | 
| Jagpal Singh Gill | bb024eb | 2024-04-07 23:34:00 -0700 | [diff] [blame] | 141 | void ItemUpdater::createActivationWithApplyTime( | 
|  | 142 | std::string& id, std::string& path, | 
|  | 143 | ApplyTimeIntf::RequestedApplyTimes applyTime) | 
|  | 144 | { | 
|  | 145 | info("Creating Activation object for id: {ID}", "ID", id); | 
|  | 146 | AssociationList associations = {}; | 
|  | 147 | // Create an association to the BMC inventory item | 
|  | 148 | associations.emplace_back( | 
|  | 149 | std::make_tuple(ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION, | 
|  | 150 | bmcInventoryPath)); | 
|  | 151 | activations.insert(std::make_pair( | 
|  | 152 | id, std::make_unique<Activation>( | 
| Jagpal Singh Gill | 6397c09 | 2025-07-16 21:18:00 -0700 | [diff] [blame] | 153 | ctx, path, *this, id, server::Activation::Activations::NotReady, | 
| Jagpal Singh Gill | bb024eb | 2024-04-07 23:34:00 -0700 | [diff] [blame] | 154 | associations))); | 
|  | 155 | activations[id]->applyTime = applyTime; | 
|  | 156 | } | 
|  | 157 |  | 
|  | 158 | ActivationIntf::Activations ItemUpdater::verifyAndCreateObjects( | 
|  | 159 | std::string& id, std::string& path, std::string& version, | 
|  | 160 | VersionClass::VersionPurpose purpose, std::string& extendedVersion, | 
|  | 161 | std ::string& filePath, std::vector<std::string>& compatibleNames) | 
|  | 162 | { | 
|  | 163 | // Determine the Activation state by processing the given image dir. | 
|  | 164 | auto activationState = server::Activation::Activations::Invalid; | 
|  | 165 | ItemUpdater::ActivationStatus result; | 
|  | 166 | if (purpose == VersionPurpose::BMC || purpose == VersionPurpose::System) | 
|  | 167 | { | 
|  | 168 | result = ItemUpdater::validateSquashFSImage(filePath); | 
|  | 169 | } | 
|  | 170 | else | 
|  | 171 | { | 
|  | 172 | result = ItemUpdater::ActivationStatus::ready; | 
|  | 173 | } | 
|  | 174 |  | 
|  | 175 | AssociationList associations = {}; | 
|  | 176 |  | 
|  | 177 | if (result == ItemUpdater::ActivationStatus::ready) | 
|  | 178 | { | 
|  | 179 | activationState = server::Activation::Activations::Ready; | 
|  | 180 | // Create an association to the BMC inventory item | 
|  | 181 | associations.emplace_back( | 
|  | 182 | std::make_tuple(ACTIVATION_FWD_ASSOCIATION, | 
|  | 183 | ACTIVATION_REV_ASSOCIATION, bmcInventoryPath)); | 
|  | 184 | } | 
|  | 185 |  | 
|  | 186 | auto versionPtr = std::make_unique<VersionClass>( | 
|  | 187 | bus, path, version, purpose, extendedVersion, filePath, compatibleNames, | 
|  | 188 | std::bind(&ItemUpdater::erase, this, std::placeholders::_1), id); | 
|  | 189 | versionPtr->deleteObject = | 
|  | 190 | std::make_unique<phosphor::software::manager::Delete>( | 
|  | 191 | bus, path, *versionPtr); | 
|  | 192 | versions.insert(std::make_pair(id, std::move(versionPtr))); | 
|  | 193 |  | 
|  | 194 | auto activation = activations.find(id); | 
|  | 195 | if (activation == activations.end()) | 
|  | 196 | { | 
|  | 197 | activations.insert(std::make_pair( | 
| Jagpal Singh Gill | 6397c09 | 2025-07-16 21:18:00 -0700 | [diff] [blame] | 198 | id, std::make_unique<Activation>(ctx, path, *this, id, | 
| Jagpal Singh Gill | bb024eb | 2024-04-07 23:34:00 -0700 | [diff] [blame] | 199 | activationState, associations))); | 
|  | 200 | } | 
|  | 201 | else | 
|  | 202 | { | 
|  | 203 | activation->second->activation(activationState); | 
|  | 204 | } | 
|  | 205 | return activationState; | 
|  | 206 | } | 
|  | 207 |  | 
|  | 208 | bool ItemUpdater::requestActivation(std::string& id) | 
|  | 209 | { | 
|  | 210 | auto activation = activations.find(id); | 
|  | 211 | if (activation == activations.end()) | 
|  | 212 | { | 
|  | 213 | error("Activation object not found for id: {ID}", "ID", id); | 
|  | 214 | return false; | 
|  | 215 | } | 
|  | 216 | activation->second->requestedActivation( | 
|  | 217 | server::Activation::RequestedActivations::Active); | 
|  | 218 | return true; | 
|  | 219 | } | 
|  | 220 |  | 
|  | 221 | bool ItemUpdater::updateActivationStatus(std::string& id, | 
|  | 222 | ActivationIntf::Activations status) | 
|  | 223 | { | 
|  | 224 | auto activation = activations.find(id); | 
|  | 225 | if (activation == activations.end()) | 
|  | 226 | { | 
|  | 227 | error("Activation object not found for id: {ID}", "ID", id); | 
|  | 228 | return false; | 
|  | 229 | } | 
|  | 230 | activation->second->activation(status); | 
|  | 231 | return true; | 
|  | 232 | } | 
|  | 233 |  | 
| Jagpal Singh Gill | 6d131aa | 2024-04-07 23:56:48 -0700 | [diff] [blame] | 234 | void ItemUpdater::createUpdateObject(const std::string& id, | 
|  | 235 | const std::string& path) | 
|  | 236 | { | 
|  | 237 | if (updateManagers.find(id) != updateManagers.end()) | 
|  | 238 | { | 
|  | 239 | error("UpdateManager object already exists for id: {ID}", "ID", id); | 
|  | 240 | return; | 
|  | 241 | } | 
|  | 242 | updateManagers.insert( | 
|  | 243 | std::make_pair(id, std::make_unique<UpdateManager>(ctx, path, *this))); | 
|  | 244 | } | 
|  | 245 |  | 
| Saqib Khan | ba23988 | 2017-05-26 08:41:54 -0500 | [diff] [blame] | 246 | void ItemUpdater::processBMCImage() | 
|  | 247 | { | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 248 | using VersionClass = phosphor::software::manager::Version; | 
| Lei YU | 269bff3 | 2018-08-21 15:21:40 +0800 | [diff] [blame] | 249 |  | 
|  | 250 | // Check MEDIA_DIR and create if it does not exist | 
|  | 251 | try | 
|  | 252 | { | 
|  | 253 | if (!fs::is_directory(MEDIA_DIR)) | 
|  | 254 | { | 
|  | 255 | fs::create_directory(MEDIA_DIR); | 
|  | 256 | } | 
|  | 257 | } | 
|  | 258 | catch (const fs::filesystem_error& e) | 
|  | 259 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 260 | error("Failed to prepare dir: {ERROR}", "ERROR", e); | 
| Lei YU | 269bff3 | 2018-08-21 15:21:40 +0800 | [diff] [blame] | 261 | return; | 
|  | 262 | } | 
|  | 263 |  | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 264 | // Functional images are mounted as rofs-<location>-functional | 
|  | 265 | constexpr auto functionalSuffix = "-functional"; | 
| Lei YU | d474d9c | 2021-12-10 16:21:21 +0800 | [diff] [blame] | 266 | bool functionalFound = false; | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 267 |  | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 268 | // Read os-release from folders under /media/ to get | 
|  | 269 | // BMC Software Versions. | 
| George Liu | 44b9fef | 2023-02-07 14:31:32 +0800 | [diff] [blame] | 270 | std::error_code ec; | 
|  | 271 | for (const auto& iter : fs::directory_iterator(MEDIA_DIR, ec)) | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 272 | { | 
|  | 273 | auto activationState = server::Activation::Activations::Active; | 
| Saqib Khan | 6fab70d | 2017-09-07 00:13:50 -0500 | [diff] [blame] | 274 | static const auto BMC_RO_PREFIX_LEN = strlen(BMC_ROFS_PREFIX); | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 275 |  | 
|  | 276 | // Check if the BMC_RO_PREFIXis the prefix of the iter.path | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 277 | if (0 == | 
|  | 278 | iter.path().native().compare(0, BMC_RO_PREFIX_LEN, BMC_ROFS_PREFIX)) | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 279 | { | 
| Adriana Kobylak | 716cd78 | 2020-06-08 13:27:43 -0500 | [diff] [blame] | 280 | // Get the version to calculate the id | 
| Adriana Kobylak | 24a8d83 | 2020-06-10 08:29:36 -0500 | [diff] [blame] | 281 | fs::path releaseFile(OS_RELEASE_FILE); | 
|  | 282 | auto osRelease = iter.path() / releaseFile.relative_path(); | 
| George Liu | 44b9fef | 2023-02-07 14:31:32 +0800 | [diff] [blame] | 283 | if (!fs::is_regular_file(osRelease, ec)) | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 284 | { | 
| Lei YU | e56bf87 | 2022-02-22 18:40:37 +0800 | [diff] [blame] | 285 | #ifdef BMC_STATIC_DUAL_IMAGE | 
|  | 286 | // For dual image, it is possible that the secondary image is | 
|  | 287 | // empty or contains invalid data, ignore such case. | 
| George Liu | 44b9fef | 2023-02-07 14:31:32 +0800 | [diff] [blame] | 288 | info("Unable to find osRelease: {PATH}: {ERROR_MSG}", "PATH", | 
|  | 289 | osRelease, "ERROR_MSG", ec.message()); | 
| Lei YU | e56bf87 | 2022-02-22 18:40:37 +0800 | [diff] [blame] | 290 | #else | 
| George Liu | 44b9fef | 2023-02-07 14:31:32 +0800 | [diff] [blame] | 291 | error("Failed to read osRelease: {PATH}: {ERROR_MSG}", "PATH", | 
|  | 292 | osRelease, "ERROR_MSG", ec.message()); | 
| Adriana Kobylak | 716cd78 | 2020-06-08 13:27:43 -0500 | [diff] [blame] | 293 |  | 
|  | 294 | // Try to get the version id from the mount directory name and | 
|  | 295 | // call to delete it as this version may be corrupted. Dynamic | 
|  | 296 | // volumes created by the UBI layout for example have the id in | 
|  | 297 | // the mount directory name. The worst that can happen is that | 
|  | 298 | // erase() is called with an non-existent id and returns. | 
|  | 299 | auto id = iter.path().native().substr(BMC_RO_PREFIX_LEN); | 
| Saqib Khan | 021c365 | 2017-09-26 12:11:02 -0500 | [diff] [blame] | 300 | ItemUpdater::erase(id); | 
| Lei YU | e56bf87 | 2022-02-22 18:40:37 +0800 | [diff] [blame] | 301 | #endif | 
| Adriana Kobylak | 716cd78 | 2020-06-08 13:27:43 -0500 | [diff] [blame] | 302 |  | 
| Saqib Khan | 021c365 | 2017-09-26 12:11:02 -0500 | [diff] [blame] | 303 | continue; | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 304 | } | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 305 | auto version = VersionClass::getBMCVersion(osRelease); | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 306 | if (version.empty()) | 
|  | 307 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 308 | error("Failed to read version from osRelease: {PATH}", "PATH", | 
|  | 309 | osRelease); | 
| Adriana Kobylak | 716cd78 | 2020-06-08 13:27:43 -0500 | [diff] [blame] | 310 |  | 
|  | 311 | // Try to delete the version, same as above if the | 
|  | 312 | // OS_RELEASE_FILE does not exist. | 
|  | 313 | auto id = iter.path().native().substr(BMC_RO_PREFIX_LEN); | 
|  | 314 | ItemUpdater::erase(id); | 
|  | 315 |  | 
|  | 316 | continue; | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 317 | } | 
| Saqib Khan | 021c365 | 2017-09-26 12:11:02 -0500 | [diff] [blame] | 318 |  | 
| Adriana Kobylak | 59b640b | 2022-01-21 19:45:22 +0000 | [diff] [blame] | 319 | // The flash location is part of the mount name: rofs-<location> | 
|  | 320 | auto flashId = iter.path().native().substr(BMC_RO_PREFIX_LEN); | 
|  | 321 |  | 
|  | 322 | auto id = VersionClass::getId(version + flashId); | 
| Adriana Kobylak | 716cd78 | 2020-06-08 13:27:43 -0500 | [diff] [blame] | 323 |  | 
| Adriana Kobylak | f383d27 | 2020-06-16 15:17:22 -0500 | [diff] [blame] | 324 | // Check if the id has already been added. This can happen if the | 
|  | 325 | // BMC partitions / devices were manually flashed with the same | 
|  | 326 | // image. | 
|  | 327 | if (versions.find(id) != versions.end()) | 
|  | 328 | { | 
|  | 329 | continue; | 
|  | 330 | } | 
|  | 331 |  | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 332 | auto functional = false; | 
|  | 333 | if (iter.path().native().find(functionalSuffix) != | 
|  | 334 | std::string::npos) | 
|  | 335 | { | 
|  | 336 | // Set functional to true and remove the functional suffix | 
|  | 337 | functional = true; | 
|  | 338 | flashId.erase(flashId.length() - strlen(functionalSuffix)); | 
| Lei YU | d474d9c | 2021-12-10 16:21:21 +0800 | [diff] [blame] | 339 | functionalFound = true; | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 340 | } | 
| Adriana Kobylak | 780220f | 2022-01-18 20:01:53 +0000 | [diff] [blame] | 341 |  | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 342 | auto purpose = server::Version::VersionPurpose::BMC; | 
| Adriana Kobylak | 780220f | 2022-01-18 20:01:53 +0000 | [diff] [blame] | 343 | restorePurpose(flashId, purpose); | 
| Adriana Kobylak | ec4eec3 | 2019-11-13 14:28:35 -0600 | [diff] [blame] | 344 |  | 
| Chanh Nguyen | 1fd6ddd | 2021-01-06 11:09:09 +0700 | [diff] [blame] | 345 | // Read os-release from /etc/ to get the BMC extended version | 
|  | 346 | std::string extendedVersion = | 
|  | 347 | VersionClass::getBMCExtendedVersion(osRelease); | 
|  | 348 |  | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 349 | auto path = fs::path(SOFTWARE_OBJPATH) / id; | 
|  | 350 |  | 
| Adriana Kobylak | 7b11bba | 2024-06-24 10:14:49 -0500 | [diff] [blame] | 351 | // Create functional association and minimum ship level instance if | 
|  | 352 | // this is the functional version | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 353 | if (functional) | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 354 | { | 
|  | 355 | createFunctionalAssociation(path); | 
| Adriana Kobylak | 7b11bba | 2024-06-24 10:14:49 -0500 | [diff] [blame] | 356 |  | 
|  | 357 | if (minimum_ship_level::enabled()) | 
|  | 358 | { | 
|  | 359 | minimumVersionObject = | 
|  | 360 | std::make_unique<MinimumVersion>(bus, path); | 
|  | 361 | minimumVersionObject->minimumVersion( | 
|  | 362 | minimum_ship_level::getMinimumVersion()); | 
|  | 363 | } | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 364 | } | 
|  | 365 |  | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 366 | AssociationList associations; | 
| Gunnar Mills | 43b25cd | 2017-09-07 13:19:34 -0500 | [diff] [blame] | 367 |  | 
|  | 368 | if (activationState == server::Activation::Activations::Active) | 
|  | 369 | { | 
|  | 370 | // Create an association to the BMC inventory item | 
|  | 371 | associations.emplace_back(std::make_tuple( | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 372 | ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION, | 
|  | 373 | bmcInventoryPath)); | 
| Gunnar Mills | 43b25cd | 2017-09-07 13:19:34 -0500 | [diff] [blame] | 374 |  | 
|  | 375 | // Create an active association since this image is active | 
|  | 376 | createActiveAssociation(path); | 
|  | 377 | } | 
|  | 378 |  | 
| Adriana Kobylak | ee590c7 | 2017-09-26 15:16:06 -0500 | [diff] [blame] | 379 | // Create Version instance for this version. | 
|  | 380 | auto versionPtr = std::make_unique<VersionClass>( | 
| Adriana Kobylak | a84f06d | 2022-01-18 15:41:57 +0000 | [diff] [blame] | 381 | bus, path, version, purpose, extendedVersion, flashId, | 
| Justin Ledford | 054bb0b | 2022-03-15 15:46:58 -0700 | [diff] [blame] | 382 | std::vector<std::string>(), | 
| Adriana Kobylak | 59b640b | 2022-01-21 19:45:22 +0000 | [diff] [blame] | 383 | std::bind(&ItemUpdater::erase, this, std::placeholders::_1), | 
|  | 384 | id); | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 385 | if (functional) | 
|  | 386 | { | 
|  | 387 | versionPtr->setFunctional(true); | 
|  | 388 | } | 
|  | 389 | else | 
| Michael Tritz | 4254bec | 2017-10-03 17:18:22 -0500 | [diff] [blame] | 390 | { | 
| Saqib Khan | ee13e83 | 2017-10-23 12:53:11 -0500 | [diff] [blame] | 391 | versionPtr->deleteObject = | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 392 | std::make_unique<phosphor::software::manager::Delete>( | 
|  | 393 | bus, path, *versionPtr); | 
| Michael Tritz | 4254bec | 2017-10-03 17:18:22 -0500 | [diff] [blame] | 394 | } | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 395 | versions.insert(std::make_pair(id, std::move(versionPtr))); | 
| Michael Tritz | 4254bec | 2017-10-03 17:18:22 -0500 | [diff] [blame] | 396 |  | 
| Saqib Khan | ee13e83 | 2017-10-23 12:53:11 -0500 | [diff] [blame] | 397 | // Create Activation instance for this version. | 
|  | 398 | activations.insert(std::make_pair( | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 399 | id, std::make_unique<Activation>( | 
| Jagpal Singh Gill | 6397c09 | 2025-07-16 21:18:00 -0700 | [diff] [blame] | 400 | ctx, path, *this, id, activationState, associations))); | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 401 |  | 
| Lei YU | bdf2d6c | 2021-12-15 14:05:01 +0800 | [diff] [blame] | 402 | #ifdef BMC_STATIC_DUAL_IMAGE | 
|  | 403 | uint8_t priority; | 
|  | 404 | if ((functional && (runningImageSlot == 0)) || | 
|  | 405 | (!functional && (runningImageSlot == 1))) | 
|  | 406 | { | 
|  | 407 | priority = 0; | 
|  | 408 | } | 
|  | 409 | else | 
|  | 410 | { | 
|  | 411 | priority = 1; | 
|  | 412 | } | 
|  | 413 | activations.find(id)->second->redundancyPriority = | 
|  | 414 | std::make_unique<RedundancyPriority>( | 
|  | 415 | bus, path, *(activations.find(id)->second), priority, | 
|  | 416 | false); | 
|  | 417 | #else | 
| Lei YU | 269bff3 | 2018-08-21 15:21:40 +0800 | [diff] [blame] | 418 | // If Active, create RedundancyPriority instance for this | 
|  | 419 | // version. | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 420 | if (activationState == server::Activation::Activations::Active) | 
|  | 421 | { | 
|  | 422 | uint8_t priority = std::numeric_limits<uint8_t>::max(); | 
| Adriana Kobylak | 780220f | 2022-01-18 20:01:53 +0000 | [diff] [blame] | 423 | if (!restorePriority(flashId, priority)) | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 424 | { | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 425 | if (functional) | 
| Adriana Kobylak | ee590c7 | 2017-09-26 15:16:06 -0500 | [diff] [blame] | 426 | { | 
|  | 427 | priority = 0; | 
|  | 428 | } | 
|  | 429 | else | 
|  | 430 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 431 | error( | 
|  | 432 | "Unable to restore priority from file for {VERSIONID}", | 
|  | 433 | "VERSIONID", id); | 
| Adriana Kobylak | ee590c7 | 2017-09-26 15:16:06 -0500 | [diff] [blame] | 434 | } | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 435 | } | 
|  | 436 | activations.find(id)->second->redundancyPriority = | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 437 | std::make_unique<RedundancyPriority>( | 
|  | 438 | bus, path, *(activations.find(id)->second), priority, | 
|  | 439 | false); | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 440 | } | 
| Lei YU | bdf2d6c | 2021-12-15 14:05:01 +0800 | [diff] [blame] | 441 | #endif | 
| Saqib Khan | 1eef62d | 2017-08-10 15:29:34 -0500 | [diff] [blame] | 442 | } | 
|  | 443 | } | 
| Saqib Khan | dcbfa04 | 2017-09-18 13:08:39 -0500 | [diff] [blame] | 444 |  | 
| Jagpal Singh Gill | c111b93 | 2024-08-12 16:07:33 -0700 | [diff] [blame] | 445 | for (const auto& version : versions) | 
|  | 446 | { | 
|  | 447 | if ((versions.size() == 1) || (!version.second->isFunctional())) | 
|  | 448 | { | 
|  | 449 | // This is the only BMC version or the non-functional BMC version | 
|  | 450 | // (in a system with more than one flash), hence create Update | 
|  | 451 | // object and Updateable association for this version | 
|  | 452 | if (useUpdateDBusInterface) | 
|  | 453 | { | 
|  | 454 | createUpdateObject(version.first, version.second->objPath); | 
|  | 455 | } | 
|  | 456 | createUpdateableAssociation(version.second->objPath); | 
|  | 457 | } | 
|  | 458 | } | 
|  | 459 |  | 
| Lei YU | d474d9c | 2021-12-10 16:21:21 +0800 | [diff] [blame] | 460 | if (!functionalFound) | 
| Saqib Khan | dcbfa04 | 2017-09-18 13:08:39 -0500 | [diff] [blame] | 461 | { | 
| Lei YU | d474d9c | 2021-12-10 16:21:21 +0800 | [diff] [blame] | 462 | // If there is no functional version found, read the /etc/os-release and | 
|  | 463 | // create rofs-<versionId>-functional under MEDIA_DIR, then call again | 
|  | 464 | // processBMCImage() to create the D-Bus interface for it. | 
| Gunnar Mills | d16bcbd | 2017-10-08 16:50:42 -0500 | [diff] [blame] | 465 | auto version = VersionClass::getBMCVersion(OS_RELEASE_FILE); | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 466 | auto id = phosphor::software::manager::Version::getId( | 
|  | 467 | version + functionalSuffix); | 
| Adriana Kobylak | 1e81f23 | 2022-01-18 22:28:47 +0000 | [diff] [blame] | 468 | auto versionFileDir = BMC_ROFS_PREFIX + id + functionalSuffix + "/etc/"; | 
| Saqib Khan | dcbfa04 | 2017-09-18 13:08:39 -0500 | [diff] [blame] | 469 | try | 
|  | 470 | { | 
| Gunnar Mills | d16bcbd | 2017-10-08 16:50:42 -0500 | [diff] [blame] | 471 | if (!fs::is_directory(versionFileDir)) | 
| Saqib Khan | dcbfa04 | 2017-09-18 13:08:39 -0500 | [diff] [blame] | 472 | { | 
|  | 473 | fs::create_directories(versionFileDir); | 
|  | 474 | } | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 475 | auto versionFilePath = | 
|  | 476 | BMC_ROFS_PREFIX + id + functionalSuffix + OS_RELEASE_FILE; | 
| Saqib Khan | dcbfa04 | 2017-09-18 13:08:39 -0500 | [diff] [blame] | 477 | fs::create_directory_symlink(OS_RELEASE_FILE, versionFilePath); | 
|  | 478 | ItemUpdater::processBMCImage(); | 
|  | 479 | } | 
|  | 480 | catch (const std::exception& e) | 
|  | 481 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 482 | error("Exception during processing: {ERROR}", "ERROR", e); | 
| Saqib Khan | dcbfa04 | 2017-09-18 13:08:39 -0500 | [diff] [blame] | 483 | } | 
|  | 484 | } | 
| Eddie James | eaa1ee0 | 2018-03-01 10:09:10 -0600 | [diff] [blame] | 485 |  | 
|  | 486 | mirrorUbootToAlt(); | 
| Saqib Khan | ba23988 | 2017-05-26 08:41:54 -0500 | [diff] [blame] | 487 | return; | 
|  | 488 | } | 
|  | 489 |  | 
| Leonel Gonzalez | 3526ef7 | 2017-07-07 14:38:25 -0500 | [diff] [blame] | 490 | void ItemUpdater::erase(std::string entryId) | 
|  | 491 | { | 
| Eddie James | 6d87371 | 2017-09-01 11:29:07 -0500 | [diff] [blame] | 492 | // Find entry in versions map | 
|  | 493 | auto it = versions.find(entryId); | 
|  | 494 | if (it != versions.end()) | 
|  | 495 | { | 
| Alexander Hansen | 01ba956 | 2025-05-12 16:19:36 +0200 | [diff] [blame] | 496 | // NOLINTNEXTLINE(misc-redundant-expression) | 
| Lei YU | 0f88b5a | 2018-08-21 15:28:53 +0800 | [diff] [blame] | 497 | if (it->second->isFunctional() && ACTIVE_BMC_MAX_ALLOWED > 1) | 
| Eddie James | 6d87371 | 2017-09-01 11:29:07 -0500 | [diff] [blame] | 498 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 499 | error( | 
|  | 500 | "Version ({VERSIONID}) is currently running on the BMC; unable to remove.", | 
|  | 501 | "VERSIONID", entryId); | 
| Gunnar Mills | d16bcbd | 2017-10-08 16:50:42 -0500 | [diff] [blame] | 502 | return; | 
| Eddie James | 6d87371 | 2017-09-01 11:29:07 -0500 | [diff] [blame] | 503 | } | 
| Adriana Kobylak | d1a55ad | 2020-06-11 14:01:28 -0500 | [diff] [blame] | 504 | } | 
| Eddie James | 6d87371 | 2017-09-01 11:29:07 -0500 | [diff] [blame] | 505 |  | 
| Adriana Kobylak | d1a55ad | 2020-06-11 14:01:28 -0500 | [diff] [blame] | 506 | // First call resetUbootEnvVars() so that the BMC points to a valid image to | 
|  | 507 | // boot from. If resetUbootEnvVars() is called after the image is actually | 
|  | 508 | // deleted from the BMC flash, there'd be a time window where the BMC would | 
|  | 509 | // be pointing to a non-existent image to boot from. | 
|  | 510 | // Need to remove the entries from the activations map before that call so | 
|  | 511 | // that resetUbootEnvVars() doesn't use the version to be deleted. | 
|  | 512 | auto iteratorActivations = activations.find(entryId); | 
|  | 513 | if (iteratorActivations == activations.end()) | 
|  | 514 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 515 | error( | 
|  | 516 | "Failed to find version ({VERSIONID}) in item updater activations map; unable to remove.", | 
|  | 517 | "VERSIONID", entryId); | 
| Adriana Kobylak | d1a55ad | 2020-06-11 14:01:28 -0500 | [diff] [blame] | 518 | } | 
|  | 519 | else | 
|  | 520 | { | 
|  | 521 | removeAssociations(iteratorActivations->second->path); | 
| Zami Seck | ae06d76 | 2021-08-13 20:11:15 -0500 | [diff] [blame] | 522 | iteratorActivations->second->deleteImageManagerObject(); | 
| Adriana Kobylak | d1a55ad | 2020-06-11 14:01:28 -0500 | [diff] [blame] | 523 | this->activations.erase(entryId); | 
|  | 524 | } | 
|  | 525 | ItemUpdater::resetUbootEnvVars(); | 
|  | 526 |  | 
|  | 527 | if (it != versions.end()) | 
|  | 528 | { | 
| Adriana Kobylak | 780220f | 2022-01-18 20:01:53 +0000 | [diff] [blame] | 529 | auto flashId = it->second->path(); | 
|  | 530 |  | 
| Adriana Kobylak | 25773a7 | 2022-01-21 15:24:48 +0000 | [diff] [blame] | 531 | // Delete version data if it has been installed on flash (path is not | 
|  | 532 | // the upload directory) | 
|  | 533 | if (flashId.find(IMG_UPLOAD_DIR) == std::string::npos) | 
|  | 534 | { | 
|  | 535 | removeReadOnlyPartition(entryId); | 
|  | 536 | removePersistDataDirectory(flashId); | 
|  | 537 | helper.clearEntry(flashId); | 
|  | 538 | } | 
| Saqib Khan | ee13e83 | 2017-10-23 12:53:11 -0500 | [diff] [blame] | 539 |  | 
|  | 540 | // Removing entry in versions map | 
|  | 541 | this->versions.erase(entryId); | 
| Eddie James | 6d87371 | 2017-09-01 11:29:07 -0500 | [diff] [blame] | 542 | } | 
| Leonel Gonzalez | 3526ef7 | 2017-07-07 14:38:25 -0500 | [diff] [blame] | 543 |  | 
| Jagpal Singh Gill | 6d131aa | 2024-04-07 23:56:48 -0700 | [diff] [blame] | 544 | // Removing entry in updateManagers map | 
|  | 545 | auto updateManagerIt = updateManagers.find(entryId); | 
|  | 546 | if (updateManagerIt != updateManagers.end()) | 
|  | 547 | { | 
|  | 548 | updateManagers.erase(entryId); | 
|  | 549 | } | 
|  | 550 |  | 
| Saqib Khan | ee13e83 | 2017-10-23 12:53:11 -0500 | [diff] [blame] | 551 | return; | 
| Leonel Gonzalez | 3526ef7 | 2017-07-07 14:38:25 -0500 | [diff] [blame] | 552 | } | 
|  | 553 |  | 
| Michael Tritz | bc1bf3a | 2017-09-18 16:38:23 -0500 | [diff] [blame] | 554 | void ItemUpdater::deleteAll() | 
|  | 555 | { | 
| Adriana Kobylak | 83cd21f | 2018-02-28 15:48:48 -0600 | [diff] [blame] | 556 | std::vector<std::string> deletableVersions; | 
|  | 557 |  | 
| Michael Tritz | bc1bf3a | 2017-09-18 16:38:23 -0500 | [diff] [blame] | 558 | for (const auto& versionIt : versions) | 
|  | 559 | { | 
|  | 560 | if (!versionIt.second->isFunctional()) | 
|  | 561 | { | 
| Adriana Kobylak | 83cd21f | 2018-02-28 15:48:48 -0600 | [diff] [blame] | 562 | deletableVersions.push_back(versionIt.first); | 
| Michael Tritz | bc1bf3a | 2017-09-18 16:38:23 -0500 | [diff] [blame] | 563 | } | 
|  | 564 | } | 
|  | 565 |  | 
| Adriana Kobylak | 83cd21f | 2018-02-28 15:48:48 -0600 | [diff] [blame] | 566 | for (const auto& deletableIt : deletableVersions) | 
|  | 567 | { | 
|  | 568 | ItemUpdater::erase(deletableIt); | 
|  | 569 | } | 
|  | 570 |  | 
| Lei YU | 56aaf45 | 2018-06-21 16:09:44 +0800 | [diff] [blame] | 571 | helper.cleanup(); | 
| Michael Tritz | bc1bf3a | 2017-09-18 16:38:23 -0500 | [diff] [blame] | 572 | } | 
|  | 573 |  | 
| Patrick Williams | eae5ec9 | 2025-02-01 08:21:45 -0500 | [diff] [blame] | 574 | ItemUpdater::ActivationStatus ItemUpdater::validateSquashFSImage( | 
|  | 575 | const std::string& filePath) | 
| Saqib Khan | 35e83f3 | 2017-05-22 11:37:32 -0500 | [diff] [blame] | 576 | { | 
| Bright Cheng | 8e9ccfe | 2019-11-18 16:18:44 +0800 | [diff] [blame] | 577 | bool valid = true; | 
| Saqib Khan | 35e83f3 | 2017-05-22 11:37:32 -0500 | [diff] [blame] | 578 |  | 
| Bright Cheng | 8e9ccfe | 2019-11-18 16:18:44 +0800 | [diff] [blame] | 579 | // Record the images which are being updated | 
|  | 580 | // First check for the fullimage, then check for images with partitions | 
|  | 581 | imageUpdateList.push_back(bmcFullImages); | 
|  | 582 | valid = checkImage(filePath, imageUpdateList); | 
|  | 583 | if (!valid) | 
| Saqib Khan | 35e83f3 | 2017-05-22 11:37:32 -0500 | [diff] [blame] | 584 | { | 
| Bright Cheng | 8e9ccfe | 2019-11-18 16:18:44 +0800 | [diff] [blame] | 585 | imageUpdateList.clear(); | 
|  | 586 | imageUpdateList.assign(bmcImages.begin(), bmcImages.end()); | 
|  | 587 | valid = checkImage(filePath, imageUpdateList); | 
|  | 588 | if (!valid) | 
| Michael Tritz | b1cfdf9 | 2017-08-14 14:33:30 -0500 | [diff] [blame] | 589 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 590 | error("Failed to find the needed BMC images."); | 
| Bright Cheng | 8e9ccfe | 2019-11-18 16:18:44 +0800 | [diff] [blame] | 591 | return ItemUpdater::ActivationStatus::invalid; | 
| Michael Tritz | b1cfdf9 | 2017-08-14 14:33:30 -0500 | [diff] [blame] | 592 | } | 
| Saqib Khan | 35e83f3 | 2017-05-22 11:37:32 -0500 | [diff] [blame] | 593 | } | 
| Michael Tritz | b1cfdf9 | 2017-08-14 14:33:30 -0500 | [diff] [blame] | 594 |  | 
| Michael Tritz | b1cfdf9 | 2017-08-14 14:33:30 -0500 | [diff] [blame] | 595 | return ItemUpdater::ActivationStatus::ready; | 
| Saqib Khan | 35e83f3 | 2017-05-22 11:37:32 -0500 | [diff] [blame] | 596 | } | 
|  | 597 |  | 
| Adriana Kobylak | bbcb7be | 2018-07-17 15:47:34 -0500 | [diff] [blame] | 598 | void ItemUpdater::savePriority(const std::string& versionId, uint8_t value) | 
|  | 599 | { | 
| Adriana Kobylak | 780220f | 2022-01-18 20:01:53 +0000 | [diff] [blame] | 600 | auto flashId = versions.find(versionId)->second->path(); | 
|  | 601 | storePriority(flashId, value); | 
| Adriana Kobylak | 25773a7 | 2022-01-21 15:24:48 +0000 | [diff] [blame] | 602 | helper.setEntry(flashId, value); | 
| Adriana Kobylak | bbcb7be | 2018-07-17 15:47:34 -0500 | [diff] [blame] | 603 | } | 
|  | 604 |  | 
| Saqib Khan | b9da663 | 2017-09-13 09:48:37 -0500 | [diff] [blame] | 605 | void ItemUpdater::freePriority(uint8_t value, const std::string& versionId) | 
| Saqib Khan | 4c1aec0 | 2017-07-06 11:46:13 -0500 | [diff] [blame] | 606 | { | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 607 | std::map<std::string, uint8_t> priorityMap; | 
|  | 608 |  | 
|  | 609 | // Insert the requested version and priority, it may not exist yet. | 
|  | 610 | priorityMap.insert(std::make_pair(versionId, value)); | 
|  | 611 |  | 
| Saqib Khan | 4c1aec0 | 2017-07-06 11:46:13 -0500 | [diff] [blame] | 612 | for (const auto& intf : activations) | 
|  | 613 | { | 
| Gunnar Mills | 9a78224 | 2017-08-22 16:23:15 -0500 | [diff] [blame] | 614 | if (intf.second->redundancyPriority) | 
| Saqib Khan | 4c1aec0 | 2017-07-06 11:46:13 -0500 | [diff] [blame] | 615 | { | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 616 | priorityMap.insert(std::make_pair( | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 617 | intf.first, intf.second->redundancyPriority->priority())); | 
| Saqib Khan | 4c1aec0 | 2017-07-06 11:46:13 -0500 | [diff] [blame] | 618 | } | 
|  | 619 | } | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 620 |  | 
|  | 621 | // Lambda function to compare 2 priority values, use <= to allow duplicates | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 622 | typedef std::function<bool(std::pair<std::string, uint8_t>, | 
|  | 623 | std::pair<std::string, uint8_t>)> | 
|  | 624 | cmpPriority; | 
|  | 625 | cmpPriority cmpPriorityFunc = | 
| Pavithra Barithaya | 6d17852 | 2024-06-24 04:17:29 -0500 | [diff] [blame] | 626 | [](const std::pair<std::string, uint8_t>& priority1, | 
|  | 627 | const std::pair<std::string, uint8_t>& priority2) { | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 628 | return priority1.second <= priority2.second; | 
|  | 629 | }; | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 630 |  | 
|  | 631 | // Sort versions by ascending priority | 
|  | 632 | std::set<std::pair<std::string, uint8_t>, cmpPriority> prioritySet( | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 633 | priorityMap.begin(), priorityMap.end(), cmpPriorityFunc); | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 634 |  | 
|  | 635 | auto freePriorityValue = value; | 
|  | 636 | for (auto& element : prioritySet) | 
|  | 637 | { | 
|  | 638 | if (element.first == versionId) | 
|  | 639 | { | 
|  | 640 | continue; | 
|  | 641 | } | 
|  | 642 | if (element.second == freePriorityValue) | 
|  | 643 | { | 
|  | 644 | ++freePriorityValue; | 
|  | 645 | auto it = activations.find(element.first); | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 646 | it->second->redundancyPriority->sdbusPriority(freePriorityValue); | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 647 | } | 
|  | 648 | } | 
|  | 649 |  | 
|  | 650 | auto lowestVersion = prioritySet.begin()->first; | 
|  | 651 | if (value == prioritySet.begin()->second) | 
|  | 652 | { | 
|  | 653 | lowestVersion = versionId; | 
|  | 654 | } | 
|  | 655 | updateUbootEnvVars(lowestVersion); | 
| Saqib Khan | 4c1aec0 | 2017-07-06 11:46:13 -0500 | [diff] [blame] | 656 | } | 
|  | 657 |  | 
| Michael Tritz | 37a5904 | 2017-07-12 13:44:53 -0500 | [diff] [blame] | 658 | void ItemUpdater::reset() | 
|  | 659 | { | 
| Pavithra Barithaya | c5f6e7e | 2024-06-24 09:50:21 -0500 | [diff] [blame] | 660 | phosphor::software::updater::Helper::factoryReset(); | 
| Michael Tritz | 37a5904 | 2017-07-12 13:44:53 -0500 | [diff] [blame] | 661 |  | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 662 | info("BMC factory reset will take effect upon reboot."); | 
| Michael Tritz | 37a5904 | 2017-07-12 13:44:53 -0500 | [diff] [blame] | 663 | } | 
|  | 664 |  | 
| Pavithra Barithaya | 6d17852 | 2024-06-24 04:17:29 -0500 | [diff] [blame] | 665 | void ItemUpdater::removeReadOnlyPartition(const std::string& versionId) | 
| Leonel Gonzalez | 3526ef7 | 2017-07-07 14:38:25 -0500 | [diff] [blame] | 666 | { | 
| Adriana Kobylak | 25773a7 | 2022-01-21 15:24:48 +0000 | [diff] [blame] | 667 | auto flashId = versions.find(versionId)->second->path(); | 
|  | 668 | helper.removeVersion(flashId); | 
| Leonel Gonzalez | 3526ef7 | 2017-07-07 14:38:25 -0500 | [diff] [blame] | 669 | } | 
|  | 670 |  | 
| Michael Tritz | 0129d92 | 2017-08-10 19:33:46 -0500 | [diff] [blame] | 671 | bool ItemUpdater::fieldModeEnabled(bool value) | 
|  | 672 | { | 
|  | 673 | // enabling field mode is intended to be one way: false -> true | 
|  | 674 | if (value && !control::FieldMode::fieldModeEnabled()) | 
|  | 675 | { | 
|  | 676 | control::FieldMode::fieldModeEnabled(value); | 
|  | 677 |  | 
| Adriana Kobylak | 22848ec | 2019-10-28 10:08:39 -0500 | [diff] [blame] | 678 | auto method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH, | 
|  | 679 | SYSTEMD_INTERFACE, "StartUnit"); | 
|  | 680 | method.append("obmc-flash-bmc-setenv@fieldmode\\x3dtrue.service", | 
|  | 681 | "replace"); | 
|  | 682 | bus.call_noreply(method); | 
|  | 683 |  | 
|  | 684 | method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH, | 
|  | 685 | SYSTEMD_INTERFACE, "StopUnit"); | 
|  | 686 | method.append("usr-local.mount", "replace"); | 
|  | 687 | bus.call_noreply(method); | 
|  | 688 |  | 
|  | 689 | std::vector<std::string> usrLocal = {"usr-local.mount"}; | 
|  | 690 |  | 
|  | 691 | method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH, | 
|  | 692 | SYSTEMD_INTERFACE, "MaskUnitFiles"); | 
|  | 693 | method.append(usrLocal, false, true); | 
|  | 694 | bus.call_noreply(method); | 
| Michael Tritz | 0129d92 | 2017-08-10 19:33:46 -0500 | [diff] [blame] | 695 | } | 
| Adriana Kobylak | d5b8f75 | 2019-05-01 11:52:35 -0500 | [diff] [blame] | 696 | else if (!value && control::FieldMode::fieldModeEnabled()) | 
|  | 697 | { | 
| Adriana Kobylak | ce82de5 | 2024-01-16 13:56:38 -0600 | [diff] [blame] | 698 | elog<NotAllowed>(xyz::openbmc_project::common::NotAllowed::REASON( | 
| Adriana Kobylak | d5b8f75 | 2019-05-01 11:52:35 -0500 | [diff] [blame] | 699 | "FieldMode is not allowed to be cleared")); | 
|  | 700 | } | 
| Michael Tritz | 0129d92 | 2017-08-10 19:33:46 -0500 | [diff] [blame] | 701 |  | 
|  | 702 | return control::FieldMode::fieldModeEnabled(); | 
|  | 703 | } | 
|  | 704 |  | 
|  | 705 | void ItemUpdater::restoreFieldModeStatus() | 
|  | 706 | { | 
| Isaac Kurth | deb86b4 | 2022-03-08 19:47:53 -0600 | [diff] [blame] | 707 | // The fieldmode u-boot environment variable may not exist since it is not | 
|  | 708 | // part of the default environment, run fw_printenv with 2>&1 to ignore the | 
|  | 709 | // error message in the journal "Error: "fieldmode" not defined" | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 710 | std::pair<int, std::string> ret = | 
|  | 711 | utils::execute("/sbin/fw_printenv", "-n", "fieldmode", "2>&1"); | 
| Michael Tritz | 0129d92 | 2017-08-10 19:33:46 -0500 | [diff] [blame] | 712 |  | 
| Isaac Kurth | deb86b4 | 2022-03-08 19:47:53 -0600 | [diff] [blame] | 713 | if (ret.first != 0) | 
|  | 714 | { | 
|  | 715 | return; | 
|  | 716 | } | 
|  | 717 |  | 
|  | 718 | // truncate any extra characters off the end to compare against a "true" str | 
|  | 719 | std::string result = ret.second.substr(0, 4); | 
|  | 720 | if (result == "true") | 
| Michael Tritz | 0129d92 | 2017-08-10 19:33:46 -0500 | [diff] [blame] | 721 | { | 
|  | 722 | ItemUpdater::fieldModeEnabled(true); | 
|  | 723 | } | 
|  | 724 | } | 
|  | 725 |  | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 726 | void ItemUpdater::setBMCInventoryPath() | 
|  | 727 | { | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 728 | auto depth = 0; | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 729 | auto mapperCall = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH, | 
|  | 730 | MAPPER_INTERFACE, "GetSubTreePaths"); | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 731 |  | 
| Adriana Kobylak | 1254c62 | 2017-12-07 12:24:56 -0600 | [diff] [blame] | 732 | mapperCall.append(INVENTORY_PATH); | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 733 | mapperCall.append(depth); | 
| Adriana Kobylak | 1254c62 | 2017-12-07 12:24:56 -0600 | [diff] [blame] | 734 | std::vector<std::string> filter = {BMC_INVENTORY_INTERFACE}; | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 735 | mapperCall.append(filter); | 
|  | 736 |  | 
| Ed Tanous | 87c7817 | 2018-08-10 12:51:53 -0700 | [diff] [blame] | 737 | try | 
|  | 738 | { | 
|  | 739 | auto response = bus.call(mapperCall); | 
|  | 740 |  | 
|  | 741 | using ObjectPaths = std::vector<std::string>; | 
|  | 742 | ObjectPaths result; | 
|  | 743 | response.read(result); | 
|  | 744 |  | 
|  | 745 | if (!result.empty()) | 
|  | 746 | { | 
|  | 747 | bmcInventoryPath = result.front(); | 
|  | 748 | } | 
|  | 749 | } | 
| Patrick Williams | bf2bb2b | 2022-07-22 19:26:52 -0500 | [diff] [blame] | 750 | catch (const sdbusplus::exception_t& e) | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 751 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 752 | error("Error in mapper GetSubTreePath: {ERROR}", "ERROR", e); | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 753 | return; | 
|  | 754 | } | 
|  | 755 |  | 
| Adriana Kobylak | 1254c62 | 2017-12-07 12:24:56 -0600 | [diff] [blame] | 756 | return; | 
| Gunnar Mills | b60add1 | 2017-08-24 16:41:42 -0500 | [diff] [blame] | 757 | } | 
|  | 758 |  | 
| Gunnar Mills | f10b232 | 2017-09-21 15:31:55 -0500 | [diff] [blame] | 759 | void ItemUpdater::createActiveAssociation(const std::string& path) | 
| Gunnar Mills | ded875d | 2017-08-28 16:44:52 -0500 | [diff] [blame] | 760 | { | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 761 | assocs.emplace_back( | 
|  | 762 | std::make_tuple(ACTIVE_FWD_ASSOCIATION, ACTIVE_REV_ASSOCIATION, path)); | 
| Gunnar Mills | ded875d | 2017-08-28 16:44:52 -0500 | [diff] [blame] | 763 | associations(assocs); | 
|  | 764 | } | 
|  | 765 |  | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 766 | void ItemUpdater::createFunctionalAssociation(const std::string& path) | 
|  | 767 | { | 
|  | 768 | assocs.emplace_back(std::make_tuple(FUNCTIONAL_FWD_ASSOCIATION, | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 769 | FUNCTIONAL_REV_ASSOCIATION, path)); | 
| Gunnar Mills | 88e8a32 | 2017-09-13 11:09:28 -0500 | [diff] [blame] | 770 | associations(assocs); | 
|  | 771 | } | 
|  | 772 |  | 
| AppaRao Puli | bbebec7 | 2020-01-28 23:57:41 +0530 | [diff] [blame] | 773 | void ItemUpdater::createUpdateableAssociation(const std::string& path) | 
|  | 774 | { | 
|  | 775 | assocs.emplace_back(std::make_tuple(UPDATEABLE_FWD_ASSOCIATION, | 
|  | 776 | UPDATEABLE_REV_ASSOCIATION, path)); | 
|  | 777 | associations(assocs); | 
|  | 778 | } | 
|  | 779 |  | 
| Adriana Kobylak | 991af7e | 2018-12-10 13:08:38 -0600 | [diff] [blame] | 780 | void ItemUpdater::removeAssociations(const std::string& path) | 
| Gunnar Mills | ded875d | 2017-08-28 16:44:52 -0500 | [diff] [blame] | 781 | { | 
|  | 782 | for (auto iter = assocs.begin(); iter != assocs.end();) | 
|  | 783 | { | 
| Pavithra Barithaya | 9e307b7 | 2024-06-26 01:27:38 -0500 | [diff] [blame] | 784 | if (std::get<2>(*iter) == path) | 
| Gunnar Mills | ded875d | 2017-08-28 16:44:52 -0500 | [diff] [blame] | 785 | { | 
|  | 786 | iter = assocs.erase(iter); | 
|  | 787 | associations(assocs); | 
|  | 788 | } | 
|  | 789 | else | 
|  | 790 | { | 
|  | 791 | ++iter; | 
|  | 792 | } | 
|  | 793 | } | 
|  | 794 | } | 
|  | 795 |  | 
| Saqib Khan | b9da663 | 2017-09-13 09:48:37 -0500 | [diff] [blame] | 796 | bool ItemUpdater::isLowestPriority(uint8_t value) | 
|  | 797 | { | 
|  | 798 | for (const auto& intf : activations) | 
|  | 799 | { | 
| Gunnar Mills | d16bcbd | 2017-10-08 16:50:42 -0500 | [diff] [blame] | 800 | if (intf.second->redundancyPriority) | 
| Saqib Khan | b9da663 | 2017-09-13 09:48:37 -0500 | [diff] [blame] | 801 | { | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 802 | if (intf.second->redundancyPriority->priority() < value) | 
| Saqib Khan | b9da663 | 2017-09-13 09:48:37 -0500 | [diff] [blame] | 803 | { | 
|  | 804 | return false; | 
|  | 805 | } | 
|  | 806 | } | 
|  | 807 | } | 
|  | 808 | return true; | 
|  | 809 | } | 
|  | 810 |  | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 811 | void ItemUpdater::updateUbootEnvVars(const std::string& versionId) | 
|  | 812 | { | 
| Lei YU | 9c76a0a | 2022-03-14 11:37:15 +0800 | [diff] [blame] | 813 | auto it = versions.find(versionId); | 
|  | 814 | if (it == versions.end()) | 
|  | 815 | { | 
|  | 816 | return; | 
|  | 817 | } | 
|  | 818 | auto flashId = it->second->path(); | 
| Adriana Kobylak | 25773a7 | 2022-01-21 15:24:48 +0000 | [diff] [blame] | 819 | helper.updateUbootVersionId(flashId); | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 820 | } | 
|  | 821 |  | 
| Saqib Khan | 49446ae | 2017-10-02 10:54:20 -0500 | [diff] [blame] | 822 | void ItemUpdater::resetUbootEnvVars() | 
|  | 823 | { | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 824 | decltype(activations.begin()->second->redundancyPriority->priority()) | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 825 | lowestPriority = std::numeric_limits<uint8_t>::max(); | 
| Saqib Khan | 49446ae | 2017-10-02 10:54:20 -0500 | [diff] [blame] | 826 | decltype(activations.begin()->second->versionId) lowestPriorityVersion; | 
|  | 827 | for (const auto& intf : activations) | 
|  | 828 | { | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 829 | if (!intf.second->redundancyPriority) | 
| Saqib Khan | 49446ae | 2017-10-02 10:54:20 -0500 | [diff] [blame] | 830 | { | 
|  | 831 | // Skip this version if the redundancyPriority is not initialized. | 
|  | 832 | continue; | 
|  | 833 | } | 
|  | 834 |  | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 835 | if (intf.second->redundancyPriority->priority() <= lowestPriority) | 
| Saqib Khan | 49446ae | 2017-10-02 10:54:20 -0500 | [diff] [blame] | 836 | { | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 837 | lowestPriority = intf.second->redundancyPriority->priority(); | 
| Saqib Khan | 49446ae | 2017-10-02 10:54:20 -0500 | [diff] [blame] | 838 | lowestPriorityVersion = intf.second->versionId; | 
|  | 839 | } | 
|  | 840 | } | 
|  | 841 |  | 
| Saqib Khan | f0382c3 | 2017-10-24 13:36:22 -0500 | [diff] [blame] | 842 | // Update the U-boot environment variable to point to the lowest priority | 
| Adriana Kobylak | b77551c | 2017-10-27 12:46:23 -0500 | [diff] [blame] | 843 | updateUbootEnvVars(lowestPriorityVersion); | 
| Saqib Khan | 49446ae | 2017-10-02 10:54:20 -0500 | [diff] [blame] | 844 | } | 
|  | 845 |  | 
| Lei YU | d8c9eea | 2021-12-16 16:08:30 +0800 | [diff] [blame] | 846 | void ItemUpdater::freeSpace([[maybe_unused]] const Activation& caller) | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 847 | { | 
| Lei YU | d8c9eea | 2021-12-16 16:08:30 +0800 | [diff] [blame] | 848 | #ifdef BMC_STATIC_DUAL_IMAGE | 
|  | 849 | // For the golden image case, always remove the version on the primary side | 
|  | 850 | std::string versionIDtoErase; | 
|  | 851 | for (const auto& iter : activations) | 
|  | 852 | { | 
|  | 853 | if (iter.second->redundancyPriority && | 
| Pavithra Barithaya | 9d7b331 | 2024-06-26 01:58:51 -0500 | [diff] [blame] | 854 | iter.second->redundancyPriority->priority() == 0) | 
| Lei YU | d8c9eea | 2021-12-16 16:08:30 +0800 | [diff] [blame] | 855 | { | 
|  | 856 | versionIDtoErase = iter.second->versionId; | 
|  | 857 | break; | 
|  | 858 | } | 
|  | 859 | } | 
|  | 860 | if (!versionIDtoErase.empty()) | 
|  | 861 | { | 
|  | 862 | erase(versionIDtoErase); | 
|  | 863 | } | 
|  | 864 | else | 
|  | 865 | { | 
|  | 866 | warning("Failed to find version to erase"); | 
|  | 867 | } | 
|  | 868 | #else | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 869 | //  Versions with the highest priority in front | 
|  | 870 | std::priority_queue<std::pair<int, std::string>, | 
|  | 871 | std::vector<std::pair<int, std::string>>, | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 872 | std::less<std::pair<int, std::string>>> | 
|  | 873 | versionsPQ; | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 874 |  | 
|  | 875 | std::size_t count = 0; | 
|  | 876 | for (const auto& iter : activations) | 
|  | 877 | { | 
|  | 878 | if ((iter.second.get()->activation() == | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 879 | server::Activation::Activations::Active) || | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 880 | (iter.second.get()->activation() == | 
| Adriana Kobylak | 2285fe0 | 2018-02-27 15:36:59 -0600 | [diff] [blame] | 881 | server::Activation::Activations::Failed)) | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 882 | { | 
|  | 883 | count++; | 
|  | 884 | // Don't put the functional version on the queue since we can't | 
|  | 885 | // remove the "running" BMC version. | 
| Lei YU | 0f88b5a | 2018-08-21 15:28:53 +0800 | [diff] [blame] | 886 | // If ACTIVE_BMC_MAX_ALLOWED <= 1, there is only one active BMC, | 
|  | 887 | // so remove functional version as well. | 
| Adriana Kobylak | a696359 | 2018-09-07 14:13:29 -0500 | [diff] [blame] | 888 | // Don't delete the the Activation object that called this function. | 
|  | 889 | if ((versions.find(iter.second->versionId) | 
|  | 890 | ->second->isFunctional() && | 
|  | 891 | ACTIVE_BMC_MAX_ALLOWED > 1) || | 
|  | 892 | (iter.second->versionId == caller.versionId)) | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 893 | { | 
|  | 894 | continue; | 
|  | 895 | } | 
| Adriana Kobylak | a696359 | 2018-09-07 14:13:29 -0500 | [diff] [blame] | 896 |  | 
|  | 897 | // Failed activations don't have priority, assign them a large value | 
|  | 898 | // for sorting purposes. | 
|  | 899 | auto priority = 999; | 
|  | 900 | if (iter.second.get()->activation() == | 
| Lei YU | cfb4b20 | 2021-05-06 17:47:28 +0800 | [diff] [blame] | 901 | server::Activation::Activations::Active && | 
|  | 902 | iter.second->redundancyPriority) | 
| Adriana Kobylak | a696359 | 2018-09-07 14:13:29 -0500 | [diff] [blame] | 903 | { | 
|  | 904 | priority = iter.second->redundancyPriority.get()->priority(); | 
|  | 905 | } | 
|  | 906 |  | 
|  | 907 | versionsPQ.push(std::make_pair(priority, iter.second->versionId)); | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 908 | } | 
|  | 909 | } | 
|  | 910 |  | 
|  | 911 | // If the number of BMC versions is over ACTIVE_BMC_MAX_ALLOWED -1, | 
|  | 912 | // remove the highest priority one(s). | 
|  | 913 | while ((count >= ACTIVE_BMC_MAX_ALLOWED) && (!versionsPQ.empty())) | 
|  | 914 | { | 
|  | 915 | erase(versionsPQ.top().second); | 
|  | 916 | versionsPQ.pop(); | 
|  | 917 | count--; | 
|  | 918 | } | 
| Lei YU | d8c9eea | 2021-12-16 16:08:30 +0800 | [diff] [blame] | 919 | #endif | 
| Adriana Kobylak | 204e1e7 | 2018-01-24 16:00:05 -0600 | [diff] [blame] | 920 | } | 
|  | 921 |  | 
| Eddie James | eaa1ee0 | 2018-03-01 10:09:10 -0600 | [diff] [blame] | 922 | void ItemUpdater::mirrorUbootToAlt() | 
|  | 923 | { | 
| Lei YU | 56aaf45 | 2018-06-21 16:09:44 +0800 | [diff] [blame] | 924 | helper.mirrorAlt(); | 
| Eddie James | eaa1ee0 | 2018-03-01 10:09:10 -0600 | [diff] [blame] | 925 | } | 
|  | 926 |  | 
| Bright Cheng | 8e9ccfe | 2019-11-18 16:18:44 +0800 | [diff] [blame] | 927 | bool ItemUpdater::checkImage(const std::string& filePath, | 
|  | 928 | const std::vector<std::string>& imageList) | 
|  | 929 | { | 
|  | 930 | bool valid = true; | 
|  | 931 |  | 
|  | 932 | for (auto& bmcImage : imageList) | 
|  | 933 | { | 
|  | 934 | fs::path file(filePath); | 
|  | 935 | file /= bmcImage; | 
|  | 936 | std::ifstream efile(file.c_str()); | 
|  | 937 | if (efile.good() != 1) | 
|  | 938 | { | 
|  | 939 | valid = false; | 
|  | 940 | break; | 
|  | 941 | } | 
|  | 942 | } | 
|  | 943 |  | 
|  | 944 | return valid; | 
|  | 945 | } | 
|  | 946 |  | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 947 | #ifdef HOST_BIOS_UPGRADE | 
|  | 948 | void ItemUpdater::createBIOSObject() | 
|  | 949 | { | 
|  | 950 | std::string path = BIOS_OBJPATH; | 
|  | 951 | // Get version id from last item in the path | 
| Pavithra Barithaya | 27d4938 | 2024-06-24 01:52:42 -0500 | [diff] [blame] | 952 | auto pos = path.rfind('/'); | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 953 | if (pos == std::string::npos) | 
|  | 954 | { | 
| Patrick Williams | c9bb642 | 2021-08-27 06:18:35 -0500 | [diff] [blame] | 955 | error("No version id found in object path {PATH}", "PATH", path); | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 956 | return; | 
|  | 957 | } | 
|  | 958 |  | 
|  | 959 | createActiveAssociation(path); | 
|  | 960 | createFunctionalAssociation(path); | 
| Daniel Hsu | 27d734f | 2024-05-21 15:20:23 +0800 | [diff] [blame] | 961 | createUpdateableAssociation(path); | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 962 |  | 
|  | 963 | auto versionId = path.substr(pos + 1); | 
|  | 964 | auto version = "null"; | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 965 | AssociationList assocs; | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 966 | biosActivation = std::make_unique<Activation>( | 
| Jagpal Singh Gill | 6397c09 | 2025-07-16 21:18:00 -0700 | [diff] [blame] | 967 | ctx, path, *this, versionId, server::Activation::Activations::Active, | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 968 | assocs); | 
| Pavithra Barithaya | 6d17852 | 2024-06-24 04:17:29 -0500 | [diff] [blame] | 969 | auto dummyErase = [](const std::string& /*entryId*/) { | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 970 | // Do nothing; | 
|  | 971 | }; | 
|  | 972 | biosVersion = std::make_unique<VersionClass>( | 
|  | 973 | bus, path, version, VersionPurpose::Host, "", "", | 
| Justin Ledford | 054bb0b | 2022-03-15 15:46:58 -0700 | [diff] [blame] | 974 | std::vector<std::string>(), | 
| Adriana Kobylak | 59b640b | 2022-01-21 19:45:22 +0000 | [diff] [blame] | 975 | std::bind(dummyErase, std::placeholders::_1), ""); | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 976 | biosVersion->deleteObject = | 
| Patrick Williams | fc33ba8 | 2024-08-16 15:19:54 -0400 | [diff] [blame] | 977 | std::make_unique<phosphor::software::manager::Delete>( | 
|  | 978 | bus, path, *biosVersion); | 
| Jagpal Singh Gill | 6d131aa | 2024-04-07 23:56:48 -0700 | [diff] [blame] | 979 |  | 
|  | 980 | if (useUpdateDBusInterface) | 
|  | 981 | { | 
|  | 982 | createUpdateObject(versionId, path); | 
|  | 983 | } | 
| Lei YU | 6e9fb1d | 2021-02-19 18:01:40 +0800 | [diff] [blame] | 984 | } | 
|  | 985 | #endif | 
|  | 986 |  | 
| Lei YU | 531fbc2 | 2021-12-10 20:03:18 +0800 | [diff] [blame] | 987 | void ItemUpdater::getRunningSlot() | 
|  | 988 | { | 
|  | 989 | // Check /run/media/slot to get the slot number | 
|  | 990 | constexpr auto slotFile = "/run/media/slot"; | 
|  | 991 | std::fstream f(slotFile, std::ios_base::in); | 
|  | 992 | f >> runningImageSlot; | 
|  | 993 | } | 
|  | 994 |  | 
| Gunnar Mills | ec1b41c | 2017-05-02 12:20:36 -0500 | [diff] [blame] | 995 | } // namespace updater | 
|  | 996 | } // namespace software | 
|  | 997 | } // namespace phosphor |