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