Manager: Support for switching running image
ActiveSoftwareImage was added to Manager in v1_6_0.
It is a Readwrite property.
This supports only switching the BMC running image.
This assumes the BMC updater is
xyz.openbmc_project.Software.BMC.Updater.
This assumes the switch is Immediate and calls a BMC
reboot after setting the priority to 0.
Patch set 1 had a new Redfish.Settings for the Manager at
/redfish/v1/Managers/bmc/Settings and a new setting,
ActiveSoftwareImageApplyTime.
This was removed due to complexity.
A future commit could add support to allow this Setting to be
exposed and PATCHed.
From https://redfish.dmtf.org/schemas/v1/Manager.v1_9_0.json
"ActiveSoftwareImage": {
...
"description": "The link to the software inventory resource that
represents the active firmware image for this manager.",
"longDescription": "This property shall contain a link to a resource
of type SoftwareInventory that represents the active firmware image for
this manager.",
"readonly": false,
"versionAdded": "v1_6_0"
Tested: Validator passes
curl -k -X PATCH -d \
'{ "Links": { "ActiveSoftwareImage": { "@odata.id":"/redfish/v1/UpdateService/FirmwareInventory/82d3ec86"}}}' \
https://$bmc/redfish/v1/Managers/bmc
{
"@Message.ExtendedInfo": [
{
"@odata.type": "#Message.v1_0_0.Message",
"Message": "Successfully Completed Request",
Change-Id: Id9d29af5332f9e76ec172e1e4351980a6772477c
Signed-off-by: Gunnar Mills <gmills@us.ibm.com>
diff --git a/redfish-core/lib/managers.hpp b/redfish-core/lib/managers.hpp
index 7832e81..b65f89d 100644
--- a/redfish-core/lib/managers.hpp
+++ b/redfish-core/lib/managers.hpp
@@ -25,6 +25,7 @@
#include <utils/fw_utils.hpp>
#include <utils/systemd_utils.hpp>
+#include <cstdint>
#include <memory>
#include <sstream>
#include <variant>
@@ -1808,11 +1809,12 @@
const std::vector<std::string>& params) override
{
std::optional<nlohmann::json> oem;
+ std::optional<nlohmann::json> links;
std::optional<std::string> datetime;
std::shared_ptr<AsyncResp> response = std::make_shared<AsyncResp>(res);
if (!json_util::readJson(req, response->res, "Oem", oem, "DateTime",
- datetime))
+ datetime, "Links", links))
{
return;
}
@@ -1843,6 +1845,29 @@
}
}
}
+ if (links)
+ {
+ std::optional<nlohmann::json> activeSoftwareImage;
+ if (!redfish::json_util::readJson(
+ *links, res, "ActiveSoftwareImage", activeSoftwareImage))
+ {
+ return;
+ }
+ if (activeSoftwareImage)
+ {
+ std::optional<std::string> odataId;
+ if (!json_util::readJson(*activeSoftwareImage, res, "@odata.id",
+ odataId))
+ {
+ return;
+ }
+
+ if (odataId)
+ {
+ setActiveFirmwareImage(response, std::move(*odataId));
+ }
+ }
+ }
if (datetime)
{
setDateTime(response, std::move(*datetime));
@@ -1884,6 +1909,116 @@
"xyz.openbmc_project.State.BMC", "LastRebootTime");
}
+ /**
+ * @brief Set the running firmware image
+ *
+ * @param[i,o] aResp - Async response object
+ * @param[i] runningFirmwareTarget - Image to make the running image
+ *
+ * @return void
+ */
+ void setActiveFirmwareImage(std::shared_ptr<AsyncResp> aResp,
+ const std::string&& runningFirmwareTarget)
+ {
+ // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id>
+ std::string::size_type idPos = runningFirmwareTarget.rfind("/");
+ if (idPos == std::string::npos)
+ {
+ messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
+ "@odata.id");
+ BMCWEB_LOG_DEBUG << "Can't parse firmware ID!";
+ return;
+ }
+ idPos++;
+ if (idPos >= runningFirmwareTarget.size())
+ {
+ messages::propertyValueNotInList(aResp->res, runningFirmwareTarget,
+ "@odata.id");
+ BMCWEB_LOG_DEBUG << "Invalid firmware ID.";
+ return;
+ }
+ std::string firmwareId = runningFirmwareTarget.substr(idPos);
+
+ // Make sure the image is valid before setting priority
+ crow::connections::systemBus->async_method_call(
+ [aResp, firmwareId,
+ runningFirmwareTarget](const boost::system::error_code ec,
+ ManagedObjectType& subtree) {
+ if (ec)
+ {
+ BMCWEB_LOG_DEBUG << "D-Bus response error getting objects.";
+ messages::internalError(aResp->res);
+ return;
+ }
+
+ if (subtree.size() == 0)
+ {
+ BMCWEB_LOG_DEBUG << "Can't find image!";
+ messages::internalError(aResp->res);
+ return;
+ }
+
+ bool foundImage = false;
+ for (auto& object : subtree)
+ {
+ const std::string& path =
+ static_cast<const std::string&>(object.first);
+ std::size_t idPos2 = path.rfind("/");
+
+ if (idPos2 == std::string::npos)
+ {
+ continue;
+ }
+
+ idPos2++;
+ if (idPos2 >= path.size())
+ {
+ continue;
+ }
+
+ if (path.substr(idPos2) == firmwareId)
+ {
+ foundImage = true;
+ break;
+ }
+ }
+
+ if (!foundImage)
+ {
+ messages::propertyValueNotInList(
+ aResp->res, runningFirmwareTarget, "@odata.id");
+ BMCWEB_LOG_DEBUG << "Invalid firmware ID.";
+ return;
+ }
+
+ BMCWEB_LOG_DEBUG << "Setting firmware version " + firmwareId +
+ " to priority 0.";
+
+ // Only support Immediate
+ // An addition could be a Redfish Setting like
+ // ActiveSoftwareImageApplyTime and support OnReset
+ crow::connections::systemBus->async_method_call(
+ [aResp](const boost::system::error_code ec) {
+ if (ec)
+ {
+ BMCWEB_LOG_DEBUG << "D-Bus response error setting.";
+ messages::internalError(aResp->res);
+ return;
+ }
+ doBMCGracefulRestart(aResp);
+ },
+
+ "xyz.openbmc_project.Software.BMC.Updater",
+ "/xyz/openbmc_project/software/" + firmwareId,
+ "org.freedesktop.DBus.Properties", "Set",
+ "xyz.openbmc_project.Software.RedundancyPriority",
+ "Priority", std::variant<uint8_t>(static_cast<uint8_t>(0)));
+ },
+ "xyz.openbmc_project.Software.BMC.Updater",
+ "/xyz/openbmc_project/software",
+ "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
+ }
+
void setDateTime(std::shared_ptr<AsyncResp> aResp,
std::string datetime) const
{