xref: /openbmc/phosphor-power/phosphor-power-supply/util.hpp (revision 92261f88729b618b1c31d20f28328a8aff73b83b)
13f1242f3SBrandon Wyman #pragma once
23f1242f3SBrandon Wyman #include "util_base.hpp"
33f1242f3SBrandon Wyman #include "utility.hpp"
4681b2a36SB. J. Wyman #include "xyz/openbmc_project/Common/error.hpp"
5681b2a36SB. J. Wyman 
6681b2a36SB. J. Wyman #include <gpiod.hpp>
7681b2a36SB. J. Wyman #include <phosphor-logging/elog-errors.hpp>
8681b2a36SB. J. Wyman #include <phosphor-logging/elog.hpp>
9681b2a36SB. J. Wyman #include <phosphor-logging/log.hpp>
103f1242f3SBrandon Wyman 
1152245b69SAdriana Kobylak #include <bitset>
1218a24d92SBrandon Wyman #include <chrono>
13768d2269SShawn McCarney #include <format>
1452245b69SAdriana Kobylak 
153f1242f3SBrandon Wyman namespace phosphor::power::psu
163f1242f3SBrandon Wyman {
173f1242f3SBrandon Wyman 
180975eaf4SMatt Spinler using Property = std::string;
190975eaf4SMatt Spinler using Value = std::variant<bool, std::string>;
200975eaf4SMatt Spinler using PropertyMap = std::map<Property, Value>;
210975eaf4SMatt Spinler using Interface = std::string;
220975eaf4SMatt Spinler using InterfaceMap = std::map<Interface, PropertyMap>;
230975eaf4SMatt Spinler using Object = sdbusplus::message::object_path;
240975eaf4SMatt Spinler using ObjectMap = std::map<Object, InterfaceMap>;
250975eaf4SMatt Spinler 
263f1242f3SBrandon Wyman class Util : public UtilBase
273f1242f3SBrandon Wyman {
283f1242f3SBrandon Wyman   public:
getPresence(sdbusplus::bus_t & bus,const std::string & invpath) const297354ce62SPatrick Williams     bool getPresence(sdbusplus::bus_t& bus,
303f1242f3SBrandon Wyman                      const std::string& invpath) const override
313f1242f3SBrandon Wyman     {
323f1242f3SBrandon Wyman         bool present = false;
333f1242f3SBrandon Wyman 
343f1242f3SBrandon Wyman         // Use getProperty utility function to get presence status.
353f1242f3SBrandon Wyman         util::getProperty(INVENTORY_IFACE, PRESENT_PROP, invpath,
363f1242f3SBrandon Wyman                           INVENTORY_MGR_IFACE, bus, present);
373f1242f3SBrandon Wyman 
383f1242f3SBrandon Wyman         return present;
393f1242f3SBrandon Wyman     }
40681b2a36SB. J. Wyman 
setPresence(sdbusplus::bus_t & bus,const std::string & invpath,bool present,const std::string & name) const417354ce62SPatrick Williams     void setPresence(sdbusplus::bus_t& bus, const std::string& invpath,
42681b2a36SB. J. Wyman                      bool present, const std::string& name) const override
43681b2a36SB. J. Wyman     {
44e4fa48c2SGeorge Liu         using namespace phosphor::logging;
45768d2269SShawn McCarney         log<level::INFO>(std::format("Updating inventory present property. "
46e4fa48c2SGeorge Liu                                      "present:{} invpath:{} name:{}",
47e4fa48c2SGeorge Liu                                      present, invpath, name)
48e4fa48c2SGeorge Liu                              .c_str());
49e4fa48c2SGeorge Liu 
50681b2a36SB. J. Wyman         using InternalFailure =
51681b2a36SB. J. Wyman             sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure;
52681b2a36SB. J. Wyman         PropertyMap invProp;
53681b2a36SB. J. Wyman 
54681b2a36SB. J. Wyman         invProp.emplace("Present", present);
55681b2a36SB. J. Wyman         invProp.emplace("PrettyName", name);
56681b2a36SB. J. Wyman 
57681b2a36SB. J. Wyman         InterfaceMap invIntf;
58681b2a36SB. J. Wyman         invIntf.emplace("xyz.openbmc_project.Inventory.Item",
59681b2a36SB. J. Wyman                         std::move(invProp));
60681b2a36SB. J. Wyman 
61681b2a36SB. J. Wyman         Interface extraIface = "xyz.openbmc_project.Inventory.Item.PowerSupply";
62681b2a36SB. J. Wyman 
63681b2a36SB. J. Wyman         invIntf.emplace(extraIface, PropertyMap());
64681b2a36SB. J. Wyman 
65681b2a36SB. J. Wyman         ObjectMap invObj;
66681b2a36SB. J. Wyman         invObj.emplace(std::move(invpath), std::move(invIntf));
67681b2a36SB. J. Wyman 
68681b2a36SB. J. Wyman         try
69681b2a36SB. J. Wyman         {
70681b2a36SB. J. Wyman             auto invService = phosphor::power::util::getService(
71681b2a36SB. J. Wyman                 INVENTORY_OBJ_PATH, INVENTORY_MGR_IFACE, bus);
72681b2a36SB. J. Wyman 
73681b2a36SB. J. Wyman             // Update inventory
74f5402197SPatrick Williams             auto invMsg =
75f5402197SPatrick Williams                 bus.new_method_call(invService.c_str(), INVENTORY_OBJ_PATH,
76681b2a36SB. J. Wyman                                     INVENTORY_MGR_IFACE, "Notify");
77681b2a36SB. J. Wyman             invMsg.append(std::move(invObj));
78681b2a36SB. J. Wyman             auto invMgrResponseMsg = bus.call(invMsg);
79681b2a36SB. J. Wyman         }
80681b2a36SB. J. Wyman         catch (const std::exception& e)
81681b2a36SB. J. Wyman         {
82681b2a36SB. J. Wyman             log<level::ERR>(
83768d2269SShawn McCarney                 std::format(
84681b2a36SB. J. Wyman                     "Error in inventory manager call to update inventory: {}",
85681b2a36SB. J. Wyman                     e.what())
86681b2a36SB. J. Wyman                     .c_str());
87681b2a36SB. J. Wyman             elog<InternalFailure>();
88681b2a36SB. J. Wyman         }
89681b2a36SB. J. Wyman     }
900975eaf4SMatt Spinler 
setAvailable(sdbusplus::bus_t & bus,const std::string & invpath,bool available) const917354ce62SPatrick Williams     void setAvailable(sdbusplus::bus_t& bus, const std::string& invpath,
920975eaf4SMatt Spinler                       bool available) const override
930975eaf4SMatt Spinler     {
940975eaf4SMatt Spinler         PropertyMap invProp;
950975eaf4SMatt Spinler         InterfaceMap invIntf;
960975eaf4SMatt Spinler         ObjectMap invObj;
970975eaf4SMatt Spinler 
980975eaf4SMatt Spinler         invProp.emplace(AVAILABLE_PROP, available);
990975eaf4SMatt Spinler         invIntf.emplace(AVAILABILITY_IFACE, std::move(invProp));
1000975eaf4SMatt Spinler 
1010975eaf4SMatt Spinler         invObj.emplace(std::move(invpath), std::move(invIntf));
1020975eaf4SMatt Spinler 
1030975eaf4SMatt Spinler         try
1040975eaf4SMatt Spinler         {
1050975eaf4SMatt Spinler             auto invService = phosphor::power::util::getService(
1060975eaf4SMatt Spinler                 INVENTORY_OBJ_PATH, INVENTORY_MGR_IFACE, bus);
1070975eaf4SMatt Spinler 
108f5402197SPatrick Williams             auto invMsg =
109f5402197SPatrick Williams                 bus.new_method_call(invService.c_str(), INVENTORY_OBJ_PATH,
1100975eaf4SMatt Spinler                                     INVENTORY_MGR_IFACE, "Notify");
1110975eaf4SMatt Spinler             invMsg.append(std::move(invObj));
1120975eaf4SMatt Spinler             auto invMgrResponseMsg = bus.call(invMsg);
1130975eaf4SMatt Spinler         }
1147354ce62SPatrick Williams         catch (const sdbusplus::exception_t& e)
1150975eaf4SMatt Spinler         {
1160975eaf4SMatt Spinler             using namespace phosphor::logging;
1170975eaf4SMatt Spinler             log<level::ERR>(
118768d2269SShawn McCarney                 std::format("Error in inventory manager call to update "
1190975eaf4SMatt Spinler                             "availability interface: {}",
1200975eaf4SMatt Spinler                             e.what())
1210975eaf4SMatt Spinler                     .c_str());
1220975eaf4SMatt Spinler             throw;
1230975eaf4SMatt Spinler         }
1240975eaf4SMatt Spinler     }
125ca1e9ea1SMatt Spinler 
handleChassisHealthRollup(sdbusplus::bus_t & bus,const std::string & invpath,bool addRollup) const1267354ce62SPatrick Williams     void handleChassisHealthRollup(sdbusplus::bus_t& bus,
127ca1e9ea1SMatt Spinler                                    const std::string& invpath,
128ca1e9ea1SMatt Spinler                                    bool addRollup) const override
129ca1e9ea1SMatt Spinler     {
130ca1e9ea1SMatt Spinler         using AssociationTuple =
131ca1e9ea1SMatt Spinler             std::tuple<std::string, std::string, std::string>;
132ca1e9ea1SMatt Spinler         using AssociationsProperty = std::vector<AssociationTuple>;
133ca1e9ea1SMatt Spinler         try
134ca1e9ea1SMatt Spinler         {
135ca1e9ea1SMatt Spinler             auto chassisPath = getChassis(bus, invpath);
136ca1e9ea1SMatt Spinler 
137ca1e9ea1SMatt Spinler             auto service = phosphor::power::util::getService(
138ca1e9ea1SMatt Spinler                 invpath, ASSOC_DEF_IFACE, bus);
139ca1e9ea1SMatt Spinler 
140ca1e9ea1SMatt Spinler             AssociationsProperty associations;
141ca1e9ea1SMatt Spinler             phosphor::power::util::getProperty<AssociationsProperty>(
142ca1e9ea1SMatt Spinler                 ASSOC_DEF_IFACE, ASSOC_PROP, invpath, service, bus,
143ca1e9ea1SMatt Spinler                 associations);
144ca1e9ea1SMatt Spinler 
145ca1e9ea1SMatt Spinler             AssociationTuple critAssociation{"health_rollup", "critical",
146ca1e9ea1SMatt Spinler                                              chassisPath};
147ca1e9ea1SMatt Spinler 
148ca1e9ea1SMatt Spinler             auto assocIt = std::find(associations.begin(), associations.end(),
149ca1e9ea1SMatt Spinler                                      critAssociation);
150ca1e9ea1SMatt Spinler             if (addRollup)
151ca1e9ea1SMatt Spinler             {
152ca1e9ea1SMatt Spinler                 if (assocIt != associations.end())
153ca1e9ea1SMatt Spinler                 {
154ca1e9ea1SMatt Spinler                     // It's already there
155ca1e9ea1SMatt Spinler                     return;
156ca1e9ea1SMatt Spinler                 }
157ca1e9ea1SMatt Spinler 
158ca1e9ea1SMatt Spinler                 associations.push_back(critAssociation);
159ca1e9ea1SMatt Spinler             }
160ca1e9ea1SMatt Spinler             else
161ca1e9ea1SMatt Spinler             {
162ca1e9ea1SMatt Spinler                 if (assocIt == associations.end())
163ca1e9ea1SMatt Spinler                 {
164ca1e9ea1SMatt Spinler                     // It's already been removed.
165ca1e9ea1SMatt Spinler                     return;
166ca1e9ea1SMatt Spinler                 }
167ca1e9ea1SMatt Spinler 
168ca1e9ea1SMatt Spinler                 // If the object still isn't functional, then don't clear
169ca1e9ea1SMatt Spinler                 // the association.
170ca1e9ea1SMatt Spinler                 bool functional = false;
171ca1e9ea1SMatt Spinler                 phosphor::power::util::getProperty<bool>(
172ca1e9ea1SMatt Spinler                     OPERATIONAL_STATE_IFACE, FUNCTIONAL_PROP, invpath, service,
173ca1e9ea1SMatt Spinler                     bus, functional);
174ca1e9ea1SMatt Spinler 
175ca1e9ea1SMatt Spinler                 if (!functional)
176ca1e9ea1SMatt Spinler                 {
177ca1e9ea1SMatt Spinler                     return;
178ca1e9ea1SMatt Spinler                 }
179ca1e9ea1SMatt Spinler 
180ca1e9ea1SMatt Spinler                 associations.erase(assocIt);
181ca1e9ea1SMatt Spinler             }
182ca1e9ea1SMatt Spinler 
183ca1e9ea1SMatt Spinler             phosphor::power::util::setProperty(ASSOC_DEF_IFACE, ASSOC_PROP,
184ca1e9ea1SMatt Spinler                                                invpath, service, bus,
185ca1e9ea1SMatt Spinler                                                associations);
186ca1e9ea1SMatt Spinler         }
1877354ce62SPatrick Williams         catch (const sdbusplus::exception_t& e)
188ca1e9ea1SMatt Spinler         {
189ca1e9ea1SMatt Spinler             using namespace phosphor::logging;
190768d2269SShawn McCarney             log<level::INFO>(std::format("Error trying to handle health rollup "
191ca1e9ea1SMatt Spinler                                          "associations for {}: {}",
192ca1e9ea1SMatt Spinler                                          invpath, e.what())
193ca1e9ea1SMatt Spinler                                  .c_str());
194ca1e9ea1SMatt Spinler         }
195ca1e9ea1SMatt Spinler     }
196ca1e9ea1SMatt Spinler 
getChassis(sdbusplus::bus_t & bus,const std::string & invpath) const1977354ce62SPatrick Williams     std::string getChassis(sdbusplus::bus_t& bus,
198592bd27cSMatt Spinler                            const std::string& invpath) const override
199ca1e9ea1SMatt Spinler     {
20006594229SMatt Spinler         sdbusplus::message::object_path assocPath = invpath + "/powering";
20106594229SMatt Spinler         sdbusplus::message::object_path basePath{"/"};
20206594229SMatt Spinler         std::vector<std::string> interfaces{CHASSIS_IFACE};
203ca1e9ea1SMatt Spinler 
20406594229SMatt Spinler         // Find the object path that implements the chassis interface
20506594229SMatt Spinler         // and also shows up in the endpoints list of the powering assoc.
20606594229SMatt Spinler         auto chassisPaths = phosphor::power::util::getAssociatedSubTreePaths(
20706594229SMatt Spinler             bus, assocPath, basePath, interfaces, 0);
208ca1e9ea1SMatt Spinler 
20906594229SMatt Spinler         if (chassisPaths.empty())
210ca1e9ea1SMatt Spinler         {
211768d2269SShawn McCarney             throw std::runtime_error(std::format(
21206594229SMatt Spinler                 "No association to a chassis found for {}", invpath));
213ca1e9ea1SMatt Spinler         }
214ca1e9ea1SMatt Spinler 
21506594229SMatt Spinler         return chassisPaths[0];
216ca1e9ea1SMatt Spinler     }
217681b2a36SB. J. Wyman };
218681b2a36SB. J. Wyman 
2193ca062aeSAdriana Kobylak std::unique_ptr<GPIOInterfaceBase> createGPIO(const std::string& namedGpio);
220681b2a36SB. J. Wyman 
2213ca062aeSAdriana Kobylak class GPIOInterface : public GPIOInterfaceBase
222681b2a36SB. J. Wyman {
223681b2a36SB. J. Wyman   public:
2243ca062aeSAdriana Kobylak     GPIOInterface() = delete;
2253ca062aeSAdriana Kobylak     virtual ~GPIOInterface() = default;
2263ca062aeSAdriana Kobylak     GPIOInterface(const GPIOInterface&) = default;
2273ca062aeSAdriana Kobylak     GPIOInterface& operator=(const GPIOInterface&) = default;
2283ca062aeSAdriana Kobylak     GPIOInterface(GPIOInterface&&) = default;
2293ca062aeSAdriana Kobylak     GPIOInterface& operator=(GPIOInterface&&) = default;
230681b2a36SB. J. Wyman 
231681b2a36SB. J. Wyman     /**
232681b2a36SB. J. Wyman      * Constructor
233681b2a36SB. J. Wyman      *
234681b2a36SB. J. Wyman      * @param[in] namedGpio - The string for the gpio-line-name
235681b2a36SB. J. Wyman      */
2363ca062aeSAdriana Kobylak     GPIOInterface(const std::string& namedGpio);
237681b2a36SB. J. Wyman 
238*92261f88SPatrick Williams     static std::unique_ptr<GPIOInterfaceBase> createGPIO(
239*92261f88SPatrick Williams         const std::string& namedGpio);
240681b2a36SB. J. Wyman 
241681b2a36SB. J. Wyman     /**
242681b2a36SB. J. Wyman      * @brief Attempts to read the state of the GPIO line.
243681b2a36SB. J. Wyman      *
244681b2a36SB. J. Wyman      * Throws an exception if line not found, request line fails, or get_value
245681b2a36SB. J. Wyman      * from line fails.
246681b2a36SB. J. Wyman      *
247681b2a36SB. J. Wyman      * @return 1 for active (low/present), 0 for not active (high/not present).
248681b2a36SB. J. Wyman      */
249681b2a36SB. J. Wyman     int read() override;
250681b2a36SB. J. Wyman 
251d8b8cb15SB. J. Wyman     /**
25252245b69SAdriana Kobylak      * @brief Attempts to set the state of the GPIO line to the specified value.
25352245b69SAdriana Kobylak      *
25452245b69SAdriana Kobylak      * Throws an exception if line not found, request line fails, or set_value
25552245b69SAdriana Kobylak      * to line fails.
25652245b69SAdriana Kobylak      *
25752245b69SAdriana Kobylak      * @param[in] value - The value to set the state of the GPIO line, 1 or 0.
25852245b69SAdriana Kobylak      * @param[in] flags - Additional line request flags as defined in gpiod.hpp.
25952245b69SAdriana Kobylak      */
26052245b69SAdriana Kobylak     void write(int value, std::bitset<32> flags) override;
26152245b69SAdriana Kobylak 
26252245b69SAdriana Kobylak     /**
26318a24d92SBrandon Wyman      * @brief Attempts to toggle (write) a GPIO low then high.
26418a24d92SBrandon Wyman      *
26518a24d92SBrandon Wyman      * Relies on write, so throws exception if line not found, etc.
26618a24d92SBrandon Wyman      *
26718a24d92SBrandon Wyman      * @param[in] delay - Milliseconds to delay betwen low/high toggle.
26818a24d92SBrandon Wyman      */
26918a24d92SBrandon Wyman     void toggleLowHigh(const std::chrono::milliseconds& delay) override;
27018a24d92SBrandon Wyman 
27118a24d92SBrandon Wyman     /**
272d8b8cb15SB. J. Wyman      * @brief Returns the name of the GPIO, if not empty.
273d8b8cb15SB. J. Wyman      */
274d8b8cb15SB. J. Wyman     std::string getName() const override;
275d8b8cb15SB. J. Wyman 
276681b2a36SB. J. Wyman   private:
277681b2a36SB. J. Wyman     gpiod::line line;
2783f1242f3SBrandon Wyman };
2793f1242f3SBrandon Wyman 
2803f1242f3SBrandon Wyman } // namespace phosphor::power::psu
281