#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 #include #include #include #include #ifdef WANT_SIGNATURE_VERIFY #include #endif namespace phosphor { namespace software { namespace updater { #ifdef WANT_SIGNATURE_VERIFY namespace fs = std::filesystem; #endif using AssociationList = std::vector>; 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; 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>; using ApplyTimeIntf = sdbusplus::common::xyz::openbmc_project::software::ApplyTime; 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() override { disableRebootGuard(); } ActivationBlocksTransition(const ActivationBlocksTransition&) = delete; ActivationBlocksTransition& operator=(const ActivationBlocksTransition&) = delete; ActivationBlocksTransition(ActivationBlocksTransition&&) = delete; ActivationBlocksTransition& operator=(ActivationBlocksTransition&&) = delete; 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& /*msg*/); #endif /** @brief Overloaded function that acts on service file state changes */ void onStateChanges(sdbusplus::message_t& /*msg*/) 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; /** @brief Persistent RedundancyPriority dbus object */ std::unique_ptr redundancyPriority; /** @brief Persistent ActivationProgress dbus object */ std::unique_ptr activationProgress; /** @brief Apply time */ ApplyTimeIntf::RequestedApplyTimes applyTime; /** @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 */ static 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