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