#include "config.h" #include "item_updater.hpp" #include "utils.hpp" #include #include #include #include #include #include #include namespace { constexpr auto MANIFEST_VERSION = "version"; constexpr auto MANIFEST_EXTENDED_VERSION = "extended_version"; } // namespace namespace phosphor { namespace software { namespace updater { namespace server = sdbusplus::xyz::openbmc_project::Software::server; using namespace sdbusplus::xyz::openbmc_project::Common::Error; using namespace phosphor::logging; using SVersion = server::Version; using VersionPurpose = SVersion::VersionPurpose; void ItemUpdater::createActivation(sdbusplus::message_t& m) { sdbusplus::message::object_path objPath; std::map>> interfaces; m.read(objPath, interfaces); std::string path(std::move(objPath)); std::string filePath; auto purpose = VersionPurpose::Unknown; std::string version; for (const auto& [interfaceName, propertyMap] : interfaces) { if (interfaceName == VERSION_IFACE) { for (const auto& [propertyName, propertyValue] : propertyMap) { if (propertyName == "Purpose") { // Only process the PSU images auto value = SVersion::convertVersionPurposeFromString( std::get(propertyValue)); if (value == VersionPurpose::PSU) { purpose = value; } } else if (propertyName == VERSION) { version = std::get(propertyValue); } } } else if (interfaceName == FILEPATH_IFACE) { const auto& it = propertyMap.find("Path"); if (it != propertyMap.end()) { filePath = std::get(it->second); } } } if ((filePath.empty()) || (purpose == VersionPurpose::Unknown)) { return; } // If we are only installing PSU images from the built-in directory, ignore // PSU images from other directories if (ALWAYS_USE_BUILTIN_IMG_DIR && !filePath.starts_with(IMG_DIR_BUILTIN)) { return; } // Version id is the last item in the path auto pos = path.rfind('/'); if (pos == std::string::npos) { lg2::error("No version id found in object path {OBJPATH}", "OBJPATH", path); return; } auto versionId = path.substr(pos + 1); if (activations.find(versionId) == activations.end()) { // Determine the Activation state by processing the given image dir. AssociationList associations; auto activationState = Activation::Status::Ready; associations.emplace_back(std::make_tuple( ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION, PSU_INVENTORY_PATH_BASE)); fs::path manifestPath(filePath); manifestPath /= MANIFEST_FILE; std::string extendedVersion = Version::getValue(manifestPath, {MANIFEST_EXTENDED_VERSION}); auto activation = createActivationObject(path, versionId, extendedVersion, activationState, associations, filePath); activations.emplace(versionId, std::move(activation)); auto versionPtr = createVersionObject(path, versionId, version, purpose); versions.emplace(versionId, std::move(versionPtr)); } return; } void ItemUpdater::erase(const std::string& versionId) { auto it = versions.find(versionId); if (it == versions.end()) { lg2::error("Error: Failed to find version {VERSION_ID} in " "item updater versions map. Unable to remove.", "VERSION_ID", versionId); } else { versionStrings.erase(it->second->getVersionString()); versions.erase(it); } // Removing entry in activations map auto ita = activations.find(versionId); if (ita == activations.end()) { lg2::error("Error: Failed to find version {VERSION_ID} in " "item updater activations map. Unable to remove.", "VERSION_ID", versionId); } else { activations.erase(versionId); } } void ItemUpdater::createActiveAssociation(const std::string& path) { assocs.emplace_back( std::make_tuple(ACTIVE_FWD_ASSOCIATION, ACTIVE_REV_ASSOCIATION, path)); associations(assocs); } void ItemUpdater::addFunctionalAssociation(const std::string& path) { assocs.emplace_back(std::make_tuple(FUNCTIONAL_FWD_ASSOCIATION, FUNCTIONAL_REV_ASSOCIATION, path)); associations(assocs); } void ItemUpdater::addUpdateableAssociation(const std::string& path) { assocs.emplace_back(std::make_tuple(UPDATEABLE_FWD_ASSOCIATION, UPDATEABLE_REV_ASSOCIATION, path)); associations(assocs); } void ItemUpdater::removeAssociation(const std::string& path) { for (auto iter = assocs.begin(); iter != assocs.end();) { if ((std::get<2>(*iter)) == path) { iter = assocs.erase(iter); associations(assocs); } else { ++iter; } } } void ItemUpdater::onUpdateDone(const std::string& versionId, const std::string& psuInventoryPath) { // After update is done, remove old activation objects for (auto it = activations.begin(); it != activations.end(); ++it) { if (it->second->getVersionId() != versionId && utils::isAssociated(psuInventoryPath, it->second->associations())) { removePsuObject(psuInventoryPath); break; } } auto it = activations.find(versionId); assert(it != activations.end()); psuPathActivationMap.emplace(psuInventoryPath, it->second); } std::unique_ptr ItemUpdater::createActivationObject( const std::string& path, const std::string& versionId, const std::string& extVersion, Activation::Status activationStatus, const AssociationList& assocs, const std::string& filePath) { return std::make_unique(bus, path, versionId, extVersion, activationStatus, assocs, filePath, this, this); } void ItemUpdater::createPsuObject(const std::string& psuInventoryPath, const std::string& psuVersion) { auto versionId = utils::getVersionId(psuVersion); auto path = std::string(SOFTWARE_OBJPATH) + "/" + versionId; auto it = activations.find(versionId); if (it != activations.end()) { // The versionId is already created, associate the path auto associations = it->second->associations(); associations.emplace_back( std::make_tuple(ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION, psuInventoryPath)); it->second->associations(associations); psuPathActivationMap.emplace(psuInventoryPath, it->second); } else { // Create a new object for running PSU inventory AssociationList associations; auto activationState = Activation::Status::Active; associations.emplace_back( std::make_tuple(ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION, psuInventoryPath)); auto activation = createActivationObject( path, versionId, "", activationState, associations, ""); activations.emplace(versionId, std::move(activation)); psuPathActivationMap.emplace(psuInventoryPath, activations[versionId]); auto versionPtr = createVersionObject(path, versionId, psuVersion, VersionPurpose::PSU); versions.emplace(versionId, std::move(versionPtr)); createActiveAssociation(path); addFunctionalAssociation(path); addUpdateableAssociation(path); } } void ItemUpdater::removePsuObject(const std::string& psuInventoryPath) { auto it = psuPathActivationMap.find(psuInventoryPath); if (it == psuPathActivationMap.end()) { lg2::error("No Activation found for PSU {PSUPATH}", "PSUPATH", psuInventoryPath); return; } const auto& activationPtr = it->second; psuPathActivationMap.erase(psuInventoryPath); auto associations = activationPtr->associations(); for (auto iter = associations.begin(); iter != associations.end();) { if ((std::get<2>(*iter)) == psuInventoryPath) { iter = associations.erase(iter); } else { ++iter; } } if (associations.empty()) { // Remove the activation erase(activationPtr->getVersionId()); } else { // Update association activationPtr->associations(associations); } } void ItemUpdater::addPsuToStatusMap(const std::string& psuPath) { if (!psuStatusMap.contains(psuPath)) { psuStatusMap[psuPath] = {false, ""}; // Add PropertiesChanged listener for Item interface so we are notified // when Present property changes psuMatches.emplace_back( bus, MatchRules::propertiesChanged(psuPath, ITEM_IFACE), std::bind(&ItemUpdater::onPsuInventoryChangedMsg, this, std::placeholders::_1)); } } void ItemUpdater::handlePSUPresenceChanged(const std::string& psuPath) { if (psuStatusMap.contains(psuPath)) { if (psuStatusMap[psuPath].present) { // PSU is now present psuStatusMap[psuPath].model = utils::getModel(psuPath); auto version = utils::getVersion(psuPath); if (!version.empty() && !psuPathActivationMap.contains(psuPath)) { createPsuObject(psuPath, version); } } else { // PSU is now missing psuStatusMap[psuPath].model.clear(); if (psuPathActivationMap.contains(psuPath)) { removePsuObject(psuPath); } } } } std::unique_ptr ItemUpdater::createVersionObject( const std::string& objPath, const std::string& versionId, const std::string& versionString, sdbusplus::xyz::openbmc_project::Software::server::Version::VersionPurpose versionPurpose) { versionStrings.insert(versionString); auto version = std::make_unique( bus, objPath, versionId, versionString, versionPurpose, std::bind(&ItemUpdater::erase, this, std::placeholders::_1)); return version; } void ItemUpdater::onPsuInventoryChangedMsg(sdbusplus::message_t& msg) { using Interface = std::string; Interface interface; Properties properties; std::string psuPath = msg.get_path(); msg.read(interface, properties); onPsuInventoryChanged(psuPath, properties); } void ItemUpdater::onPsuInventoryChanged(const std::string& psuPath, const Properties& properties) { try { if (psuStatusMap.contains(psuPath) && properties.contains(PRESENT)) { psuStatusMap[psuPath].present = std::get(properties.at(PRESENT)); handlePSUPresenceChanged(psuPath); if (psuStatusMap[psuPath].present) { // Check if there are new PSU images to update processStoredImage(); syncToLatestImage(); } } } catch (const std::exception& e) { lg2::error( "Unable to handle inventory PropertiesChanged event: {ERROR}", "ERROR", e); } } void ItemUpdater::processPSUImage() { try { auto paths = utils::getPSUInventoryPaths(bus); for (const auto& p : paths) { try { addPsuToStatusMap(p); auto service = utils::getService(bus, p.c_str(), ITEM_IFACE); psuStatusMap[p].present = utils::getProperty( bus, service.c_str(), p.c_str(), ITEM_IFACE, PRESENT); handlePSUPresenceChanged(p); } catch (const std::exception& e) { // Ignore errors; the information might not be available yet } } } catch (const std::exception& e) { // Ignore errors; the information might not be available yet } } void ItemUpdater::processStoredImage() { scanDirectory(IMG_DIR_BUILTIN); if (!ALWAYS_USE_BUILTIN_IMG_DIR) { scanDirectory(IMG_DIR_PERSIST); } } void ItemUpdater::scanDirectory(const fs::path& dir) { auto manifest = dir; auto path = dir; // The directory shall put PSU images in directories named with model if (!fs::exists(dir)) { // Skip return; } if (!fs::is_directory(dir)) { lg2::error("The path is not a directory: {PATH}", "PATH", dir); return; } for (const auto& [key, item] : psuStatusMap) { if (!item.model.empty()) { path = path / item.model; manifest = dir / item.model / MANIFEST_FILE; break; } } if (path == dir) { lg2::error("Model directory not found"); return; } if (!fs::is_directory(path)) { lg2::error("The path is not a directory: {PATH}", "PATH", path); return; } if (!fs::exists(manifest)) { lg2::error("No MANIFEST found at {PATH}", "PATH", manifest); return; } // If the model in manifest does not match the dir name // Log a warning if (fs::is_regular_file(manifest)) { auto ret = Version::getValues( manifest.string(), {MANIFEST_VERSION, MANIFEST_EXTENDED_VERSION}); auto version = ret[MANIFEST_VERSION]; auto extVersion = ret[MANIFEST_EXTENDED_VERSION]; auto info = Version::getExtVersionInfo(extVersion); auto model = info["model"]; if (path.stem() != model) { lg2::error("Unmatched model: path={PATH}, model={MODEL}", "PATH", path, "MODEL", model); } else { auto versionId = utils::getVersionId(version); auto it = activations.find(versionId); if (it == activations.end()) { // This is a version that is different than the running PSUs auto activationState = Activation::Status::Ready; auto purpose = VersionPurpose::PSU; auto objPath = std::string(SOFTWARE_OBJPATH) + "/" + versionId; auto activation = createActivationObject( objPath, versionId, extVersion, activationState, {}, path); activations.emplace(versionId, std::move(activation)); auto versionPtr = createVersionObject(objPath, versionId, version, purpose); versions.emplace(versionId, std::move(versionPtr)); } else { // This is a version that a running PSU is using, set the path // on the version object it->second->path(path); } } } else { lg2::error("MANIFEST is not a file: {PATH}", "PATH", manifest); } } std::optional ItemUpdater::getLatestVersionId() { std::string latestVersion; if (ALWAYS_USE_BUILTIN_IMG_DIR) { latestVersion = getFWVersionFromBuiltinDir(); } else { latestVersion = utils::getLatestVersion(versionStrings); } if (latestVersion.empty()) { return {}; } std::optional versionId; for (const auto& v : versions) { if (v.second->version() == latestVersion) { versionId = v.first; break; } } assert(versionId.has_value()); return versionId; } void ItemUpdater::syncToLatestImage() { auto latestVersionId = getLatestVersionId(); if (!latestVersionId) { return; } const auto& it = activations.find(*latestVersionId); assert(it != activations.end()); const auto& activation = it->second; const auto& assocs = activation->associations(); auto paths = utils::getPSUInventoryPaths(bus); for (const auto& p : paths) { // If there is a present PSU that is not associated with the latest // image, run the activation so that all PSUs are running the same // latest image. if (psuStatusMap.contains(p) && psuStatusMap[p].present) { if (!utils::isAssociated(p, assocs)) { lg2::info("Automatically update PSUs to version {VERSION_ID}", "VERSION_ID", *latestVersionId); invokeActivation(activation); break; } } } } void ItemUpdater::invokeActivation( const std::unique_ptr& activation) { activation->requestedActivation(Activation::RequestedActivations::Active); } void ItemUpdater::onPSUInterfaceAdded(sdbusplus::message_t& msg) { // Maintain static set of valid PSU paths. This is needed if PSU interface // comes in a separate InterfacesAdded message from Item interface. static std::set psuPaths{}; try { sdbusplus::message::object_path objPath; std::map>> interfaces; msg.read(objPath, interfaces); std::string path = objPath.str; if (interfaces.contains(PSU_INVENTORY_IFACE)) { psuPaths.insert(path); } if (interfaces.contains(ITEM_IFACE) && psuPaths.contains(path) && !psuStatusMap.contains(path)) { auto interface = interfaces[ITEM_IFACE]; if (interface.contains(PRESENT)) { addPsuToStatusMap(path); psuStatusMap[path].present = std::get(interface[PRESENT]); handlePSUPresenceChanged(path); if (psuStatusMap[path].present) { // Check if there are new PSU images to update processStoredImage(); syncToLatestImage(); } } } } catch (const std::exception& e) { lg2::error("Unable to handle inventory InterfacesAdded event: {ERROR}", "ERROR", e); } } void ItemUpdater::processPSUImageAndSyncToLatest() { processPSUImage(); processStoredImage(); syncToLatestImage(); } std::string ItemUpdater::getFWVersionFromBuiltinDir() { std::string version; for (const auto& activation : activations) { if (activation.second->path().starts_with(IMG_DIR_BUILTIN)) { std::string versionId = activation.second->getVersionId(); auto it = versions.find(versionId); if (it != versions.end()) { const auto& versionPtr = it->second; version = versionPtr->version(); break; } } } return version; } } // namespace updater } // namespace software } // namespace phosphor