1 #pragma once
2 #include <sdbusplus/bus/match.hpp>
3 #include <sdeventplus/clock.hpp>
4 #include <sdeventplus/event.hpp>
5 #include <sdeventplus/utility/timer.hpp>
6 #include "average.hpp"
7 #include "device.hpp"
8 #include "maximum.hpp"
9 #include "names_values.hpp"
10 #include "pmbus.hpp"
11 #include "record_manager.hpp"
12 
13 namespace witherspoon
14 {
15 namespace power
16 {
17 namespace psu
18 {
19 
20 namespace sdbusRule = sdbusplus::bus::match::rules;
21 
22 constexpr auto FAULT_COUNT = 3;
23 
24 /**
25  * @class PowerSupply
26  * Represents a PMBus power supply device.
27  */
28 class PowerSupply : public Device
29 {
30     public:
31         PowerSupply() = delete;
32         PowerSupply(const PowerSupply&) = delete;
33         PowerSupply(PowerSupply&&) = default;
34         PowerSupply& operator=(const PowerSupply&) = default;
35         PowerSupply& operator=(PowerSupply&&) = default;
36         ~PowerSupply() = default;
37 
38         /**
39          * Constructor
40          *
41          * @param[in] name - the device name
42          * @param[in] inst - the device instance
43          * @param[in] objpath - the path to monitor
44          * @param[in] invpath - the inventory path to use
45          * @param[in] bus - D-Bus bus object
46          * @param[in] e - event object
47          * @param[in] t - time to allow power supply to assert PG#
48          * @param[in] p - time to allow power supply presence state to
49          *                settle/deglitch and allow for application of power
50          *                prior to fault checking
51          */
52         PowerSupply(const std::string& name, size_t inst,
53                     const std::string& objpath,
54                     const std::string& invpath,
55                     sdbusplus::bus::bus& bus,
56                     const sdeventplus::Event& e,
57                     std::chrono::seconds& t,
58                     std::chrono::seconds& p);
59 
60         /**
61          * Power supply specific function to analyze for faults/errors.
62          *
63          * Various PMBus status bits will be checked for fault conditions.
64          * If a certain fault bits are on, the appropriate error will be
65          * committed.
66          */
67         void analyze() override;
68 
69         /**
70          * Write PMBus CLEAR_FAULTS
71          *
72          * This function will be called in various situations in order to clear
73          * any fault status bits that may have been set, in order to start over
74          * with a clean state. Presence changes and power state changes will
75          * want to clear any faults logged.
76          */
77         void clearFaults() override;
78 
79         /**
80          * Mark error for specified callout and message as resolved.
81          *
82          * @param[in] callout - The callout to be resolved (inventory path)
83          * @parma[in] message - The message for the fault to be resolved
84          */
85         void resolveError(const std::string& callout,
86                           const std::string& message);
87 
88         /**
89          * Enables making the input power history available on D-Bus
90          *
91          * @param[in] objectPath - the D-Bus object path to use
92          * @param[in] maxRecords - the number of history records to keep
93          * @param[in] syncGPIOPath - The gpiochip device path to use for
94          *                           sending the sync command
95          * @paramp[in] syncGPIONum - the GPIO number for the sync command
96          */
97         void enableHistory(const std::string& objectPath,
98                            size_t numRecords,
99                            const std::string& syncGPIOPath,
100                            size_t syncGPIONum);
101 
102     private:
103         /**
104          * The path to use for reading various PMBus bits/words.
105          */
106         std::string monitorPath;
107 
108         /**
109          * @brief Pointer to the PMBus interface
110          *
111          * Used to read out of or write to the /sysfs tree(s) containing files
112          * that a device driver monitors the PMBus interface to the power
113          * supplies.
114          */
115         witherspoon::pmbus::PMBus pmbusIntf;
116 
117         /**
118          * @brief D-Bus path to use for this power supply's inventory status.
119          */
120         std::string inventoryPath;
121 
122         /** @brief Connection for sdbusplus bus */
123         sdbusplus::bus::bus& bus;
124 
125         /** @brief True if the power supply is present. */
126         bool present = false;
127 
128         /** @brief Used to subscribe to D-Bus property changes for Present */
129         std::unique_ptr<sdbusplus::bus::match_t> presentMatch;
130 
131         /**
132          * @brief Interval for setting present to true.
133          *
134          * The amount of time to wait from not present to present change before
135          * updating the internal present indicator. Allows person servicing
136          * the power supply some time to plug in the cable.
137          */
138         std::chrono::seconds presentInterval;
139 
140         /**
141          * @brief Timer used to delay setting the internal present state.
142          *
143          * The timer used to do the callback after the present property has
144          * changed.
145          */
146         sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic> presentTimer;
147 
148         /** @brief True if a fault has already been found and not cleared */
149         bool faultFound = false;
150 
151         /** @brief True if the power is on. */
152         bool powerOn = false;
153 
154         /**
155          * @brief Equal to FAULT_COUNT if power on fault has been
156          * detected.
157          */
158         size_t powerOnFault = 0;
159 
160         /**
161          * @brief Interval to setting powerOn to true.
162          *
163          * The amount of time to wait from power state on to setting the
164          * internal powerOn state to true. The amount of time the power supply
165          * is allowed to delay setting DGood/PG#.
166          */
167         std::chrono::seconds powerOnInterval;
168 
169         /**
170          * @brief Timer used to delay setting the internal powerOn state.
171          *
172          * The timer used to do the callback after the power state has been on
173          * long enough.
174          */
175         sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic> powerOnTimer;
176 
177         /** @brief Used to subscribe to D-Bus power on state changes */
178         std::unique_ptr<sdbusplus::bus::match_t> powerOnMatch;
179 
180         /** @brief Indicates that a read failure has occurred.
181          *
182          * @details This will be incremented each time a read failure is
183          *          encountered. If it is incremented to FAULT_COUNT, an error
184          *          will be logged.
185          */
186         size_t readFail = 0;
187 
188         /** @brief Has a PMBus read failure already been logged? */
189         bool readFailLogged = false;
190 
191         /**
192          * @brief Indicates an input fault or warning if equal to FAULT_COUNT.
193          *
194          * @details This is the "INPUT FAULT OR WARNING" bit in the high byte,
195          *          or the VIN_UV_FAULT bit in the low byte in the STATUS_WORD
196          *          command response. If either of those bits are on, this will
197          *          be incremented.
198          */
199         size_t inputFault = 0;
200 
201         /**
202          * @brief Indicates output over current fault if equal to FAULT_COUNT
203          *
204          * @details This is incremented when the "IOUT_OC_FAULT" bit in the low
205          *          byte from the STATUS_WORD command response is on.
206          */
207         size_t outputOCFault = 0;
208 
209         /**
210          * @brief Indicates output overvoltage fault if equal to FAULT_COUNT.
211          *
212          * @details This is incremented when the "VOUT_OV_FAULT" bit in the
213          *          STATUS_WORD command response is on.
214          */
215         size_t outputOVFault = 0;
216 
217         /**
218          * @brief Indicates a fan fault or warning condition was detected if
219          *        equal to FAULT_COUNT.
220          *
221          * @details This is incremented when the 'FAN_FAULT' bit in the
222          *          STATUS_WORD command response is on.
223          */
224         size_t fanFault = 0;
225 
226         /**
227          * @brief Indicates a temperature fault or warn condition was detected
228          *        if equal to FAULT_COUNT.
229          *
230          * @details This is incremented when the 'TEMPERATURE_FAULT_WARN' bit
231          *          in the STATUS_WORD command response is on, or if the
232          *          'OT_FAULT' bit in the STATUS_TEMPERATURE command response
233          *          is on.
234          */
235         size_t temperatureFault = 0;
236 
237         /**
238          * @brief Class that manages the input power history records.
239          */
240         std::unique_ptr<history::RecordManager> recordManager;
241 
242         /**
243          * @brief The D-Bus object for the average input power history
244          */
245         std::unique_ptr<history::Average> average;
246 
247         /**
248          * @brief The D-Bus object for the maximum input power history
249          */
250         std::unique_ptr<history::Maximum> maximum;
251 
252         /**
253          * @brief The base D-Bus object path to use for the average
254          *        and maximum objects.
255          */
256         std::string historyObjectPath;
257 
258         /**
259          * @brief The GPIO device path to use for sending the 'sync'
260          *        command to the PS.
261          */
262         std::string syncGPIODevPath;
263 
264         /**
265          * @brief The GPIO number to use for sending the 'sync'
266          *        command to the PS.
267          */
268         size_t syncGPIONumber = 0;
269 
270         /**
271          * @brief Callback for inventory property changes
272          *
273          * Process change of Present property for power supply.
274          *
275          * @param[in]  msg - Data associated with Present change signal
276          *
277          */
278         void inventoryChanged(sdbusplus::message::message& msg);
279 
280         /**
281          * Updates the presence status by querying D-Bus
282          *
283          * The D-Bus inventory properties for this power supply will be read to
284          * determine if the power supply is present or not and update this
285          * objects present member variable to reflect current status.
286          */
287         void updatePresence();
288 
289         /**
290          * @brief Updates the poweredOn status by querying D-Bus
291          *
292          * The D-Bus property for the system power state will be read to
293          * determine if the system is powered on or not.
294          */
295         void updatePowerState();
296 
297         /**
298          * @brief Callback for power state property changes
299          *
300          * Process changes to the powered on stat property for the system.
301          *
302          * @param[in] msg - Data associated with the power state signal
303          */
304         void powerStateChanged(sdbusplus::message::message& msg);
305 
306         /**
307          * @brief Wrapper for PMBus::read() and adding metadata
308          *
309          * @param[out] nv - NamesValues instance to store cmd string and value
310          * @param[in] cmd - String for the command to read data from.
311          * @param[in] type - The type of file to read the command from.
312          */
313         void captureCmd(util::NamesValues& nv, const std::string& cmd,
314                         witherspoon::pmbus::Type type);
315 
316         /**
317          * @brief Checks for input voltage faults and logs error if needed.
318          *
319          * Check for voltage input under voltage fault (VIN_UV_FAULT) and/or
320          * input fault or warning (INPUT_FAULT), and logs appropriate error(s).
321          *
322          * @param[in] statusWord  - 2 byte STATUS_WORD value read from sysfs
323          */
324         void checkInputFault(const uint16_t statusWord);
325 
326         /**
327          * @brief Checks for power good negated or unit is off in wrong state
328          *
329          * @param[in] statusWord  - 2 byte STATUS_WORD value read from sysfs
330          */
331         void checkPGOrUnitOffFault(const uint16_t statusWord);
332 
333         /**
334          * @brief Checks for output current over current fault.
335          *
336          * IOUT_OC_FAULT is checked, if on, appropriate error is logged.
337          *
338          * @param[in] statusWord  - 2 byte STATUS_WORD value read from sysfs
339          */
340         void checkCurrentOutOverCurrentFault(const uint16_t statusWord);
341 
342         /**
343          * @brief Checks for output overvoltage fault.
344          *
345          * VOUT_OV_FAULT is checked, if on, appropriate error is logged.
346          *
347          * @param[in] statusWord  - 2 byte STATUS_WORD value read from sysfs
348          */
349         void checkOutputOvervoltageFault(const uint16_t statusWord);
350 
351         /**
352          * @brief Checks for a fan fault or warning condition.
353          *
354          * The high byte of STATUS_WORD is checked to see if the "FAN FAULT OR
355          * WARNING" bit is turned on. If it is on, log an error.
356          *
357          * @param[in] statusWord - 2 byte STATUS_WORD value read from sysfs
358          */
359         void checkFanFault(const uint16_t statusWord);
360 
361         /**
362          * @brief Checks for a temperature fault or warning condition.
363          *
364          * The low byte of STATUS_WORD is checked to see if the "TEMPERATURE
365          * FAULT OR WARNING" bit is turned on. If it is on, log an error,
366          * call out the power supply indicating the fault/warning condition.
367          *
368          * @parma[in] statusWord - 2 byte STATUS_WORD value read from sysfs
369          */
370         void checkTemperatureFault(const uint16_t statusWord);
371 
372         /**
373          * @brief Adds properties to the inventory.
374          *
375          * Reads the values from the device and writes them to the
376          * associated power supply D-Bus inventory object.
377          *
378          * This needs to be done on startup, and each time the presence
379          * state changes.
380          *
381          * Properties added:
382          * - Serial Number
383          * - Part Number
384          * - CCIN (Customer Card Identification Number) - added as the Model
385          * - Firmware version
386          */
387         void updateInventory();
388 
389         /**
390          * @brief Toggles the GPIO to sync power supply input history readings
391          *
392          * This GPIO is connected to all supplies.  This will clear the
393          * previous readings out of the supplies and restart them both at the
394          * same time zero and at record ID 0.  The supplies will return 0
395          * bytes of data for the input history command right after this until
396          * a new entry shows up.
397          *
398          * This will cause the code to delete all previous history data and
399          * start fresh.
400          */
401         void syncHistory();
402 
403         /**
404          * @brief Reads the most recent input history record from the power
405          *        supply and updates the average and maximum properties in
406          *        D-Bus if there is a new reading available.
407          *
408          * This will still run every time analyze() is called so code can
409          * post new data as soon as possible and the timestamp will more
410          * accurately reflect the correct time.
411          *
412          * D-Bus is only updated if there is a change and the oldest record
413          * will be pruned if the property already contains the max number of
414          * records.
415          */
416         void updateHistory();
417 };
418 
419 }
420 }
421 }
422