1 #pragma once
2 
3 #include "config.h"
4 
5 #include "activation_listener.hpp"
6 #include "association_interface.hpp"
7 #include "types.hpp"
8 #include "version.hpp"
9 
10 #include <sdbusplus/server.hpp>
11 #include <xyz/openbmc_project/Association/Definitions/server.hpp>
12 #include <xyz/openbmc_project/Common/FilePath/server.hpp>
13 #include <xyz/openbmc_project/Software/Activation/server.hpp>
14 #include <xyz/openbmc_project/Software/ActivationBlocksTransition/server.hpp>
15 #include <xyz/openbmc_project/Software/ActivationProgress/server.hpp>
16 #include <xyz/openbmc_project/Software/ExtendedVersion/server.hpp>
17 
18 #include <queue>
19 #include <string>
20 
21 class TestActivation;
22 
23 namespace phosphor
24 {
25 namespace software
26 {
27 namespace updater
28 {
29 
30 namespace sdbusRule = sdbusplus::bus::match::rules;
31 
32 using ActivationBlocksTransitionInherit =
33     sdbusplus::server::object_t<sdbusplus::xyz::openbmc_project::Software::
34                                     server::ActivationBlocksTransition>;
35 
36 /** @class ActivationBlocksTransition
37  *  @brief OpenBMC ActivationBlocksTransition implementation.
38  *  @details A concrete implementation for
39  *  xyz.openbmc_project.Software.ActivationBlocksTransition DBus API.
40  */
41 class ActivationBlocksTransition : public ActivationBlocksTransitionInherit
42 {
43   public:
44     ActivationBlocksTransition() = delete;
45     ActivationBlocksTransition(const ActivationBlocksTransition&) = delete;
46     ActivationBlocksTransition&
47         operator=(const ActivationBlocksTransition&) = delete;
48     ActivationBlocksTransition(ActivationBlocksTransition&&) = delete;
49     ActivationBlocksTransition&
50         operator=(ActivationBlocksTransition&&) = delete;
51 
52     /** @brief Constructs ActivationBlocksTransition.
53      *
54      * @param[in] bus    - The Dbus bus object
55      * @param[in] path   - The Dbus object path
56      */
57     ActivationBlocksTransition(sdbusplus::bus_t& bus, const std::string& path) :
58         ActivationBlocksTransitionInherit(bus, path.c_str(),
59                                           action::emit_interface_added),
60         bus(bus)
61     {
62         enableRebootGuard();
63     }
64 
65     ~ActivationBlocksTransition() override
66     {
67         disableRebootGuard();
68     }
69 
70   private:
71     sdbusplus::bus_t& bus;
72 
73     /** @brief Enables a Guard that blocks any BMC reboot commands */
74     void enableRebootGuard();
75 
76     /** @brief Disables any guard that was blocking the BMC reboot */
77     void disableRebootGuard();
78 };
79 
80 using ActivationProgressInherit = sdbusplus::server::object_t<
81     sdbusplus::xyz::openbmc_project::Software::server::ActivationProgress>;
82 
83 class ActivationProgress : public ActivationProgressInherit
84 {
85   public:
86     /** @brief Constructs ActivationProgress.
87      *
88      * @param[in] bus    - The Dbus bus object
89      * @param[in] path   - The Dbus object path
90      */
91     ActivationProgress(sdbusplus::bus_t& bus, const std::string& path) :
92         ActivationProgressInherit(bus, path.c_str(),
93                                   action::emit_interface_added)
94     {
95         progress(0);
96     }
97 };
98 
99 using ActivationInherit = sdbusplus::server::object_t<
100     sdbusplus::xyz::openbmc_project::Software::server::ExtendedVersion,
101     sdbusplus::xyz::openbmc_project::Software::server::Activation,
102     sdbusplus::xyz::openbmc_project::Association::server::Definitions,
103     sdbusplus::xyz::openbmc_project::Common::server::FilePath>;
104 
105 /** @class Activation
106  *  @brief OpenBMC activation software management implementation.
107  *  @details A concrete implementation for
108  *  xyz.openbmc_project.Software.Activation DBus API.
109  */
110 class Activation : public ActivationInherit
111 {
112   public:
113     friend class ::TestActivation;
114     using Status = Activations;
115 
116     /** @brief Constructs Activation Software Manager
117      *
118      * @param[in] bus    - The Dbus bus object
119      * @param[in] path   - The Dbus object path
120      * @param[in] versionId  - The software version id
121      * @param[in] extVersion - The extended version
122      * @param[in] activationStatus - The status of Activation
123      * @param[in] assocs - Association objects
124      * @param[in] filePath - The image filesystem path
125      */
126     Activation(sdbusplus::bus_t& bus, const std::string& objPath,
127                const std::string& versionId, const std::string& extVersion,
128                Status activationStatus, const AssociationList& assocs,
129                const std::string& filePath,
130                AssociationInterface* associationInterface,
131                ActivationListener* activationListener) :
132         ActivationInherit(bus, objPath.c_str(),
133                           ActivationInherit::action::defer_emit),
134         bus(bus), objPath(objPath), versionId(versionId),
135         systemdSignals(
136             bus,
137             sdbusRule::type::signal() + sdbusRule::member("JobRemoved") +
138                 sdbusRule::path("/org/freedesktop/systemd1") +
139                 sdbusRule::interface("org.freedesktop.systemd1.Manager"),
140             std::bind(&Activation::unitStateChange, this,
141                       std::placeholders::_1)),
142         associationInterface(associationInterface),
143         activationListener(activationListener)
144     {
145         // Set Properties.
146         extendedVersion(extVersion);
147         activation(activationStatus);
148         associations(assocs);
149         path(filePath);
150 
151         auto info = Version::getExtVersionInfo(extVersion);
152         manufacturer = info["manufacturer"];
153         model = info["model"];
154 
155         // Emit deferred signal.
156         emit_object_added();
157     }
158 
159     /** @brief Overloaded Activation property setter function
160      *
161      * @param[in] value - One of Activation::Activations
162      *
163      * @return Success or exception thrown
164      */
165     Status activation(Status value) override;
166 
167     /** @brief Activation */
168     using ActivationInherit::activation;
169 
170     /** @brief Overloaded requestedActivation property setter function
171      *
172      * @param[in] value - One of Activation::RequestedActivations
173      *
174      * @return Success or exception thrown
175      */
176     RequestedActivations
177         requestedActivation(RequestedActivations value) override;
178 
179     /** @brief Get the object path */
180     const std::string& getObjectPath() const
181     {
182         return objPath;
183     }
184 
185     /** @brief Get the version ID */
186     const std::string& getVersionId() const
187     {
188         return versionId;
189     }
190 
191   private:
192     /** @brief Check if systemd state change is relevant to this object
193      *
194      * Instance specific interface to handle the detected systemd state
195      * change
196      *
197      * @param[in]  msg       - Data associated with subscribed signal
198      *
199      */
200     void unitStateChange(sdbusplus::message_t& msg);
201 
202     /**
203      * @brief Delete the version from Image Manager and the
204      *        untar image from image upload dir.
205      */
206     void deleteImageManagerObject();
207 
208     /** @brief Invoke the update service for the PSU
209      *
210      * @param[in] psuInventoryPath - The PSU inventory to be updated.
211      *
212      * @return true if the update starts, and false if it fails.
213      */
214     bool doUpdate(const std::string& psuInventoryPath);
215 
216     /** @brief Do PSU update one-by-one
217      *
218      * @return true if the update starts, and false if it fails.
219      */
220     bool doUpdate();
221 
222     /** @brief Handle an update done event */
223     void onUpdateDone();
224 
225     /** @brief Handle an update failure event */
226     void onUpdateFailed();
227 
228     /** @brief Start PSU update */
229     Status startActivation();
230 
231     /** @brief Finish PSU update */
232     void finishActivation();
233 
234     /** @brief Check if the PSU is present */
235     bool isPresent(const std::string& psuInventoryPath);
236 
237     /** @brief Check if the PSU is compatible with this software*/
238     bool isCompatible(const std::string& psuInventoryPath);
239 
240     /** @brief Store the updated PSU image to persistent dir */
241     void storeImage();
242 
243     /** @brief Construct the systemd service name
244      *
245      *  @details Throws an exception if an error occurs
246      *
247      * @param[in] psuInventoryPath - The PSU inventory to be updated.
248      *
249      * @return The escaped string of systemd unit to do the PSU update.
250      */
251     std::string getUpdateService(const std::string& psuInventoryPath);
252 
253     /** @brief Persistent sdbusplus DBus bus connection */
254     sdbusplus::bus_t& bus;
255 
256     /** @brief Persistent DBus object path */
257     std::string objPath;
258 
259     /** @brief Version id */
260     std::string versionId;
261 
262     /** @brief Used to subscribe to dbus systemd signals */
263     sdbusplus::bus::match_t systemdSignals;
264 
265     /** @brief The queue of psu objects to be updated */
266     std::queue<std::string> psuQueue;
267 
268     /** @brief The progress step for each PSU update is done */
269     uint32_t progressStep;
270 
271     /** @brief The PSU update systemd unit */
272     std::string psuUpdateUnit;
273 
274     /** @brief The PSU Inventory path of the current updating PSU */
275     std::string currentUpdatingPsu;
276 
277     /** @brief Persistent ActivationBlocksTransition dbus object */
278     std::unique_ptr<ActivationBlocksTransition> activationBlocksTransition;
279 
280     /** @brief Persistent ActivationProgress dbus object */
281     std::unique_ptr<ActivationProgress> activationProgress;
282 
283     /** @brief The AssociationInterface pointer */
284     AssociationInterface* associationInterface;
285 
286     /** @brief The activationListener pointer */
287     ActivationListener* activationListener;
288 
289     /** @brief The PSU manufacturer of the software */
290     std::string manufacturer;
291 
292     /** @brief The PSU model of the software */
293     std::string model;
294 };
295 
296 } // namespace updater
297 } // namespace software
298 } // namespace phosphor
299