xref: /openbmc/openpower-pnor-code-mgmt/ubi/activation_ubi.cpp (revision f8e024296d02170b360c894349d900acfd940a4d)
19b21efc5SLei YU #include "activation_ubi.hpp"
29b21efc5SLei YU 
39b21efc5SLei YU #include "item_updater.hpp"
49b21efc5SLei YU #include "serialize.hpp"
59b21efc5SLei YU 
64d3d9126SJayashankar Padath #include <phosphor-logging/log.hpp>
79b21efc5SLei YU 
89f44c99aSBrad Bishop #include <filesystem>
98facccfaSBrad Bishop 
109b21efc5SLei YU namespace openpower
119b21efc5SLei YU {
129b21efc5SLei YU namespace software
139b21efc5SLei YU {
149b21efc5SLei YU namespace updater
159b21efc5SLei YU {
169b21efc5SLei YU namespace softwareServer = sdbusplus::xyz::openbmc_project::Software::server;
174d3d9126SJayashankar Padath using namespace phosphor::logging;
189b21efc5SLei YU 
priority(uint8_t value)199b21efc5SLei YU uint8_t RedundancyPriorityUbi::priority(uint8_t value)
209b21efc5SLei YU {
219b21efc5SLei YU     storeToFile(parent.versionId, value);
229b21efc5SLei YU     return RedundancyPriority::priority(value);
239b21efc5SLei YU }
249b21efc5SLei YU 
activation(Activations value)259b21efc5SLei YU auto ActivationUbi::activation(Activations value) -> Activations
269b21efc5SLei YU {
279b21efc5SLei YU     if (value != softwareServer::Activation::Activations::Active)
289b21efc5SLei YU     {
299b21efc5SLei YU         redundancyPriority.reset(nullptr);
309b21efc5SLei YU     }
319b21efc5SLei YU 
329b21efc5SLei YU     if (value == softwareServer::Activation::Activations::Activating)
339b21efc5SLei YU     {
349b21efc5SLei YU         parent.freeSpace();
359b21efc5SLei YU         softwareServer::Activation::activation(value);
369b21efc5SLei YU 
379b21efc5SLei YU         if (ubiVolumesCreated == false)
389b21efc5SLei YU         {
399b21efc5SLei YU             // Enable systemd signals
409b21efc5SLei YU             subscribeToSystemdSignals();
419b21efc5SLei YU 
429b21efc5SLei YU #ifdef WANT_SIGNATURE_VERIFY
439b21efc5SLei YU             // Validate the signed image.
442b2d2298SLei YU             if (!validateSignature(squashFSImage))
459b21efc5SLei YU             {
469b21efc5SLei YU                 // Cleanup
479b21efc5SLei YU                 activationBlocksTransition.reset(nullptr);
489b21efc5SLei YU                 activationProgress.reset(nullptr);
499b21efc5SLei YU 
509b21efc5SLei YU                 return softwareServer::Activation::activation(
519b21efc5SLei YU                     softwareServer::Activation::Activations::Failed);
529b21efc5SLei YU             }
539b21efc5SLei YU #endif
549b21efc5SLei YU             startActivation();
559b21efc5SLei YU             return softwareServer::Activation::activation(value);
569b21efc5SLei YU         }
579b21efc5SLei YU         else if (ubiVolumesCreated == true)
589b21efc5SLei YU         {
599b21efc5SLei YU             // Only when the squashfs image is finished loading AND the RW
609b21efc5SLei YU             // volumes have been created do we proceed with activation. To
619b21efc5SLei YU             // verify that this happened, we check for the mount dirs PNOR_PRSV
629b21efc5SLei YU             // and PNOR_RW_PREFIX_<versionid>, as well as the image dir R0.
639b21efc5SLei YU 
649f44c99aSBrad Bishop             if ((std::filesystem::is_directory(PNOR_PRSV)) &&
659f44c99aSBrad Bishop                 (std::filesystem::is_directory(PNOR_RW_PREFIX + versionId)) &&
669f44c99aSBrad Bishop                 (std::filesystem::is_directory(PNOR_RO_PREFIX + versionId)))
679b21efc5SLei YU             {
689b21efc5SLei YU                 finishActivation();
694d3d9126SJayashankar Padath                 if (Activation::checkApplyTimeImmediate())
704d3d9126SJayashankar Padath                 {
714d3d9126SJayashankar Padath                     log<level::INFO>("Image Active. ApplyTime is immediate, "
724d3d9126SJayashankar Padath                                      "rebooting Host.");
734d3d9126SJayashankar Padath                     Activation::rebootHost();
744d3d9126SJayashankar Padath                 }
759b21efc5SLei YU                 return softwareServer::Activation::activation(
769b21efc5SLei YU                     softwareServer::Activation::Activations::Active);
779b21efc5SLei YU             }
789b21efc5SLei YU             else
799b21efc5SLei YU             {
809b21efc5SLei YU                 activationBlocksTransition.reset(nullptr);
819b21efc5SLei YU                 activationProgress.reset(nullptr);
829b21efc5SLei YU                 return softwareServer::Activation::activation(
839b21efc5SLei YU                     softwareServer::Activation::Activations::Failed);
849b21efc5SLei YU             }
859b21efc5SLei YU         }
869b21efc5SLei YU     }
879b21efc5SLei YU     else
889b21efc5SLei YU     {
899b21efc5SLei YU         activationBlocksTransition.reset(nullptr);
909b21efc5SLei YU         activationProgress.reset(nullptr);
919b21efc5SLei YU     }
929b21efc5SLei YU 
939b21efc5SLei YU     return softwareServer::Activation::activation(value);
949b21efc5SLei YU }
959b21efc5SLei YU 
requestedActivation(RequestedActivations value)969b21efc5SLei YU auto ActivationUbi::requestedActivation(RequestedActivations value)
979b21efc5SLei YU     -> RequestedActivations
989b21efc5SLei YU {
999b21efc5SLei YU     ubiVolumesCreated = false;
1009b21efc5SLei YU     return Activation::requestedActivation(value);
1019b21efc5SLei YU }
1029b21efc5SLei YU 
startActivation()1039b21efc5SLei YU void ActivationUbi::startActivation()
1049b21efc5SLei YU {
1059b21efc5SLei YU     // Since the squashfs image has not yet been loaded to pnor and the
1069b21efc5SLei YU     // RW volumes have not yet been created, we need to start the
1079b21efc5SLei YU     // service files for each of those actions.
1089b21efc5SLei YU 
1099b21efc5SLei YU     if (!activationProgress)
1109b21efc5SLei YU     {
1119b21efc5SLei YU         activationProgress = std::make_unique<ActivationProgress>(bus, path);
1129b21efc5SLei YU     }
1139b21efc5SLei YU 
1149b21efc5SLei YU     if (!activationBlocksTransition)
1159b21efc5SLei YU     {
1169b21efc5SLei YU         activationBlocksTransition =
1179b21efc5SLei YU             std::make_unique<ActivationBlocksTransition>(bus, path);
1189b21efc5SLei YU     }
1199b21efc5SLei YU 
1209b21efc5SLei YU     constexpr auto ubimountService = "obmc-flash-bios-ubimount@";
121*f8e02429SPatrick Williams     auto ubimountServiceFile =
122*f8e02429SPatrick Williams         std::string(ubimountService) + versionId + ".service";
1239b21efc5SLei YU     auto method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH,
1249b21efc5SLei YU                                       SYSTEMD_INTERFACE, "StartUnit");
1259b21efc5SLei YU     method.append(ubimountServiceFile, "replace");
1269b21efc5SLei YU     bus.call_noreply(method);
1279b21efc5SLei YU 
1289b21efc5SLei YU     activationProgress->progress(10);
1299b21efc5SLei YU }
1309b21efc5SLei YU 
unitStateChange(sdbusplus::message_t & msg)1310dea1992SPatrick Williams void ActivationUbi::unitStateChange(sdbusplus::message_t& msg)
1329b21efc5SLei YU {
1339b21efc5SLei YU     uint32_t newStateID{};
1349b21efc5SLei YU     sdbusplus::message::object_path newStateObjPath;
1359b21efc5SLei YU     std::string newStateUnit{};
1369b21efc5SLei YU     std::string newStateResult{};
1379b21efc5SLei YU 
1389b21efc5SLei YU     // Read the msg and populate each variable
1399b21efc5SLei YU     msg.read(newStateID, newStateObjPath, newStateUnit, newStateResult);
1409b21efc5SLei YU 
141*f8e02429SPatrick Williams     auto ubimountServiceFile =
142*f8e02429SPatrick Williams         "obmc-flash-bios-ubimount@" + versionId + ".service";
1439b21efc5SLei YU 
1449b21efc5SLei YU     if (newStateUnit == ubimountServiceFile && newStateResult == "done")
1459b21efc5SLei YU     {
1469b21efc5SLei YU         ubiVolumesCreated = true;
1479b21efc5SLei YU         activationProgress->progress(activationProgress->progress() + 50);
1489b21efc5SLei YU     }
1499b21efc5SLei YU 
1509b21efc5SLei YU     if (ubiVolumesCreated)
1519b21efc5SLei YU     {
1529b21efc5SLei YU         activation(softwareServer::Activation::Activations::Activating);
1539b21efc5SLei YU     }
1549b21efc5SLei YU 
1559b21efc5SLei YU     if ((newStateUnit == ubimountServiceFile) &&
1569b21efc5SLei YU         (newStateResult == "failed" || newStateResult == "dependency"))
1579b21efc5SLei YU     {
1589b21efc5SLei YU         activation(softwareServer::Activation::Activations::Failed);
1599b21efc5SLei YU     }
1609b21efc5SLei YU 
1619b21efc5SLei YU     return;
1629b21efc5SLei YU }
1639b21efc5SLei YU 
finishActivation()1649b21efc5SLei YU void ActivationUbi::finishActivation()
1659b21efc5SLei YU {
1669b21efc5SLei YU     activationProgress->progress(90);
1679b21efc5SLei YU 
1689b21efc5SLei YU     // Set Redundancy Priority before setting to Active
1699b21efc5SLei YU     if (!redundancyPriority)
1709b21efc5SLei YU     {
171*f8e02429SPatrick Williams         redundancyPriority =
172*f8e02429SPatrick Williams             std::make_unique<RedundancyPriorityUbi>(bus, path, *this, 0);
1739b21efc5SLei YU     }
1749b21efc5SLei YU 
1759b21efc5SLei YU     activationProgress->progress(100);
1769b21efc5SLei YU 
1779b21efc5SLei YU     activationBlocksTransition.reset(nullptr);
1789b21efc5SLei YU     activationProgress.reset(nullptr);
1799b21efc5SLei YU 
1809b21efc5SLei YU     ubiVolumesCreated = false;
1819b21efc5SLei YU     unsubscribeFromSystemdSignals();
1829b21efc5SLei YU     // Remove version object from image manager
1839b21efc5SLei YU     deleteImageManagerObject();
1849b21efc5SLei YU     // Create active association
1859b21efc5SLei YU     parent.createActiveAssociation(path);
1863c81037eSAdriana Kobylak     // Create updateable association as this
1873c81037eSAdriana Kobylak     // can be re-programmed.
1883c81037eSAdriana Kobylak     parent.createUpdateableAssociation(path);
1899b21efc5SLei YU }
1909b21efc5SLei YU 
1919b21efc5SLei YU } // namespace updater
1929b21efc5SLei YU } // namespace software
1939b21efc5SLei YU } // namespace openpower
194