xref: /openbmc/bmcweb/redfish-core/lib/power.hpp (revision 6fd29553)
1 /*
2 // Copyright (c) 2018 Intel Corporation
3 // Copyright (c) 2018 Ampere Computing LLC
4 /
5 // Licensed under the Apache License, Version 2.0 (the "License");
6 // you may not use this file except in compliance with the License.
7 // You may obtain a copy of the License at
8 //
9 //      http://www.apache.org/licenses/LICENSE-2.0
10 //
11 // Unless required by applicable law or agreed to in writing, software
12 // distributed under the License is distributed on an "AS IS" BASIS,
13 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 // See the License for the specific language governing permissions and
15 // limitations under the License.
16 */
17 #pragma once
18 
19 #include "app.hpp"
20 #include "dbus_utility.hpp"
21 #include "query.hpp"
22 #include "registries/privilege_registry.hpp"
23 #include "sensors.hpp"
24 #include "utils/chassis_utils.hpp"
25 
26 #include <sdbusplus/asio/property.hpp>
27 
28 #include <array>
29 #include <string_view>
30 
31 namespace redfish
32 {
33 inline void setPowerCapOverride(
34     const std::shared_ptr<SensorsAsyncResp>& sensorsAsyncResp,
35     std::vector<nlohmann::json>& powerControlCollections)
36 {
37     auto getChassisPath =
38         [sensorsAsyncResp, powerControlCollections](
39             const std::optional<std::string>& chassisPath) mutable {
40         if (!chassisPath)
41         {
42             BMCWEB_LOG_WARNING("Don't find valid chassis path ");
43             messages::resourceNotFound(sensorsAsyncResp->asyncResp->res,
44                                        "Chassis", sensorsAsyncResp->chassisId);
45             return;
46         }
47 
48         if (powerControlCollections.size() != 1)
49         {
50             BMCWEB_LOG_WARNING("Don't support multiple hosts at present ");
51             messages::resourceNotFound(sensorsAsyncResp->asyncResp->res,
52                                        "Power", "PowerControl");
53             return;
54         }
55 
56         auto& item = powerControlCollections[0];
57 
58         std::optional<nlohmann::json> powerLimit;
59         if (!json_util::readJson(item, sensorsAsyncResp->asyncResp->res,
60                                  "PowerLimit", powerLimit))
61         {
62             return;
63         }
64         if (!powerLimit)
65         {
66             return;
67         }
68         std::optional<uint32_t> value;
69         if (!json_util::readJson(*powerLimit, sensorsAsyncResp->asyncResp->res,
70                                  "LimitInWatts", value))
71         {
72             return;
73         }
74         if (!value)
75         {
76             return;
77         }
78         sdbusplus::asio::getProperty<bool>(
79             *crow::connections::systemBus, "xyz.openbmc_project.Settings",
80             "/xyz/openbmc_project/control/host0/power_cap",
81             "xyz.openbmc_project.Control.Power.Cap", "PowerCapEnable",
82             [value, sensorsAsyncResp](const boost::system::error_code& ec,
83                                       bool powerCapEnable) {
84             if (ec)
85             {
86                 messages::internalError(sensorsAsyncResp->asyncResp->res);
87                 BMCWEB_LOG_ERROR("powerCapEnable Get handler: Dbus error {}",
88                                  ec);
89                 return;
90             }
91             if (!powerCapEnable)
92             {
93                 messages::actionNotSupported(
94                     sensorsAsyncResp->asyncResp->res,
95                     "Setting LimitInWatts when PowerLimit feature is disabled");
96                 BMCWEB_LOG_ERROR("PowerLimit feature is disabled ");
97                 return;
98             }
99 
100             sdbusplus::asio::setProperty(
101                 *crow::connections::systemBus, "xyz.openbmc_project.Settings",
102                 "/xyz/openbmc_project/control/host0/power_cap",
103                 "xyz.openbmc_project.Control.Power.Cap", "PowerCap", *value,
104                 [sensorsAsyncResp](const boost::system::error_code& ec2) {
105                 if (ec2)
106                 {
107                     BMCWEB_LOG_DEBUG("Power Limit Set: Dbus error: {}", ec2);
108                     messages::internalError(sensorsAsyncResp->asyncResp->res);
109                     return;
110                 }
111                 sensorsAsyncResp->asyncResp->res.result(
112                     boost::beast::http::status::no_content);
113                 });
114             });
115     };
116     redfish::chassis_utils::getValidChassisPath(sensorsAsyncResp->asyncResp,
117                                                 sensorsAsyncResp->chassisId,
118                                                 std::move(getChassisPath));
119 }
120 inline void requestRoutesPower(App& app)
121 {
122     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Power/")
123         .privileges(redfish::privileges::getPower)
124         .methods(boost::beast::http::verb::get)(
125             [&app](const crow::Request& req,
126                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
127                    const std::string& chassisName) {
128         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
129         {
130             return;
131         }
132         asyncResp->res.jsonValue["PowerControl"] = nlohmann::json::array();
133 
134         auto sensorAsyncResp = std::make_shared<SensorsAsyncResp>(
135             asyncResp, chassisName, sensors::dbus::powerPaths,
136             sensors::node::power);
137 
138         getChassisData(sensorAsyncResp);
139 
140         // This callback verifies that the power limit is only provided
141         // for the chassis that implements the Chassis inventory item.
142         // This prevents things like power supplies providing the
143         // chassis power limit
144 
145         using Mapper = dbus::utility::MapperGetSubTreePathsResponse;
146         auto chassisHandler =
147             [sensorAsyncResp](const boost::system::error_code& ec2,
148                               const Mapper& chassisPaths) {
149             if (ec2)
150             {
151                 BMCWEB_LOG_ERROR(
152                     "Power Limit GetSubTreePaths handler Dbus error {}", ec2);
153                 return;
154             }
155 
156             bool found = false;
157             for (const std::string& chassis : chassisPaths)
158             {
159                 size_t len = std::string::npos;
160                 size_t lastPos = chassis.rfind('/');
161                 if (lastPos == std::string::npos)
162                 {
163                     continue;
164                 }
165 
166                 if (lastPos == chassis.size() - 1)
167                 {
168                     size_t end = lastPos;
169                     lastPos = chassis.rfind('/', lastPos - 1);
170                     if (lastPos == std::string::npos)
171                     {
172                         continue;
173                     }
174 
175                     len = end - (lastPos + 1);
176                 }
177 
178                 std::string interfaceChassisName = chassis.substr(lastPos + 1,
179                                                                   len);
180                 if (interfaceChassisName == sensorAsyncResp->chassisId)
181                 {
182                     found = true;
183                     break;
184                 }
185             }
186 
187             if (!found)
188             {
189                 BMCWEB_LOG_DEBUG("Power Limit not present for {}",
190                                  sensorAsyncResp->chassisId);
191                 return;
192             }
193 
194             auto valueHandler =
195                 [sensorAsyncResp](
196                     const boost::system::error_code& ec,
197                     const dbus::utility::DBusPropertiesMap& properties) {
198                 if (ec)
199                 {
200                     messages::internalError(sensorAsyncResp->asyncResp->res);
201                     BMCWEB_LOG_ERROR(
202                         "Power Limit GetAll handler: Dbus error {}", ec);
203                     return;
204                 }
205 
206                 nlohmann::json& tempArray =
207                     sensorAsyncResp->asyncResp->res.jsonValue["PowerControl"];
208 
209                 // Put multiple "sensors" into a single PowerControl, 0,
210                 // so only create the first one
211                 if (tempArray.empty())
212                 {
213                     // Mandatory properties odata.id and MemberId
214                     // A warning without a odata.type
215                     nlohmann::json::object_t powerControl;
216                     powerControl["@odata.type"] = "#Power.v1_0_0.PowerControl";
217                     powerControl["@odata.id"] = "/redfish/v1/Chassis/" +
218                                                 sensorAsyncResp->chassisId +
219                                                 "/Power#/PowerControl/0";
220                     powerControl["Name"] = "Chassis Power Control";
221                     powerControl["MemberId"] = "0";
222                     tempArray.emplace_back(std::move(powerControl));
223                 }
224 
225                 nlohmann::json& sensorJson = tempArray.back();
226                 bool enabled = false;
227                 double powerCap = 0.0;
228                 int64_t scale = 0;
229 
230                 for (const std::pair<std::string,
231                                      dbus::utility::DbusVariantType>& property :
232                      properties)
233                 {
234                     if (property.first == "Scale")
235                     {
236                         const int64_t* i =
237                             std::get_if<int64_t>(&property.second);
238 
239                         if (i != nullptr)
240                         {
241                             scale = *i;
242                         }
243                     }
244                     else if (property.first == "PowerCap")
245                     {
246                         const double* d = std::get_if<double>(&property.second);
247                         const int64_t* i =
248                             std::get_if<int64_t>(&property.second);
249                         const uint32_t* u =
250                             std::get_if<uint32_t>(&property.second);
251 
252                         if (d != nullptr)
253                         {
254                             powerCap = *d;
255                         }
256                         else if (i != nullptr)
257                         {
258                             powerCap = static_cast<double>(*i);
259                         }
260                         else if (u != nullptr)
261                         {
262                             powerCap = *u;
263                         }
264                     }
265                     else if (property.first == "PowerCapEnable")
266                     {
267                         const bool* b = std::get_if<bool>(&property.second);
268 
269                         if (b != nullptr)
270                         {
271                             enabled = *b;
272                         }
273                     }
274                 }
275 
276                 // LimitException is Mandatory attribute as per OCP
277                 // Baseline Profile – v1.0.0, so currently making it
278                 // "NoAction" as default value to make it OCP Compliant.
279                 sensorJson["PowerLimit"]["LimitException"] = "NoAction";
280 
281                 if (enabled)
282                 {
283                     // Redfish specification indicates PowerLimit should
284                     // be null if the limit is not enabled.
285                     sensorJson["PowerLimit"]["LimitInWatts"] =
286                         powerCap * std::pow(10, scale);
287                 }
288             };
289 
290             sdbusplus::asio::getAllProperties(
291                 *crow::connections::systemBus, "xyz.openbmc_project.Settings",
292                 "/xyz/openbmc_project/control/host0/power_cap",
293                 "xyz.openbmc_project.Control.Power.Cap",
294                 std::move(valueHandler));
295         };
296 
297         constexpr std::array<std::string_view, 2> interfaces = {
298             "xyz.openbmc_project.Inventory.Item.Board",
299             "xyz.openbmc_project.Inventory.Item.Chassis"};
300 
301         dbus::utility::getSubTreePaths("/xyz/openbmc_project/inventory", 0,
302                                        interfaces, std::move(chassisHandler));
303         });
304 
305     BMCWEB_ROUTE(app, "/redfish/v1/Chassis/<str>/Power/")
306         .privileges(redfish::privileges::patchPower)
307         .methods(boost::beast::http::verb::patch)(
308             [&app](const crow::Request& req,
309                    const std::shared_ptr<bmcweb::AsyncResp>& asyncResp,
310                    const std::string& chassisName) {
311         if (!redfish::setUpRedfishRoute(app, req, asyncResp))
312         {
313             return;
314         }
315         auto sensorAsyncResp = std::make_shared<SensorsAsyncResp>(
316             asyncResp, chassisName, sensors::dbus::powerPaths,
317             sensors::node::power);
318 
319         std::optional<std::vector<nlohmann::json>> voltageCollections;
320         std::optional<std::vector<nlohmann::json>> powerCtlCollections;
321 
322         if (!json_util::readJsonPatch(req, sensorAsyncResp->asyncResp->res,
323                                       "PowerControl", powerCtlCollections,
324                                       "Voltages", voltageCollections))
325         {
326             return;
327         }
328 
329         if (powerCtlCollections)
330         {
331             setPowerCapOverride(sensorAsyncResp, *powerCtlCollections);
332         }
333         if (voltageCollections)
334         {
335             std::unordered_map<std::string, std::vector<nlohmann::json>>
336                 allCollections;
337             allCollections.emplace("Voltages", *std::move(voltageCollections));
338             setSensorsOverride(sensorAsyncResp, allCollections);
339         }
340         });
341 }
342 
343 } // namespace redfish
344