blob: e4d7b99f5768dc5ffe3dc3f6fa72bf9ed3294a05 [file] [log] [blame] [edit]
#pragma once
#include "config.h"
#include "activation.hpp"
#include "association_interface.hpp"
#include "types.hpp"
#include "utils.hpp"
#include "version.hpp"
#include <phosphor-logging/log.hpp>
#include <sdbusplus/server.hpp>
#include <xyz/openbmc_project/Association/Definitions/server.hpp>
#include <xyz/openbmc_project/Collection/DeleteAll/server.hpp>
#include <filesystem>
#include <map>
#include <string>
#include <variant>
#include <vector>
class TestItemUpdater;
namespace phosphor
{
namespace software
{
namespace updater
{
class Version;
using ItemUpdaterInherit = sdbusplus::server::object_t<
sdbusplus::xyz::openbmc_project::Association::server::Definitions>;
namespace MatchRules = sdbusplus::bus::match::rules;
namespace fs = std::filesystem;
/** @class ItemUpdater
* @brief Manages the activation of the PSU version items.
*/
class ItemUpdater :
public ItemUpdaterInherit,
public AssociationInterface,
public ActivationListener
{
friend class ::TestItemUpdater;
public:
/** @brief Constructs ItemUpdater
*
* @param[in] bus - The D-Bus bus object
* @param[in] path - The D-Bus path
*/
ItemUpdater(sdbusplus::bus_t& bus, const std::string& path) :
ItemUpdaterInherit(bus, path.c_str()), bus(bus),
versionMatch(
bus,
MatchRules::interfacesAdded() + MatchRules::path(SOFTWARE_OBJPATH),
std::bind(std::mem_fn(&ItemUpdater::onVersionInterfacesAddedMsg),
this, std::placeholders::_1)),
psuInterfaceMatch(
bus,
MatchRules::interfacesAdded() +
MatchRules::path("/xyz/openbmc_project/inventory") +
MatchRules::sender("xyz.openbmc_project.Inventory.Manager"),
std::bind(std::mem_fn(&ItemUpdater::onPSUInterfacesAdded), this,
std::placeholders::_1))
{
processPSUImageAndSyncToLatest();
}
/** @brief Deletes version
*
* @param[in] versionId - Id of the version to delete
*/
void erase(const std::string& versionId);
/** @brief Creates an active association to the
* newly active software image
*
* @param[in] path - The path to create the association to.
*/
void createActiveAssociation(const std::string& path) override;
/** @brief Add the functional association to the
* new "running" PSU images
*
* @param[in] path - The path to add the association to.
*/
void addFunctionalAssociation(const std::string& path) override;
/** @brief Add the updateable association to the
* "running" PSU software image
*
* @param[in] path - The path to create the association.
*/
void addUpdateableAssociation(const std::string& path) override;
/** @brief Removes the associations from the provided software image path
*
* @param[in] path - The path to remove the association from.
*/
void removeAssociation(const std::string& path) override;
/** @brief Notify a PSU is updated
*
* @param[in] versionId - The versionId of the activation
* @param[in] psuInventoryPath - The PSU inventory path that is updated
*/
void onUpdateDone(const std::string& versionId,
const std::string& psuInventoryPath) override;
private:
using Properties =
std::map<std::string, utils::UtilsInterface::PropertyType>;
using InterfacesAddedMap =
std::map<std::string,
std::map<std::string, std::variant<bool, std::string>>>;
/** @brief Callback function for Software.Version match.
*
* @param[in] msg - Data associated with subscribed signal
*/
void onVersionInterfacesAddedMsg(sdbusplus::message_t& msg);
/** @brief Called when new Software.Version interfaces are found
* @details Creates an Activation D-Bus object if appropriate
* Throws an exception if an error occurs.
*
* @param[in] path - D-Bus object path
* @param[in] interfaces - D-Bus interfaces that were added
*/
void onVersionInterfacesAdded(const std::string& path,
const InterfacesAddedMap& interfaces);
/** @brief Callback function for PSU inventory match.
*
* @param[in] msg - Data associated with subscribed signal
*/
void onPsuInventoryChangedMsg(sdbusplus::message_t& msg);
/** @brief Called when a PSU inventory object has changed
* @details Update an Activation D-Bus object for PSU inventory.
* Throws an exception if an error occurs.
*
* @param[in] psuPath - The PSU inventory path
* @param[in] properties - The updated properties
*/
void onPsuInventoryChanged(const std::string& psuPath,
const Properties& properties);
/** @brief Create Activation object */
std::unique_ptr<Activation> createActivationObject(
const std::string& path, const std::string& versionId,
const std::string& extVersion, Activation::Status activationStatus,
const AssociationList& assocs, const std::string& filePath);
/** @brief Create Version object */
std::unique_ptr<Version> createVersionObject(
const std::string& objPath, const std::string& versionId,
const std::string& versionString,
sdbusplus::xyz::openbmc_project::Software::server::Version::
VersionPurpose versionPurpose);
/** @brief Create Activation and Version object for PSU inventory
* @details If the same version exists for multiple PSUs, just add
* related association, instead of creating new objects.
* */
void createPsuObject(const std::string& psuInventoryPath,
const std::string& psuVersion);
/** @brief Remove Activation and Version object for PSU inventory
* @details If the same version exists for multiple PSUs, just remove
* related association.
* If the version has no association, the Activation and
* Version object will be removed
*/
void removePsuObject(const std::string& psuInventoryPath);
/** @brief Add PSU inventory path to the PSU status map
* @details Also adds a PropertiesChanged listener for the inventory path
* so we are notified when the Present property changes.
* Does nothing if the inventory path already exists in the map.
*
* @param[in] psuPath - The PSU inventory path
*/
void addPsuToStatusMap(const std::string& psuPath);
/** @brief Handle a change in presence for a PSU.
*
* @param[in] psuPath - The PSU inventory path
*/
void handlePSUPresenceChanged(const std::string& psuPath);
/**
* @brief Create and populate the active PSU Version.
*/
void processPSUImage();
/** @brief Create PSU Version from stored images */
void processStoredImage();
/** @brief Scan a directory and create PSU Version from stored images
* @details Throws an exception if an error occurs
*
* @param[in] dir Directory path to scan
*/
void scanDirectory(const fs::path& dir);
/** @brief Find the PSU model subdirectory within the specified directory
* @details Throws an exception if an error occurs
*
* @param[in] dir Directory path to search
*
* @return Subdirectory path, or an empty path if none found
*/
fs::path findModelDirectory(const fs::path& dir);
/** @brief Get the versionId of the latest PSU version */
std::optional<std::string> getLatestVersionId();
/** @brief Update PSUs to the latest version */
void syncToLatestImage();
/** @brief Invoke the activation via DBus */
static void invokeActivation(const std::unique_ptr<Activation>& activation);
/** @brief Callback function for interfaces added signal.
*
* This method is called when new interfaces are added. It updates the
* internal status map and processes the new PSU if it's present.
*
* @param[in] msg - Data associated with subscribed signal
*/
void onPSUInterfacesAdded(sdbusplus::message_t& msg);
/**
* @brief Handles the processing of PSU images.
*
* This function responsible for invoking the sequence of processing PSU
* images, processing stored images, and syncing to the latest firmware
* image.
*/
void processPSUImageAndSyncToLatest();
/** @brief Retrieve FW version from IMG_DIR_BUILTIN
*
* This function retrieves the firmware version from the PSU model directory
* that is in the IMG_DIR_BUILTIN. It loops through the activations map to
* find matching path starts with IMG_DIR_BUILTIN, then gets the
* corresponding version ID, and then looks it up in the versions map to
* retrieve the associated version string.
*/
std::string getFWVersionFromBuiltinDir();
/** @brief Persistent sdbusplus D-Bus bus connection. */
sdbusplus::bus_t& bus;
/** @brief Persistent map of Activation D-Bus objects and their
* version id */
std::map<std::string, std::unique_ptr<Activation>> activations;
/** @brief Persistent map of Version D-Bus objects and their
* version id */
std::map<std::string, std::unique_ptr<Version>> versions;
/** @brief The reference map of PSU Inventory objects and the
* Activation*/
std::map<std::string, const std::unique_ptr<Activation>&>
psuPathActivationMap;
/** @brief sdbusplus signal match for PSU Software*/
sdbusplus::bus::match_t versionMatch;
/** @brief sdbusplus signal matches for PSU Inventory */
std::vector<sdbusplus::bus::match_t> psuMatches;
/** @brief This entry's associations */
AssociationList assocs;
/** @brief A collection of the version strings */
std::set<std::string> versionStrings;
/** @brief A struct to hold the PSU present status and model */
struct psuStatus
{
bool present;
std::string model;
};
/** @brief The map of PSU inventory path and the psuStatus
*
* It is used to handle psu inventory changed event, that only create psu
* software object when a PSU is present and the model is retrieved */
std::map<std::string, psuStatus> psuStatusMap;
/** @brief Signal match for PSU interfaces added.
*
* This match listens for D-Bus signals indicating new interface has been
* added. When such a signal received, it triggers the
* `onInterfacesAdded` method to handle the new PSU.
*/
sdbusplus::bus::match_t psuInterfaceMatch;
};
} // namespace updater
} // namespace software
} // namespace phosphor