1 /** 2 * Copyright © 2017 IBM Corporation 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 #include "utility.hpp" 17 18 #include "types.hpp" 19 20 #include <fstream> 21 22 namespace phosphor 23 { 24 namespace power 25 { 26 namespace util 27 { 28 29 constexpr auto MAPPER_BUSNAME = "xyz.openbmc_project.ObjectMapper"; 30 constexpr auto MAPPER_PATH = "/xyz/openbmc_project/object_mapper"; 31 constexpr auto MAPPER_INTERFACE = "xyz.openbmc_project.ObjectMapper"; 32 33 using namespace phosphor::logging; 34 using json = nlohmann::json; 35 36 std::string getService(const std::string& path, const std::string& interface, 37 sdbusplus::bus_t& bus, bool logError) 38 { 39 auto method = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH, 40 MAPPER_INTERFACE, "GetObject"); 41 42 method.append(path); 43 method.append(std::vector<std::string>({interface})); 44 45 auto reply = bus.call(method); 46 47 std::map<std::string, std::vector<std::string>> response; 48 reply.read(response); 49 50 if (response.empty()) 51 { 52 if (logError) 53 { 54 log<level::ERR>( 55 std::string("Error in mapper response for getting service name " 56 "PATH=" + 57 path + " INTERFACE=" + interface) 58 .c_str()); 59 } 60 return std::string{}; 61 } 62 63 return response.begin()->first; 64 } 65 66 DbusPropertyMap getAllProperties(sdbusplus::bus_t& bus, const std::string& path, 67 const std::string& interface, 68 const std::string& service) 69 { 70 DbusPropertyMap properties; 71 72 auto serviceStr = service; 73 if (serviceStr.empty()) 74 { 75 serviceStr = getService(path, interface, bus); 76 if (serviceStr.empty()) 77 { 78 return properties; 79 } 80 } 81 82 auto method = bus.new_method_call(serviceStr.c_str(), path.c_str(), 83 PROPERTY_INTF, "GetAll"); 84 method.append(interface); 85 auto reply = bus.call(method); 86 reply.read(properties); 87 return properties; 88 } 89 90 DbusSubtree getSubTree(sdbusplus::bus_t& bus, const std::string& path, 91 const std::string& interface, int32_t depth) 92 { 93 return getSubTree(bus, path, std::vector<std::string>({interface}), depth); 94 } 95 96 DbusSubtree getSubTree(sdbusplus::bus_t& bus, const std::string& path, 97 const std::vector<std::string>& interfaces, 98 int32_t depth) 99 { 100 auto mapperCall = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH, 101 MAPPER_INTERFACE, "GetSubTree"); 102 mapperCall.append(path); 103 mapperCall.append(depth); 104 mapperCall.append(interfaces); 105 106 auto reply = bus.call(mapperCall); 107 108 DbusSubtree response; 109 reply.read(response); 110 return response; 111 } 112 113 std::vector<DbusPath> getAssociatedSubTreePaths( 114 sdbusplus::bus_t& bus, 115 const sdbusplus::message::object_path& associationPath, 116 const sdbusplus::message::object_path& path, 117 const std::vector<std::string>& interfaces, int32_t depth) 118 { 119 auto mapperCall = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH, 120 MAPPER_INTERFACE, 121 "GetAssociatedSubTreePaths"); 122 mapperCall.append(associationPath); 123 mapperCall.append(path); 124 mapperCall.append(depth); 125 mapperCall.append(interfaces); 126 127 auto reply = bus.call(mapperCall); 128 129 std::vector<DbusPath> response; 130 reply.read(response); 131 return response; 132 } 133 134 json loadJSONFromFile(const char* path) 135 { 136 std::ifstream ifs(path); 137 if (!ifs.good()) 138 { 139 log<level::ERR>(std::string("Unable to open file " 140 "PATH=" + 141 std::string(path)) 142 .c_str()); 143 return nullptr; 144 } 145 auto data = json::parse(ifs, nullptr, false); 146 if (data.is_discarded()) 147 { 148 log<level::ERR>(std::string("Failed to parse json " 149 "PATH=" + 150 std::string(path)) 151 .c_str()); 152 return nullptr; 153 } 154 return data; 155 } 156 157 phosphor::pmbus::Type getPMBusAccessType(const json& json) 158 { 159 using namespace phosphor::pmbus; 160 Type type; 161 162 auto typeStr = json.at("inventoryPMBusAccessType"); 163 164 if (typeStr == "Hwmon") 165 { 166 type = Type::Hwmon; 167 } 168 else if (typeStr == "DeviceDebug") 169 { 170 type = Type::DeviceDebug; 171 } 172 else if (typeStr == "Debug") 173 { 174 type = Type::Debug; 175 } 176 else if (typeStr == "HwmonDeviceDebug") 177 { 178 type = Type::HwmonDeviceDebug; 179 } 180 else 181 { 182 type = Type::Base; 183 } 184 return type; 185 } 186 187 bool isPoweredOn(sdbusplus::bus_t& bus, bool defaultState) 188 { 189 int32_t state = defaultState; 190 191 try 192 { 193 // When state = 1, system is powered on 194 auto service = util::getService(POWER_OBJ_PATH, POWER_IFACE, bus); 195 getProperty<int32_t>(POWER_IFACE, "state", POWER_OBJ_PATH, service, bus, 196 state); 197 } 198 catch (const std::exception& e) 199 { 200 log<level::INFO>("Failed to get power state."); 201 } 202 return state != 0; 203 } 204 205 std::vector<std::string> getPSUInventoryPaths(sdbusplus::bus_t& bus) 206 { 207 std::vector<std::string> paths; 208 auto method = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH, 209 MAPPER_INTERFACE, "GetSubTreePaths"); 210 method.append(INVENTORY_OBJ_PATH); 211 method.append(0); // Depth 0 to search all 212 method.append(std::vector<std::string>({PSU_INVENTORY_IFACE})); 213 auto reply = bus.call(method); 214 215 reply.read(paths); 216 return paths; 217 } 218 219 } // namespace util 220 } // namespace power 221 } // namespace phosphor 222