blob: f8138868644b8d90a0543a41fdefb0606c93a08b [file] [log] [blame]
#pragma once
#include "config.h"
#include "flash.hpp"
#include "utils.hpp"
#include "xyz/openbmc_project/Software/ActivationProgress/server.hpp"
#include "xyz/openbmc_project/Software/RedundancyPriority/server.hpp"
#include <sdbusplus/server.hpp>
#include <xyz/openbmc_project/Association/Definitions/server.hpp>
#include <xyz/openbmc_project/Software/Activation/server.hpp>
#include <xyz/openbmc_project/Software/ActivationBlocksTransition/server.hpp>
#ifdef WANT_SIGNATURE_VERIFY
#include <filesystem>
#endif
namespace phosphor
{
namespace software
{
namespace updater
{
#ifdef WANT_SIGNATURE_VERIFY
namespace fs = std::filesystem;
#endif
using AssociationList =
std::vector<std::tuple<std::string, std::string, std::string>>;
using ActivationInherit = sdbusplus::server::object_t<
sdbusplus::server::xyz::openbmc_project::software::Activation,
sdbusplus::server::xyz::openbmc_project::association::Definitions>;
using ActivationBlocksTransitionInherit =
sdbusplus::server::object_t<sdbusplus::server::xyz::openbmc_project::
software::ActivationBlocksTransition>;
using RedundancyPriorityInherit = sdbusplus::server::object_t<
sdbusplus::server::xyz::openbmc_project::software::RedundancyPriority>;
using ActivationProgressInherit = sdbusplus::server::object_t<
sdbusplus::server::xyz::openbmc_project::software::ActivationProgress>;
constexpr auto applyTimeImmediate =
"xyz.openbmc_project.Software.ApplyTime.RequestedApplyTimes.Immediate";
constexpr auto applyTimeIntf = "xyz.openbmc_project.Software.ApplyTime";
constexpr auto dbusPropIntf = "org.freedesktop.DBus.Properties";
constexpr auto applyTimeObjPath = "/xyz/openbmc_project/software/apply_time";
constexpr auto applyTimeProp = "RequestedApplyTime";
namespace sdbusRule = sdbusplus::bus::match::rules;
class ItemUpdater;
class Activation;
class RedundancyPriority;
/** @class RedundancyPriority
* @brief OpenBMC RedundancyPriority implementation
* @details A concrete implementation for
* xyz.openbmc_project.Software.RedundancyPriority DBus API.
*/
class RedundancyPriority : public RedundancyPriorityInherit
{
public:
/** @brief Constructs RedundancyPriority.
*
* @param[in] bus - The Dbus bus object
* @param[in] path - The Dbus object path
* @param[in] parent - Parent object.
* @param[in] value - The redundancyPriority value
* @param[in] freePriority - Call freePriorioty, default to true
*/
RedundancyPriority(sdbusplus::bus_t& bus, const std::string& path,
Activation& parent, uint8_t value,
bool freePriority = true) :
RedundancyPriorityInherit(bus, path.c_str(),
action::emit_interface_added),
parent(parent)
{
// Set Property
if (freePriority)
{
priority(value);
}
else
{
sdbusPriority(value);
}
}
/** @brief Overridden Priority property set function, calls freePriority
* to bump the duplicated priority values.
*
* @param[in] value - uint8_t
*
* @return Success or exception thrown
*/
uint8_t priority(uint8_t value) override;
/** @brief Non-Overriden Priority property set function
*
* @param[in] value - uint8_t
*
* @return Success or exception thrown
*/
uint8_t sdbusPriority(uint8_t value);
/** @brief Priority property get function
*
* @returns uint8_t - The Priority value
*/
using RedundancyPriorityInherit::priority;
/** @brief Parent Object. */
Activation& parent;
};
/** @class ActivationBlocksTransition
* @brief OpenBMC ActivationBlocksTransition implementation.
* @details A concrete implementation for
* xyz.openbmc_project.Software.ActivationBlocksTransition DBus API.
*/
class ActivationBlocksTransition : public ActivationBlocksTransitionInherit
{
public:
/** @brief Constructs ActivationBlocksTransition.
*
* @param[in] bus - The Dbus bus object
* @param[in] path - The Dbus object path
*/
ActivationBlocksTransition(sdbusplus::bus_t& bus, const std::string& path) :
ActivationBlocksTransitionInherit(bus, path.c_str(),
action::emit_interface_added),
bus(bus)
{
enableRebootGuard();
}
~ActivationBlocksTransition()
{
disableRebootGuard();
}
private:
sdbusplus::bus_t& bus;
/** @brief Enables a Guard that blocks any BMC reboot commands */
void enableRebootGuard();
/** @brief Disables any guard that was blocking the BMC reboot */
void disableRebootGuard();
};
class ActivationProgress : public ActivationProgressInherit
{
public:
/** @brief Constructs ActivationProgress.
*
* @param[in] bus - The Dbus bus object
* @param[in] path - The Dbus object path
*/
ActivationProgress(sdbusplus::bus_t& bus, const std::string& path) :
ActivationProgressInherit(bus, path.c_str(),
action::emit_interface_added)
{
progress(0);
}
};
/** @class Activation
* @brief OpenBMC activation software management implementation.
* @details A concrete implementation for
* xyz.openbmc_project.Software.Activation DBus API.
*/
class Activation : public ActivationInherit, public Flash
{
public:
/** @brief Constructs Activation Software Manager
*
* @param[in] bus - The Dbus bus object
* @param[in] path - The Dbus object path
* @param[in] parent - Parent object.
* @param[in] versionId - The software version id
* @param[in] activationStatus - The status of Activation
* @param[in] assocs - Association objects
*/
Activation(sdbusplus::bus_t& bus, const std::string& path,
ItemUpdater& parent, std::string& versionId,
sdbusplus::server::xyz::openbmc_project::software::Activation::
Activations activationStatus,
AssociationList& assocs) :
ActivationInherit(bus, path.c_str(),
ActivationInherit::action::defer_emit),
bus(bus), path(path), parent(parent), versionId(versionId),
systemdSignals(
bus,
sdbusRule::type::signal() + sdbusRule::member("JobRemoved") +
sdbusRule::path("/org/freedesktop/systemd1") +
sdbusRule::interface("org.freedesktop.systemd1.Manager"),
std::bind(std::mem_fn(&Activation::unitStateChange), this,
std::placeholders::_1))
{
// Set Properties.
activation(activationStatus);
associations(assocs);
// Emit deferred signal.
emit_object_added();
}
/** @brief Overloaded Activation property setter function
*
* @param[in] value - One of Activation::Activations
*
* @return Success or exception thrown
*/
Activations activation(Activations value) override;
/** @brief Activation */
using ActivationInherit::activation;
/** @brief Overloaded requestedActivation property setter function
*
* @param[in] value - One of Activation::RequestedActivations
*
* @return Success or exception thrown
*/
RequestedActivations
requestedActivation(RequestedActivations value) override;
/** @brief Overloaded write flash function */
void flashWrite() override;
/**
* @brief Handle the success of the flashWrite() function
*
* @details Perform anything that is necessary to mark the activation
* successful after the image has been written to flash. Sets the Activation
* value to Active.
*/
void onFlashWriteSuccess();
#ifdef HOST_BIOS_UPGRADE
/* @brief write to Host flash function */
void flashWriteHost();
/** @brief Function that acts on Bios upgrade service file state changes */
void onStateChangesBios(sdbusplus::message_t&);
#endif
/** @brief Overloaded function that acts on service file state changes */
void onStateChanges(sdbusplus::message_t&) override;
/** @brief Check if systemd state change is relevant to this object
*
* Instance specific interface to handle the detected systemd state
* change
*
* @param[in] msg - Data associated with subscribed signal
*
*/
void unitStateChange(sdbusplus::message_t& msg);
/**
* @brief subscribe to the systemd signals
*
* This object needs to capture when it's systemd targets complete
* so it can keep it's state updated
*
*/
void subscribeToSystemdSignals();
/**
* @brief unsubscribe from the systemd signals
*
* systemd signals are only of interest during the activation process.
* Once complete, we want to unsubscribe to avoid unnecessary calls of
* unitStateChange().
*
*/
void unsubscribeFromSystemdSignals();
/**
* @brief Deletes the version from Image Manager and the
* untar image from image upload dir.
*/
void deleteImageManagerObject();
/**
* @brief Determine the configured image apply time value
*
* @return true if the image apply time value is immediate
**/
bool checkApplyTimeImmediate();
/**
* @brief Reboot the BMC. Called when ApplyTime is immediate.
*
* @return none
**/
void rebootBmc();
/** @brief Persistent sdbusplus DBus bus connection */
sdbusplus::bus_t& bus;
/** @brief Persistent DBus object path */
std::string path;
/** @brief Parent Object. */
ItemUpdater& parent;
/** @brief Version id */
std::string versionId;
/** @brief Persistent ActivationBlocksTransition dbus object */
std::unique_ptr<ActivationBlocksTransition> activationBlocksTransition;
/** @brief Persistent RedundancyPriority dbus object */
std::unique_ptr<RedundancyPriority> redundancyPriority;
/** @brief Persistent ActivationProgress dbus object */
std::unique_ptr<ActivationProgress> activationProgress;
/** @brief Used to subscribe to dbus systemd signals **/
sdbusplus::bus::match_t systemdSignals;
/** @brief Tracks whether the read-write volume has been created as
* part of the activation process. **/
bool rwVolumeCreated = false;
/** @brief Tracks whether the read-only volume has been created as
* part of the activation process. **/
bool roVolumeCreated = false;
/** @brief Tracks if the service that updates the U-Boot environment
* variables has completed. **/
bool ubootEnvVarsUpdated = false;
#ifdef WANT_SIGNATURE_VERIFY
private:
/** @brief Verify signature of the images.
*
* @param[in] imageDir - The path of images to verify
* @param[in] confDir - The path of configs for verification
*
* @return true if verification successful and false otherwise
*/
bool verifySignature(const fs::path& imageDir, const fs::path& confDir);
/** @brief Called when image verification fails. */
void onVerifyFailed();
#endif
};
} // namespace updater
} // namespace software
} // namespace phosphor