blob: fac41f30be56c56bca0b3eedf0e66683cc7b4b7a [file] [log] [blame]
#pragma once
#include "activation.hpp"
#include "item_updater_helper.hpp"
#include "msl_verify.hpp"
#include "update_manager.hpp"
#include "version.hpp"
#include "xyz/openbmc_project/Collection/DeleteAll/server.hpp"
#include <sdbusplus/async.hpp>
#include <sdbusplus/server.hpp>
#include <xyz/openbmc_project/Association/Definitions/server.hpp>
#include <xyz/openbmc_project/Common/FactoryReset/server.hpp>
#include <xyz/openbmc_project/Control/FieldMode/server.hpp>
#include <xyz/openbmc_project/Software/MinimumVersion/server.hpp>
#include <string>
#include <vector>
namespace phosphor
{
namespace software
{
namespace updater
{
using ActivationIntf =
sdbusplus::xyz::openbmc_project::Software::server::Activation;
using ItemUpdaterInherit = sdbusplus::server::object_t<
sdbusplus::server::xyz::openbmc_project::common::FactoryReset,
sdbusplus::server::xyz::openbmc_project::control::FieldMode,
sdbusplus::server::xyz::openbmc_project::association::Definitions,
sdbusplus::server::xyz::openbmc_project::collection::DeleteAll>;
using MinimumVersionInherit = sdbusplus::server::object_t<
sdbusplus::server::xyz::openbmc_project::software::MinimumVersion>;
namespace MatchRules = sdbusplus::bus::match::rules;
using VersionClass = phosphor::software::manager::Version;
using AssociationList =
std::vector<std::tuple<std::string, std::string, std::string>>;
using UpdateManager = phosphor::software::update::Manager;
/** @class MinimumVersion
* @brief OpenBMC MinimumVersion implementation.
* @details A concrete implementation for
* xyz.openbmc_project.Software.MinimumVersion DBus API.
*/
class MinimumVersion : public MinimumVersionInherit
{
public:
/** @brief Constructs MinimumVersion
*
* @param[in] bus - The D-Bus bus object
* @param[in] path - The D-bus object path
*/
MinimumVersion(sdbusplus::bus_t& bus, const std::string& path) :
MinimumVersionInherit(bus, path.c_str(), action::emit_interface_added)
{}
};
/** @class ItemUpdater
* @brief Manages the activation of the BMC version items.
*/
class ItemUpdater : public ItemUpdaterInherit
{
public:
/*
* @brief Types of Activation status for image validation.
*/
enum class ActivationStatus
{
ready,
invalid,
active
};
/** @brief Types of Updater. */
enum class UpdaterType
{
BMC,
BIOS,
ALL
};
/** @brief Constructs ItemUpdater
*
* @param[in] bus - The D-Bus bus object
*/
ItemUpdater(sdbusplus::async::context& ctx, const std::string& path,
UpdaterType type = UpdaterType::ALL,
bool useUpdateDBusInterface = true) :
ItemUpdaterInherit(ctx.get_bus(), path.c_str(),
ItemUpdaterInherit::action::defer_emit),
useUpdateDBusInterface(useUpdateDBusInterface), ctx(ctx),
bus(ctx.get_bus()), helper(bus)
{
if (!useUpdateDBusInterface)
{
versionMatch = std::make_unique<sdbusplus::bus::match_t>(
bus,
MatchRules::interfacesAdded() +
MatchRules::path("/xyz/openbmc_project/software"),
std::bind(std::mem_fn(&ItemUpdater::createActivation), this,
std::placeholders::_1));
}
getRunningSlot();
setBMCInventoryPath();
if (type == UpdaterType::BMC || type == UpdaterType::ALL)
{
processBMCImage();
}
if (type == UpdaterType::BIOS || type == UpdaterType::ALL)
{
#ifdef HOST_BIOS_UPGRADE
createBIOSObject();
#endif
}
restoreFieldModeStatus();
emit_object_added();
};
/** @brief Save priority value to persistent storage (flash and optionally
* a U-Boot environment variable)
*
* @param[in] versionId - The Id of the version
* @param[in] value - The priority value
* @return None
*/
void savePriority(const std::string& versionId, uint8_t value);
/** @brief Sets the given priority free by incrementing
* any existing priority with the same value by 1
*
* @param[in] value - The priority that needs to be set free.
* @param[in] versionId - The Id of the version for which we
* are trying to free up the priority.
* @return None
*/
void freePriority(uint8_t value, const std::string& versionId);
/**
* @brief Create and populate the active BMC Version.
*/
void processBMCImage();
/**
* @brief Verifies the image at filepath and creates the version and
* activation object. In case activation object already exists for the
* specified id, update the activation status based on image verification.
* @param[in] id - The unique identifier for the update.
* @param[in] path - The object path for the relevant objects.
* @param[in] version - The version of the image.
* @param[in] purpose - The purpose of the image.
* @param[in] extendedVersion The extended version of the image.
* @param[in] filePath - The file path where the image is located.
* @param[in] compatibleNames - The compatible name for the image.
* @param[out] Activations - Whether the image is ready to activate or not.
*/
ActivationIntf::Activations verifyAndCreateObjects(
std::string& id, std::string& path, std::string& version,
VersionClass::VersionPurpose purpose, std::string& extendedVersion,
std ::string& filePath, std::vector<std::string>& compatibleNames);
/**
* @brief Creates the activation object
* @param[in] id - The unique identifier for the update.
* @param[in] path - The object path for the activation object.
* @param[in] applyTime - The apply time for the image
*/
void createActivationWithApplyTime(
std::string& id, std::string& path,
ApplyTimeIntf::RequestedApplyTimes applyTime);
/**
* @brief Request the activation for the specified update.
* @param[in] id - The unique identifier for the update.
* @param[out] bool - status for the action.
*/
bool requestActivation(std::string& id);
/**
* @brief Change the activation status for the specified update.
* @param[in] id - The unique identifier for the update.
* @param[in] status - The activation status to set.
* @param[out] bool - status for the action.
*/
bool updateActivationStatus(std::string& id,
ActivationIntf::Activations status);
/**
* @brief Create the Update object
* @param[in] id - The unique identifier for the update.
* @param[in] path - The object path for the update object.
*/
void createUpdateObject(const std::string& id, const std::string& path);
/**
* @brief Erase specified entry D-Bus object
* if Action property is not set to Active
*
* @param[in] entryId - unique identifier of the entry
*/
void erase(std::string entryId);
/**
* @brief Deletes all versions except for the current one
*/
void deleteAll() override;
/** @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);
/** @brief Removes the associations from the provided software image path
*
* @param[in] path - The path to remove the associations from.
*/
void removeAssociations(const std::string& path);
/** @brief Determine if the given priority is the lowest
*
* @param[in] value - The priority that needs to be checked.
*
* @return boolean corresponding to whether the given
* priority is lowest.
*/
bool isLowestPriority(uint8_t value);
/**
* @brief Updates the U-Boot variables to point to the requested
* versionId, so that the systems boots from this version on
* the next reboot.
*
* @param[in] versionId - The version to point the system to boot from.
*/
void updateUbootEnvVars(const std::string& versionId);
/**
* @brief Updates the uboot variables to point to BMC version with lowest
* priority, so that the system boots from this version on the
* next boot.
*/
void resetUbootEnvVars();
/** @brief Brings the total number of active BMC versions to
* ACTIVE_BMC_MAX_ALLOWED -1. This function is intended to be
* run before activating a new BMC version. If this function
* needs to delete any BMC version(s) it will delete the
* version(s) with the highest priority, skipping the
* functional BMC version.
*
* @param[in] caller - The Activation object that called this function.
*/
void freeSpace(const Activation& caller);
/** @brief Creates a updateable association to the
* "running" BMC software image
*
* @param[in] path - The path to create the association.
*/
void createUpdateableAssociation(const std::string& path);
/** @brief Persistent map of Version D-Bus objects and their
* version id */
std::map<std::string, std::unique_ptr<VersionClass>> versions;
/** @brief Vector of needed BMC images in the tarball*/
std::vector<std::string> imageUpdateList;
/** @brief The slot of running BMC image */
uint32_t runningImageSlot = 0;
/** @brief Flag to indicate if the update interface is used or not */
bool useUpdateDBusInterface;
private:
/** @brief Callback function for Software.Version match.
* @details Creates an Activation D-Bus object.
*
* @param[in] msg - Data associated with subscribed signal
*/
void createActivation(sdbusplus::message_t& msg);
/**
* @brief Validates the presence of SquashFS image in the image dir.
*
* @param[in] filePath - The path to the image dir.
* @param[out] result - ActivationStatus Enum.
* ready if validation was successful.
* invalid if validation fail.
* active if image is the current version.
*
*/
ActivationStatus validateSquashFSImage(const std::string& filePath);
/** @brief BMC factory reset - marks the read-write partition for
* recreation upon reboot. */
void reset() override;
/**
* @brief Enables field mode, if value=true.
*
* @param[in] value - If true, enables field mode.
* @param[out] result - Returns the current state of field mode.
*
*/
bool fieldModeEnabled(bool value) override;
/** @brief Sets the BMC inventory item path under
* /xyz/openbmc_project/inventory/system/chassis/. */
void setBMCInventoryPath();
/** @brief The path to the BMC inventory item. */
std::string bmcInventoryPath;
/** @brief Restores field mode status on reboot. */
void restoreFieldModeStatus();
/** @brief Creates a functional association to the
* "running" BMC software image
*
* @param[in] path - The path to create the association to.
*/
void createFunctionalAssociation(const std::string& path);
/** @brief D-Bus context */
sdbusplus::async::context& ctx;
/** @brief Persistent sdbusplus D-Bus bus connection. */
sdbusplus::bus_t& bus;
/** @brief The helper of image updater. */
Helper helper;
/** @brief Persistent map of Activation D-Bus objects and their
* version id */
std::map<std::string, std::unique_ptr<Activation>> activations;
/** @brief sdbusplus signal match for Software.Version */
std::unique_ptr<sdbusplus::bus::match_t> versionMatch;
/** @brief This entry's associations */
AssociationList assocs;
/** @brief Clears read only partition for
* given Activation D-Bus object.
*
* @param[in] versionId - The version id.
*/
void removeReadOnlyPartition(const std::string& versionId);
/** @brief Copies U-Boot from the currently booted BMC chip to the
* alternate chip.
*/
void mirrorUbootToAlt();
/** @brief Check the required image files
*
* @param[in] filePath - BMC tarball file path
* @param[in] imageList - Image filenames included in the BMC tarball
* @param[out] result - Boolean
* true if all image files are found in BMC tarball
* false if one of image files is missing
*/
static bool checkImage(const std::string& filePath,
const std::vector<std::string>& imageList);
/** @brief Persistent MinimumVersion D-Bus object */
std::unique_ptr<MinimumVersion> minimumVersionObject;
/** @brief Persistent map of Update D-Bus objects and their SwIds */
std::map<std::string, std::unique_ptr<UpdateManager>> updateManagers;
#ifdef HOST_BIOS_UPGRADE
/** @brief Create the BIOS object without knowing the version.
*
* The object is created only to provide the DBus access so that an
* external service could set the correct BIOS version.
* On BIOS code update, the version is updated accordingly.
*/
void createBIOSObject();
/** @brief Persistent Activation D-Bus object for BIOS */
std::unique_ptr<Activation> biosActivation;
public:
/** @brief Persistent Version D-Bus object for BIOS */
std::unique_ptr<VersionClass> biosVersion;
#endif
/** @brief Get the slot number of running image */
void getRunningSlot();
};
} // namespace updater
} // namespace software
} // namespace phosphor