blob: 170bf4d567e6fd071b4c85a35809c487da05fecc [file] [log] [blame]
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001/*
2// Copyright (c) 2018 Intel 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#pragma once
17
James Feistb49ac872019-05-21 15:12:01 -070018#include "health.hpp"
James Feist1c8fba92019-12-20 15:12:07 -080019#include "led.hpp"
Jason M. Billsf5c9f8b2018-12-18 16:51:18 -080020#include "pcie.hpp"
Jennifer Leec5d03ff2019-03-08 15:42:58 -080021#include "redfish_util.hpp"
22
Ed Tanous9712f8a2018-09-21 13:38:49 -070023#include <boost/container/flat_map.hpp>
24#include <node.hpp>
Andrew Geisslercb7e1e72019-02-19 13:05:38 -060025#include <utils/fw_utils.hpp>
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +020026#include <utils/json_utils.hpp>
Gunnar Mills1214b7e2020-06-04 10:11:30 -050027
Ed Tanousabf2add2019-01-22 16:40:12 -080028#include <variant>
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +020029
Ed Tanous1abe55e2018-09-05 08:30:59 -070030namespace redfish
31{
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +020032
Alpana Kumari9d3ae102019-04-12 06:49:32 -050033/**
34 * @brief Updates the Functional State of DIMMs
35 *
36 * @param[in] aResp Shared pointer for completing asynchronous calls
37 * @param[in] dimmState Dimm's Functional state, true/false
38 *
39 * @return None.
40 */
41void updateDimmProperties(std::shared_ptr<AsyncResp> aResp,
Gunnar Mills1214b7e2020-06-04 10:11:30 -050042 const std::variant<bool>& dimmState)
Alpana Kumari9d3ae102019-04-12 06:49:32 -050043{
Gunnar Mills1214b7e2020-06-04 10:11:30 -050044 const bool* isDimmFunctional = std::get_if<bool>(&dimmState);
Alpana Kumari9d3ae102019-04-12 06:49:32 -050045 if (isDimmFunctional == nullptr)
46 {
47 messages::internalError(aResp->res);
48 return;
49 }
Gunnar Mills698654b2019-10-16 13:17:37 -050050 BMCWEB_LOG_DEBUG << "Dimm Functional: " << *isDimmFunctional;
Alpana Kumari9d3ae102019-04-12 06:49:32 -050051
52 // Set it as Enabled if atleast one DIMM is functional
53 // Update STATE only if previous State was DISABLED and current Dimm is
54 // ENABLED.
Gunnar Mills1214b7e2020-06-04 10:11:30 -050055 nlohmann::json& prevMemSummary =
Alpana Kumari9d3ae102019-04-12 06:49:32 -050056 aResp->res.jsonValue["MemorySummary"]["Status"]["State"];
57 if (prevMemSummary == "Disabled")
58 {
59 if (*isDimmFunctional == true)
60 {
61 aResp->res.jsonValue["MemorySummary"]["Status"]["State"] =
62 "Enabled";
63 }
64 }
65}
66
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050067/*
68 * @brief Update "ProcessorSummary" "Count" based on Cpu PresenceState
69 *
70 * @param[in] aResp Shared pointer for completing asynchronous calls
71 * @param[in] cpuPresenceState CPU present or not
72 *
73 * @return None.
74 */
75void modifyCpuPresenceState(std::shared_ptr<AsyncResp> aResp,
Gunnar Mills1214b7e2020-06-04 10:11:30 -050076 const std::variant<bool>& cpuPresenceState)
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050077{
Gunnar Mills1214b7e2020-06-04 10:11:30 -050078 const bool* isCpuPresent = std::get_if<bool>(&cpuPresenceState);
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050079
80 if (isCpuPresent == nullptr)
81 {
82 messages::internalError(aResp->res);
83 return;
84 }
Gunnar Mills698654b2019-10-16 13:17:37 -050085 BMCWEB_LOG_DEBUG << "Cpu Present: " << *isCpuPresent;
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050086
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050087 if (*isCpuPresent == true)
88 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -050089 nlohmann::json& procCount =
James Feistb4b95952019-12-05 15:01:55 -080090 aResp->res.jsonValue["ProcessorSummary"]["Count"];
91 auto procCountPtr =
Gunnar Mills1214b7e2020-06-04 10:11:30 -050092 procCount.get_ptr<nlohmann::json::number_integer_t*>();
James Feistb4b95952019-12-05 15:01:55 -080093 if (procCountPtr != nullptr)
94 {
95 // shouldn't be possible to be nullptr
96 *procCountPtr += 1;
97 }
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050098 }
Alpana Kumari57e8c9b2019-04-15 01:09:36 -050099}
100
101/*
102 * @brief Update "ProcessorSummary" "Status" "State" based on
103 * CPU Functional State
104 *
105 * @param[in] aResp Shared pointer for completing asynchronous calls
106 * @param[in] cpuFunctionalState is CPU functional true/false
107 *
108 * @return None.
109 */
110void modifyCpuFunctionalState(std::shared_ptr<AsyncResp> aResp,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500111 const std::variant<bool>& cpuFunctionalState)
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500112{
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500113 const bool* isCpuFunctional = std::get_if<bool>(&cpuFunctionalState);
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500114
115 if (isCpuFunctional == nullptr)
116 {
117 messages::internalError(aResp->res);
118 return;
119 }
Gunnar Mills698654b2019-10-16 13:17:37 -0500120 BMCWEB_LOG_DEBUG << "Cpu Functional: " << *isCpuFunctional;
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500121
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500122 nlohmann::json& prevProcState =
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500123 aResp->res.jsonValue["ProcessorSummary"]["Status"]["State"];
124
125 // Set it as Enabled if atleast one CPU is functional
126 // Update STATE only if previous State was Non_Functional and current CPU is
127 // Functional.
128 if (prevProcState == "Disabled")
129 {
130 if (*isCpuFunctional == true)
131 {
132 aResp->res.jsonValue["ProcessorSummary"]["Status"]["State"] =
133 "Enabled";
134 }
135 }
136}
137
138/*
Ed Tanous6c34de42018-08-29 13:37:36 -0700139 * @brief Retrieves computer system properties over dbus
140 *
141 * @param[in] aResp Shared pointer for completing asynchronous calls
142 * @param[in] name Computer system name from request
143 *
144 * @return None.
145 */
James Feist5bc2dc82019-10-22 14:33:16 -0700146void getComputerSystem(std::shared_ptr<AsyncResp> aResp,
147 std::shared_ptr<HealthPopulate> systemHealth)
Ed Tanous6c34de42018-08-29 13:37:36 -0700148{
Ed Tanous6c34de42018-08-29 13:37:36 -0700149 BMCWEB_LOG_DEBUG << "Get available system components.";
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500150
Ed Tanous6c34de42018-08-29 13:37:36 -0700151 crow::connections::systemBus->async_method_call(
James Feist5bc2dc82019-10-22 14:33:16 -0700152 [aResp, systemHealth](
Ed Tanous6c34de42018-08-29 13:37:36 -0700153 const boost::system::error_code ec,
154 const std::vector<std::pair<
155 std::string,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500156 std::vector<std::pair<std::string, std::vector<std::string>>>>>&
157 subtree) {
Ed Tanous6c34de42018-08-29 13:37:36 -0700158 if (ec)
159 {
160 BMCWEB_LOG_DEBUG << "DBUS response error";
Jason M. Billsf12894f2018-10-09 12:45:45 -0700161 messages::internalError(aResp->res);
Ed Tanous6c34de42018-08-29 13:37:36 -0700162 return;
163 }
Ed Tanous6c34de42018-08-29 13:37:36 -0700164 // Iterate over all retrieved ObjectPaths.
165 for (const std::pair<std::string,
166 std::vector<std::pair<
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500167 std::string, std::vector<std::string>>>>&
168 object : subtree)
Ed Tanous6c34de42018-08-29 13:37:36 -0700169 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500170 const std::string& path = object.first;
Ed Tanous6c34de42018-08-29 13:37:36 -0700171 BMCWEB_LOG_DEBUG << "Got path: " << path;
172 const std::vector<
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500173 std::pair<std::string, std::vector<std::string>>>&
174 connectionNames = object.second;
Ed Tanous6c34de42018-08-29 13:37:36 -0700175 if (connectionNames.size() < 1)
176 {
177 continue;
178 }
Ed Tanous029573d2019-02-01 10:57:49 -0800179
James Feist5bc2dc82019-10-22 14:33:16 -0700180 auto memoryHealth = std::make_shared<HealthPopulate>(
181 aResp, aResp->res.jsonValue["MemorySummary"]["Status"]);
182
183 auto cpuHealth = std::make_shared<HealthPopulate>(
184 aResp, aResp->res.jsonValue["ProcessorSummary"]["Status"]);
185
186 systemHealth->children.emplace_back(memoryHealth);
187 systemHealth->children.emplace_back(cpuHealth);
188
Ed Tanous029573d2019-02-01 10:57:49 -0800189 // This is not system, so check if it's cpu, dimm, UUID or
190 // BiosVer
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500191 for (const auto& connection : connectionNames)
Ed Tanous6c34de42018-08-29 13:37:36 -0700192 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500193 for (const auto& interfaceName : connection.second)
Ed Tanous6c34de42018-08-29 13:37:36 -0700194 {
Ed Tanous029573d2019-02-01 10:57:49 -0800195 if (interfaceName ==
196 "xyz.openbmc_project.Inventory.Item.Dimm")
Ed Tanous6c34de42018-08-29 13:37:36 -0700197 {
Ed Tanous029573d2019-02-01 10:57:49 -0800198 BMCWEB_LOG_DEBUG
199 << "Found Dimm, now get its properties.";
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500200
Ed Tanous029573d2019-02-01 10:57:49 -0800201 crow::connections::systemBus->async_method_call(
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500202 [aResp, service{connection.first},
203 path(std::move(path))](
204 const boost::system::error_code ec,
205 const std::vector<
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500206 std::pair<std::string, VariantType>>&
207 properties) {
Ed Tanous029573d2019-02-01 10:57:49 -0800208 if (ec)
209 {
210 BMCWEB_LOG_ERROR
211 << "DBUS response error " << ec;
212 messages::internalError(aResp->res);
213 return;
214 }
215 BMCWEB_LOG_DEBUG << "Got "
216 << properties.size()
Gunnar Mills698654b2019-10-16 13:17:37 -0500217 << " Dimm properties.";
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500218
219 if (properties.size() > 0)
Ed Tanous029573d2019-02-01 10:57:49 -0800220 {
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500221 for (const std::pair<std::string,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500222 VariantType>&
223 property : properties)
Ed Tanous6c34de42018-08-29 13:37:36 -0700224 {
Cheng C Yang5fd7ba62019-11-28 15:58:08 +0800225 if (property.first !=
226 "MemorySizeInKB")
Ed Tanous6c34de42018-08-29 13:37:36 -0700227 {
Cheng C Yang5fd7ba62019-11-28 15:58:08 +0800228 continue;
Ed Tanous6c34de42018-08-29 13:37:36 -0700229 }
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500230 const uint32_t* value =
Patrick Williams8d78b7a2020-05-13 11:24:20 -0500231 std::get_if<uint32_t>(
232 &property.second);
Cheng C Yang5fd7ba62019-11-28 15:58:08 +0800233 if (value == nullptr)
234 {
235 BMCWEB_LOG_DEBUG
236 << "Find incorrect type of "
237 "MemorySize";
238 continue;
239 }
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500240 nlohmann::json& totalMemory =
Cheng C Yang5fd7ba62019-11-28 15:58:08 +0800241 aResp->res
242 .jsonValue["MemorySummar"
243 "y"]
244 ["TotalSystemMe"
245 "moryGiB"];
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500246 uint64_t* preValue =
Cheng C Yang5fd7ba62019-11-28 15:58:08 +0800247 totalMemory
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500248 .get_ptr<uint64_t*>();
Cheng C Yang5fd7ba62019-11-28 15:58:08 +0800249 if (preValue == nullptr)
250 {
251 continue;
252 }
253 aResp->res
254 .jsonValue["MemorySummary"]
255 ["TotalSystemMemoryGi"
256 "B"] =
257 *value / (1024 * 1024) +
258 *preValue;
259 aResp->res
260 .jsonValue["MemorySummary"]
261 ["Status"]["State"] =
262 "Enabled";
Ed Tanous6c34de42018-08-29 13:37:36 -0700263 }
Ed Tanous029573d2019-02-01 10:57:49 -0800264 }
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500265 else
266 {
267 auto getDimmProperties =
268 [aResp](
269 const boost::system::error_code
270 ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500271 const std::variant<bool>&
272 dimmState) {
Alpana Kumari9d3ae102019-04-12 06:49:32 -0500273 if (ec)
274 {
275 BMCWEB_LOG_ERROR
276 << "DBUS response "
277 "error "
278 << ec;
279 return;
280 }
281 updateDimmProperties(aResp,
282 dimmState);
283 };
284 crow::connections::systemBus
285 ->async_method_call(
286 std::move(getDimmProperties),
287 service, path,
288 "org.freedesktop.DBus."
289 "Properties",
290 "Get",
291 "xyz.openbmc_project.State."
292 "Decorator.OperationalStatus",
293 "Functional");
294 }
Ed Tanous029573d2019-02-01 10:57:49 -0800295 },
296 connection.first, path,
297 "org.freedesktop.DBus.Properties", "GetAll",
298 "xyz.openbmc_project.Inventory.Item.Dimm");
James Feist5bc2dc82019-10-22 14:33:16 -0700299
300 memoryHealth->inventory.emplace_back(path);
Ed Tanous029573d2019-02-01 10:57:49 -0800301 }
302 else if (interfaceName ==
303 "xyz.openbmc_project.Inventory.Item.Cpu")
304 {
305 BMCWEB_LOG_DEBUG
306 << "Found Cpu, now get its properties.";
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500307
Ed Tanous029573d2019-02-01 10:57:49 -0800308 crow::connections::systemBus->async_method_call(
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500309 [aResp, service{connection.first},
310 path(std::move(path))](
311 const boost::system::error_code ec,
312 const std::vector<
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500313 std::pair<std::string, VariantType>>&
314 properties) {
Ed Tanous029573d2019-02-01 10:57:49 -0800315 if (ec)
316 {
317 BMCWEB_LOG_ERROR
318 << "DBUS response error " << ec;
319 messages::internalError(aResp->res);
320 return;
321 }
322 BMCWEB_LOG_DEBUG << "Got "
323 << properties.size()
Gunnar Mills698654b2019-10-16 13:17:37 -0500324 << " Cpu properties.";
Ed Tanous04a258f2018-10-15 08:00:41 -0700325
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500326 if (properties.size() > 0)
327 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500328 for (const auto& property : properties)
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500329 {
330 if (property.first ==
331 "ProcessorFamily")
332 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500333 const std::string* value =
Patrick Williams8d78b7a2020-05-13 11:24:20 -0500334 std::get_if<std::string>(
335 &property.second);
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500336 if (value != nullptr)
337 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500338 nlohmann::json&
339 procSummary =
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500340 aResp->res.jsonValue
341 ["ProcessorSumm"
342 "ary"];
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500343 nlohmann::json& procCount =
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500344 procSummary["Count"];
James Feistb4b95952019-12-05 15:01:55 -0800345
346 auto procCountPtr =
347 procCount.get_ptr<
348 nlohmann::json::
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500349 number_integer_t*>();
James Feistb4b95952019-12-05 15:01:55 -0800350 if (procCountPtr != nullptr)
351 {
352 // shouldn't be possible
353 // to be nullptr
354 *procCountPtr += 1;
355 }
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500356 procSummary["Status"]
357 ["State"] =
358 "Enabled";
359 procSummary["Model"] =
360 *value;
361 }
Ed Tanous6c34de42018-08-29 13:37:36 -0700362 }
363 }
Ed Tanous029573d2019-02-01 10:57:49 -0800364 }
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500365 else
366 {
367 auto getCpuPresenceState =
368 [aResp](
369 const boost::system::error_code
370 ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500371 const std::variant<bool>&
372 cpuPresenceCheck) {
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500373 if (ec)
374 {
375 BMCWEB_LOG_ERROR
376 << "DBUS response "
377 "error "
378 << ec;
379 return;
380 }
381 modifyCpuPresenceState(
382 aResp, cpuPresenceCheck);
383 };
384
385 auto getCpuFunctionalState =
386 [aResp](
387 const boost::system::error_code
388 ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500389 const std::variant<bool>&
390 cpuFunctionalCheck) {
Alpana Kumari57e8c9b2019-04-15 01:09:36 -0500391 if (ec)
392 {
393 BMCWEB_LOG_ERROR
394 << "DBUS response "
395 "error "
396 << ec;
397 return;
398 }
399 modifyCpuFunctionalState(
400 aResp, cpuFunctionalCheck);
401 };
402 // Get the Presence of CPU
403 crow::connections::systemBus
404 ->async_method_call(
405 std::move(getCpuPresenceState),
406 service, path,
407 "org.freedesktop.DBus."
408 "Properties",
409 "Get",
410 "xyz.openbmc_project.Inventory."
411 "Item",
412 "Present");
413
414 // Get the Functional State
415 crow::connections::systemBus
416 ->async_method_call(
417 std::move(
418 getCpuFunctionalState),
419 service, path,
420 "org.freedesktop.DBus."
421 "Properties",
422 "Get",
423 "xyz.openbmc_project.State."
424 "Decorator."
425 "OperationalStatus",
426 "Functional");
427
428 // Get the MODEL from
429 // xyz.openbmc_project.Inventory.Decorator.Asset
430 // support it later as Model is Empty
431 // currently.
432 }
Ed Tanous029573d2019-02-01 10:57:49 -0800433 },
434 connection.first, path,
435 "org.freedesktop.DBus.Properties", "GetAll",
436 "xyz.openbmc_project.Inventory.Item.Cpu");
James Feist5bc2dc82019-10-22 14:33:16 -0700437
438 cpuHealth->inventory.emplace_back(path);
Ed Tanous029573d2019-02-01 10:57:49 -0800439 }
440 else if (interfaceName ==
441 "xyz.openbmc_project.Common.UUID")
442 {
443 BMCWEB_LOG_DEBUG
444 << "Found UUID, now get its properties.";
445 crow::connections::systemBus->async_method_call(
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500446 [aResp](
447 const boost::system::error_code ec,
448 const std::vector<
449 std::pair<std::string, VariantType>>&
450 properties) {
Ed Tanous029573d2019-02-01 10:57:49 -0800451 if (ec)
452 {
453 BMCWEB_LOG_DEBUG
454 << "DBUS response error " << ec;
455 messages::internalError(aResp->res);
456 return;
457 }
458 BMCWEB_LOG_DEBUG << "Got "
459 << properties.size()
Gunnar Mills698654b2019-10-16 13:17:37 -0500460 << " UUID properties.";
Ed Tanous029573d2019-02-01 10:57:49 -0800461 for (const std::pair<std::string,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500462 VariantType>&
463 property : properties)
Ed Tanous029573d2019-02-01 10:57:49 -0800464 {
Ed Tanous029573d2019-02-01 10:57:49 -0800465 if (property.first == "UUID")
466 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500467 const std::string* value =
Patrick Williams8d78b7a2020-05-13 11:24:20 -0500468 std::get_if<std::string>(
469 &property.second);
Ed Tanous04a258f2018-10-15 08:00:41 -0700470
Ed Tanous029573d2019-02-01 10:57:49 -0800471 if (value != nullptr)
472 {
473 std::string valueStr = *value;
474 if (valueStr.size() == 32)
Ed Tanous6c34de42018-08-29 13:37:36 -0700475 {
Ed Tanous029573d2019-02-01 10:57:49 -0800476 valueStr.insert(8, 1, '-');
477 valueStr.insert(13, 1, '-');
478 valueStr.insert(18, 1, '-');
479 valueStr.insert(23, 1, '-');
Ed Tanous6c34de42018-08-29 13:37:36 -0700480 }
Ed Tanous029573d2019-02-01 10:57:49 -0800481 BMCWEB_LOG_DEBUG << "UUID = "
482 << valueStr;
483 aResp->res.jsonValue["UUID"] =
484 valueStr;
Ed Tanous6c34de42018-08-29 13:37:36 -0700485 }
486 }
Ed Tanous029573d2019-02-01 10:57:49 -0800487 }
488 },
489 connection.first, path,
490 "org.freedesktop.DBus.Properties", "GetAll",
491 "xyz.openbmc_project.Common.UUID");
492 }
493 else if (interfaceName ==
494 "xyz.openbmc_project.Inventory.Item.System")
495 {
496 crow::connections::systemBus->async_method_call(
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500497 [aResp](
498 const boost::system::error_code ec,
499 const std::vector<
500 std::pair<std::string, VariantType>>&
501 propertiesList) {
Ed Tanous029573d2019-02-01 10:57:49 -0800502 if (ec)
503 {
James Feiste4a4b9a2019-06-20 14:08:07 -0700504 // doesn't have to include this
505 // interface
Ed Tanous029573d2019-02-01 10:57:49 -0800506 return;
507 }
Gunnar Mills698654b2019-10-16 13:17:37 -0500508 BMCWEB_LOG_DEBUG
509 << "Got " << propertiesList.size()
510 << " properties for system";
Ed Tanous029573d2019-02-01 10:57:49 -0800511 for (const std::pair<std::string,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500512 VariantType>&
513 property : propertiesList)
Ed Tanous029573d2019-02-01 10:57:49 -0800514 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500515 const std::string& propertyName =
beccabroekfc5afcf2019-03-05 14:35:15 -0600516 property.first;
517 if ((propertyName == "PartNumber") ||
518 (propertyName == "SerialNumber") ||
519 (propertyName == "Manufacturer") ||
520 (propertyName == "Model"))
Ed Tanous029573d2019-02-01 10:57:49 -0800521 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500522 const std::string* value =
beccabroekfc5afcf2019-03-05 14:35:15 -0600523 std::get_if<std::string>(
524 &property.second);
525 if (value != nullptr)
526 {
527 aResp->res
528 .jsonValue[propertyName] =
529 *value;
530 }
Ed Tanous029573d2019-02-01 10:57:49 -0800531 }
532 }
Gunnar Millsc1e236a2020-04-14 21:36:33 -0500533
Andrew Geisslercb7e1e72019-02-19 13:05:38 -0600534 // Grab the bios version
535 fw_util::getActiveFwVersion(
536 aResp, fw_util::biosPurpose,
537 "BiosVersion");
Ed Tanous029573d2019-02-01 10:57:49 -0800538 },
539 connection.first, path,
540 "org.freedesktop.DBus.Properties", "GetAll",
541 "xyz.openbmc_project.Inventory.Decorator."
542 "Asset");
James Feiste4a4b9a2019-06-20 14:08:07 -0700543
544 crow::connections::systemBus->async_method_call(
545 [aResp](
546 const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500547 const std::variant<std::string>& property) {
James Feiste4a4b9a2019-06-20 14:08:07 -0700548 if (ec)
549 {
550 // doesn't have to include this
551 // interface
552 return;
553 }
554
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500555 const std::string* value =
James Feiste4a4b9a2019-06-20 14:08:07 -0700556 std::get_if<std::string>(&property);
557 if (value != nullptr)
558 {
559 aResp->res.jsonValue["AssetTag"] =
560 *value;
561 }
562 },
563 connection.first, path,
564 "org.freedesktop.DBus.Properties", "Get",
565 "xyz.openbmc_project.Inventory.Decorator."
566 "AssetTag",
567 "AssetTag");
Ed Tanous6c34de42018-08-29 13:37:36 -0700568 }
569 }
570 }
571 }
Ed Tanous6c34de42018-08-29 13:37:36 -0700572 },
573 "xyz.openbmc_project.ObjectMapper",
574 "/xyz/openbmc_project/object_mapper",
575 "xyz.openbmc_project.ObjectMapper", "GetSubTree",
Ed Tanous66173382018-08-15 18:20:59 -0700576 "/xyz/openbmc_project/inventory", int32_t(0),
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500577 std::array<const char*, 5>{
Ed Tanous66173382018-08-15 18:20:59 -0700578 "xyz.openbmc_project.Inventory.Decorator.Asset",
579 "xyz.openbmc_project.Inventory.Item.Cpu",
580 "xyz.openbmc_project.Inventory.Item.Dimm",
581 "xyz.openbmc_project.Inventory.Item.System",
582 "xyz.openbmc_project.Common.UUID",
583 });
Ed Tanous6c34de42018-08-29 13:37:36 -0700584}
585
586/**
Ed Tanous6c34de42018-08-29 13:37:36 -0700587 * @brief Retrieves host state properties over dbus
588 *
589 * @param[in] aResp Shared pointer for completing asynchronous calls.
590 *
591 * @return None.
592 */
593void getHostState(std::shared_ptr<AsyncResp> aResp)
594{
595 BMCWEB_LOG_DEBUG << "Get host information.";
596 crow::connections::systemBus->async_method_call(
Jennifer Leec5d03ff2019-03-08 15:42:58 -0800597 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500598 const std::variant<std::string>& hostState) {
Ed Tanous6c34de42018-08-29 13:37:36 -0700599 if (ec)
600 {
601 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
Jason M. Billsf12894f2018-10-09 12:45:45 -0700602 messages::internalError(aResp->res);
Ed Tanous6c34de42018-08-29 13:37:36 -0700603 return;
604 }
Ed Tanous66173382018-08-15 18:20:59 -0700605
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500606 const std::string* s = std::get_if<std::string>(&hostState);
Ed Tanous66173382018-08-15 18:20:59 -0700607 BMCWEB_LOG_DEBUG << "Host state: " << *s;
608 if (s != nullptr)
Ed Tanous6c34de42018-08-29 13:37:36 -0700609 {
Ed Tanous66173382018-08-15 18:20:59 -0700610 // Verify Host State
Andrew Geissler94732662019-01-08 19:32:16 -0800611 if (*s == "xyz.openbmc_project.State.Host.HostState.Running")
Ed Tanous6c34de42018-08-29 13:37:36 -0700612 {
Ed Tanous66173382018-08-15 18:20:59 -0700613 aResp->res.jsonValue["PowerState"] = "On";
614 aResp->res.jsonValue["Status"]["State"] = "Enabled";
615 }
Andrew Geissler83935af2020-02-13 10:24:53 -0600616 else if (*s == "xyz.openbmc_project.State.Host.HostState."
Gunnar Mills8c888602020-05-01 14:25:09 -0500617 "Quiesced")
618 {
619 aResp->res.jsonValue["PowerState"] = "On";
620 aResp->res.jsonValue["Status"]["State"] = "Quiesced";
621 }
622 else if (*s == "xyz.openbmc_project.State.Host.HostState."
Andrew Geissler83935af2020-02-13 10:24:53 -0600623 "DiagnosticMode")
624 {
625 aResp->res.jsonValue["PowerState"] = "On";
626 aResp->res.jsonValue["Status"]["State"] = "InTest";
627 }
Ed Tanous66173382018-08-15 18:20:59 -0700628 else
629 {
630 aResp->res.jsonValue["PowerState"] = "Off";
631 aResp->res.jsonValue["Status"]["State"] = "Disabled";
Ed Tanous6c34de42018-08-29 13:37:36 -0700632 }
633 }
634 },
635 "xyz.openbmc_project.State.Host", "/xyz/openbmc_project/state/host0",
Ed Tanous66173382018-08-15 18:20:59 -0700636 "org.freedesktop.DBus.Properties", "Get",
637 "xyz.openbmc_project.State.Host", "CurrentHostState");
Ed Tanous6c34de42018-08-29 13:37:36 -0700638}
639
640/**
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530641 * @brief Traslates boot source DBUS property value to redfish.
642 *
643 * @param[in] dbusSource The boot source in DBUS speak.
644 *
645 * @return Returns as a string, the boot source in Redfish terms. If translation
646 * cannot be done, returns an empty string.
647 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500648static std::string dbusToRfBootSource(const std::string& dbusSource)
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530649{
650 if (dbusSource == "xyz.openbmc_project.Control.Boot.Source.Sources.Default")
651 {
652 return "None";
653 }
654 else if (dbusSource ==
655 "xyz.openbmc_project.Control.Boot.Source.Sources.Disk")
656 {
657 return "Hdd";
658 }
659 else if (dbusSource ==
Santosh Puranika71dc0b2019-05-23 20:10:49 +0530660 "xyz.openbmc_project.Control.Boot.Source.Sources.ExternalMedia")
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530661 {
662 return "Cd";
663 }
664 else if (dbusSource ==
665 "xyz.openbmc_project.Control.Boot.Source.Sources.Network")
666 {
667 return "Pxe";
668 }
Jennifer Lee9f16b2c2019-04-19 15:33:48 -0700669 else if (dbusSource ==
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700670 "xyz.openbmc_project.Control.Boot.Source.Sources.RemovableMedia")
Jennifer Lee9f16b2c2019-04-19 15:33:48 -0700671 {
672 return "Usb";
673 }
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530674 else
675 {
676 return "";
677 }
678}
679
680/**
681 * @brief Traslates boot mode DBUS property value to redfish.
682 *
683 * @param[in] dbusMode The boot mode in DBUS speak.
684 *
685 * @return Returns as a string, the boot mode in Redfish terms. If translation
686 * cannot be done, returns an empty string.
687 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500688static std::string dbusToRfBootMode(const std::string& dbusMode)
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530689{
690 if (dbusMode == "xyz.openbmc_project.Control.Boot.Mode.Modes.Regular")
691 {
692 return "None";
693 }
694 else if (dbusMode == "xyz.openbmc_project.Control.Boot.Mode.Modes.Safe")
695 {
696 return "Diags";
697 }
698 else if (dbusMode == "xyz.openbmc_project.Control.Boot.Mode.Modes.Setup")
699 {
700 return "BiosSetup";
701 }
702 else
703 {
704 return "";
705 }
706}
707
708/**
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700709 * @brief Traslates boot source from Redfish to the DBus boot paths.
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530710 *
711 * @param[in] rfSource The boot source in Redfish.
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700712 * @param[out] bootSource The DBus source
713 * @param[out] bootMode the DBus boot mode
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530714 *
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700715 * @return Integer error code.
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530716 */
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700717static int assignBootParameters(std::shared_ptr<AsyncResp> aResp,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500718 const std::string& rfSource,
719 std::string& bootSource, std::string& bootMode)
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530720{
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700721 // The caller has initialized the bootSource and bootMode to:
722 // bootMode = "xyz.openbmc_project.Control.Boot.Mode.Modes.Regular";
723 // bootSource = "xyz.openbmc_project.Control.Boot.Source.Sources.Default";
724 // Only modify the bootSource/bootMode variable needed to achieve the
725 // desired boot action.
726
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530727 if (rfSource == "None")
728 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700729 return 0;
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530730 }
731 else if (rfSource == "Pxe")
732 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700733 bootSource = "xyz.openbmc_project.Control.Boot.Source.Sources.Network";
734 }
735 else if (rfSource == "Hdd")
736 {
737 bootSource = "xyz.openbmc_project.Control.Boot.Source.Sources.Disk";
738 }
739 else if (rfSource == "Diags")
740 {
741 bootMode = "xyz.openbmc_project.Control.Boot.Mode.Modes.Safe";
742 }
743 else if (rfSource == "Cd")
744 {
745 bootSource =
746 "xyz.openbmc_project.Control.Boot.Source.Sources.ExternalMedia";
747 }
748 else if (rfSource == "BiosSetup")
749 {
750 bootMode = "xyz.openbmc_project.Control.Boot.Mode.Modes.Setup";
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530751 }
Jennifer Lee9f16b2c2019-04-19 15:33:48 -0700752 else if (rfSource == "Usb")
753 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700754 bootSource =
755 "xyz.openbmc_project.Control.Boot.Source.Sources.RemovableMedia";
Jennifer Lee9f16b2c2019-04-19 15:33:48 -0700756 }
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530757 else
758 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700759 BMCWEB_LOG_DEBUG << "Invalid property value for "
760 "BootSourceOverrideTarget: "
761 << bootSource;
762 messages::propertyValueNotInList(aResp->res, rfSource,
763 "BootSourceTargetOverride");
764 return -1;
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530765 }
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700766 return 0;
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530767}
768
769/**
770 * @brief Retrieves boot mode over DBUS and fills out the response
771 *
772 * @param[in] aResp Shared pointer for generating response message.
773 * @param[in] bootDbusObj The dbus object to query for boot properties.
774 *
775 * @return None.
776 */
777static void getBootMode(std::shared_ptr<AsyncResp> aResp,
778 std::string bootDbusObj)
779{
780 crow::connections::systemBus->async_method_call(
781 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500782 const std::variant<std::string>& bootMode) {
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530783 if (ec)
784 {
785 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
786 messages::internalError(aResp->res);
787 return;
788 }
789
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500790 const std::string* bootModeStr =
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530791 std::get_if<std::string>(&bootMode);
792
793 if (!bootModeStr)
794 {
795 messages::internalError(aResp->res);
796 return;
797 }
798
799 BMCWEB_LOG_DEBUG << "Boot mode: " << *bootModeStr;
800
801 // TODO (Santosh): Do we need to support override mode?
802 aResp->res.jsonValue["Boot"]["BootSourceOverrideMode"] = "Legacy";
803 aResp->res.jsonValue["Boot"]["BootSourceOverrideTarget@Redfish."
804 "AllowableValues"] = {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -0700805 "None", "Pxe", "Hdd", "Cd", "Diags", "BiosSetup", "Usb"};
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530806
807 if (*bootModeStr !=
808 "xyz.openbmc_project.Control.Boot.Mode.Modes.Regular")
809 {
810 auto rfMode = dbusToRfBootMode(*bootModeStr);
811 if (!rfMode.empty())
812 {
813 aResp->res.jsonValue["Boot"]["BootSourceOverrideTarget"] =
814 rfMode;
815 }
816 }
817
818 // If the BootSourceOverrideTarget is still "None" at the end,
819 // reset the BootSourceOverrideEnabled to indicate that
820 // overrides are disabled
821 if (aResp->res.jsonValue["Boot"]["BootSourceOverrideTarget"] ==
822 "None")
823 {
824 aResp->res.jsonValue["Boot"]["BootSourceOverrideEnabled"] =
825 "Disabled";
826 }
827 },
828 "xyz.openbmc_project.Settings", bootDbusObj,
829 "org.freedesktop.DBus.Properties", "Get",
830 "xyz.openbmc_project.Control.Boot.Mode", "BootMode");
831}
832
833/**
834 * @brief Retrieves boot source over DBUS
835 *
836 * @param[in] aResp Shared pointer for generating response message.
837 * @param[in] oneTimeEnable Boolean to indicate boot properties are one-time.
838 *
839 * @return None.
840 */
841static void getBootSource(std::shared_ptr<AsyncResp> aResp, bool oneTimeEnabled)
842{
843 std::string bootDbusObj =
844 oneTimeEnabled ? "/xyz/openbmc_project/control/host0/boot/one_time"
845 : "/xyz/openbmc_project/control/host0/boot";
846
847 BMCWEB_LOG_DEBUG << "Is one time: " << oneTimeEnabled;
848 aResp->res.jsonValue["Boot"]["BootSourceOverrideEnabled"] =
849 (oneTimeEnabled) ? "Once" : "Continuous";
850
851 crow::connections::systemBus->async_method_call(
852 [aResp, bootDbusObj](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500853 const std::variant<std::string>& bootSource) {
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530854 if (ec)
855 {
856 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
857 messages::internalError(aResp->res);
858 return;
859 }
860
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500861 const std::string* bootSourceStr =
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530862 std::get_if<std::string>(&bootSource);
863
864 if (!bootSourceStr)
865 {
866 messages::internalError(aResp->res);
867 return;
868 }
869 BMCWEB_LOG_DEBUG << "Boot source: " << *bootSourceStr;
870
871 auto rfSource = dbusToRfBootSource(*bootSourceStr);
872 if (!rfSource.empty())
873 {
874 aResp->res.jsonValue["Boot"]["BootSourceOverrideTarget"] =
875 rfSource;
876 }
877 },
878 "xyz.openbmc_project.Settings", bootDbusObj,
879 "org.freedesktop.DBus.Properties", "Get",
880 "xyz.openbmc_project.Control.Boot.Source", "BootSource");
881 getBootMode(std::move(aResp), std::move(bootDbusObj));
882}
883
884/**
885 * @brief Retrieves "One time" enabled setting over DBUS and calls function to
886 * get boot source and boot mode.
887 *
888 * @param[in] aResp Shared pointer for generating response message.
889 *
890 * @return None.
891 */
892static void getBootProperties(std::shared_ptr<AsyncResp> aResp)
893{
894 BMCWEB_LOG_DEBUG << "Get boot information.";
895
896 crow::connections::systemBus->async_method_call(
Jennifer Leec5d03ff2019-03-08 15:42:58 -0800897 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500898 const std::variant<bool>& oneTime) {
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530899 if (ec)
900 {
901 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
James Feist2a833c72019-07-19 10:17:13 -0700902 // not an error, don't have to have the interface
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530903 return;
904 }
905
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500906 const bool* oneTimePtr = std::get_if<bool>(&oneTime);
Santosh Puranik491d8ee2019-02-06 19:46:56 +0530907
908 if (!oneTimePtr)
909 {
910 messages::internalError(aResp->res);
911 return;
912 }
913 getBootSource(aResp, *oneTimePtr);
914 },
915 "xyz.openbmc_project.Settings",
916 "/xyz/openbmc_project/control/host0/boot/one_time",
917 "org.freedesktop.DBus.Properties", "Get",
918 "xyz.openbmc_project.Object.Enable", "Enabled");
919}
920
921/**
Gunnar Millsc0557e12020-06-30 11:26:20 -0500922 * @brief Retrieves the Last Reset Time
923 *
924 * "Reset" is an overloaded term in Redfish, "Reset" includes power on
925 * and power off. Even though this is the "system" Redfish object look at the
926 * chassis D-Bus interface for the LastStateChangeTime since this has the
927 * last power operation time.
928 *
929 * @param[in] aResp Shared pointer for generating response message.
930 *
931 * @return None.
932 */
933void getLastResetTime(std::shared_ptr<AsyncResp> aResp)
934{
935 BMCWEB_LOG_DEBUG << "Getting System Last Reset Time";
936
937 crow::connections::systemBus->async_method_call(
938 [aResp](const boost::system::error_code ec,
939 std::variant<uint64_t>& lastResetTime) {
940 if (ec)
941 {
942 BMCWEB_LOG_DEBUG << "D-BUS response error " << ec;
943 return;
944 }
945
946 const uint64_t* lastResetTimePtr =
947 std::get_if<uint64_t>(&lastResetTime);
948
949 if (!lastResetTimePtr)
950 {
951 messages::internalError(aResp->res);
952 return;
953 }
954 // LastStateChangeTime is epoch time, in milliseconds
955 // https://github.com/openbmc/phosphor-dbus-interfaces/blob/33e8e1dd64da53a66e888d33dc82001305cd0bf9/xyz/openbmc_project/State/Chassis.interface.yaml#L19
956 time_t lastResetTimeStamp =
957 static_cast<time_t>(*lastResetTimePtr / 1000);
958
959 // Convert to ISO 8601 standard
960 aResp->res.jsonValue["LastResetTime"] =
961 crow::utility::getDateTime(lastResetTimeStamp);
962 },
963 "xyz.openbmc_project.State.Chassis",
964 "/xyz/openbmc_project/state/chassis0",
965 "org.freedesktop.DBus.Properties", "Get",
966 "xyz.openbmc_project.State.Chassis", "LastStateChangeTime");
967}
968
969/**
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -0500970 * @brief Retrieves Automatic Retry properties. Known on D-Bus as AutoReboot.
971 *
972 * @param[in] aResp Shared pointer for generating response message.
973 *
974 * @return None.
975 */
976void getAutomaticRetry(std::shared_ptr<AsyncResp> aResp)
977{
978 BMCWEB_LOG_DEBUG << "Get Automatic Retry policy";
979
980 crow::connections::systemBus->async_method_call(
981 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500982 std::variant<bool>& autoRebootEnabled) {
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -0500983 if (ec)
984 {
985 BMCWEB_LOG_DEBUG << "D-BUS response error " << ec;
986 return;
987 }
988
Gunnar Mills1214b7e2020-06-04 10:11:30 -0500989 const bool* autoRebootEnabledPtr =
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -0500990 std::get_if<bool>(&autoRebootEnabled);
991
992 if (!autoRebootEnabledPtr)
993 {
994 messages::internalError(aResp->res);
995 return;
996 }
997
998 BMCWEB_LOG_DEBUG << "Auto Reboot: " << *autoRebootEnabledPtr;
999 if (*autoRebootEnabledPtr == true)
1000 {
1001 aResp->res.jsonValue["Boot"]["AutomaticRetryConfig"] =
1002 "RetryAttempts";
1003 // If AutomaticRetry (AutoReboot) is enabled see how many
1004 // attempts are left
1005 crow::connections::systemBus->async_method_call(
1006 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001007 std::variant<uint32_t>& autoRebootAttemptsLeft) {
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -05001008 if (ec)
1009 {
1010 BMCWEB_LOG_DEBUG << "D-BUS response error " << ec;
1011 return;
1012 }
1013
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001014 const uint32_t* autoRebootAttemptsLeftPtr =
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -05001015 std::get_if<uint32_t>(&autoRebootAttemptsLeft);
1016
1017 if (!autoRebootAttemptsLeftPtr)
1018 {
1019 messages::internalError(aResp->res);
1020 return;
1021 }
1022
1023 BMCWEB_LOG_DEBUG << "Auto Reboot Attempts Left: "
1024 << *autoRebootAttemptsLeftPtr;
1025
1026 aResp->res
1027 .jsonValue["Boot"]
1028 ["RemainingAutomaticRetryAttempts"] =
1029 *autoRebootAttemptsLeftPtr;
1030 },
1031 "xyz.openbmc_project.State.Host",
1032 "/xyz/openbmc_project/state/host0",
1033 "org.freedesktop.DBus.Properties", "Get",
1034 "xyz.openbmc_project.Control.Boot.RebootAttempts",
1035 "AttemptsLeft");
1036 }
1037 else
1038 {
1039 aResp->res.jsonValue["Boot"]["AutomaticRetryConfig"] =
1040 "Disabled";
1041 }
1042
1043 // Not on D-Bus. Hardcoded here:
1044 // https://github.com/openbmc/phosphor-state-manager/blob/1dbbef42675e94fb1f78edb87d6b11380260535a/meson_options.txt#L71
1045 aResp->res.jsonValue["Boot"]["AutomaticRetryAttempts"] = 3;
Gunnar Mills69f35302020-05-17 16:06:31 -05001046
1047 // "AutomaticRetryConfig" can be 3 values, Disabled, RetryAlways,
1048 // and RetryAttempts. OpenBMC only supports Disabled and
1049 // RetryAttempts.
1050 aResp->res.jsonValue["Boot"]["AutomaticRetryConfig@Redfish."
1051 "AllowableValues"] = {"Disabled",
1052 "RetryAttempts"};
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -05001053 },
1054 "xyz.openbmc_project.Settings",
1055 "/xyz/openbmc_project/control/host0/auto_reboot",
1056 "org.freedesktop.DBus.Properties", "Get",
1057 "xyz.openbmc_project.Control.Boot.RebootPolicy", "AutoReboot");
1058}
1059
1060/**
George Liuc6a620f2020-04-10 17:18:11 +08001061 * @brief Retrieves power restore policy over DBUS.
1062 *
1063 * @param[in] aResp Shared pointer for generating response message.
1064 *
1065 * @return None.
1066 */
1067void getPowerRestorePolicy(std::shared_ptr<AsyncResp> aResp)
1068{
1069 BMCWEB_LOG_DEBUG << "Get power restore policy";
1070
1071 crow::connections::systemBus->async_method_call(
1072 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001073 std::variant<std::string>& policy) {
George Liuc6a620f2020-04-10 17:18:11 +08001074 if (ec)
1075 {
1076 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1077 return;
1078 }
1079
1080 const boost::container::flat_map<std::string, std::string>
1081 policyMaps = {
1082 {"xyz.openbmc_project.Control.Power.RestorePolicy.Policy."
1083 "AlwaysOn",
1084 "AlwaysOn"},
1085 {"xyz.openbmc_project.Control.Power.RestorePolicy.Policy."
1086 "AlwaysOff",
1087 "AlwaysOff"},
1088 {"xyz.openbmc_project.Control.Power.RestorePolicy.Policy."
1089 "LastState",
1090 "LastState"}};
1091
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001092 const std::string* policyPtr = std::get_if<std::string>(&policy);
George Liuc6a620f2020-04-10 17:18:11 +08001093
1094 if (!policyPtr)
1095 {
1096 messages::internalError(aResp->res);
1097 return;
1098 }
1099
1100 auto policyMapsIt = policyMaps.find(*policyPtr);
1101 if (policyMapsIt == policyMaps.end())
1102 {
1103 messages::internalError(aResp->res);
1104 return;
1105 }
1106
1107 aResp->res.jsonValue["PowerRestorePolicy"] = policyMapsIt->second;
1108 },
1109 "xyz.openbmc_project.Settings",
1110 "/xyz/openbmc_project/control/host0/power_restore_policy",
1111 "org.freedesktop.DBus.Properties", "Get",
1112 "xyz.openbmc_project.Control.Power.RestorePolicy",
1113 "PowerRestorePolicy");
1114}
1115
1116/**
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301117 * @brief Sets boot properties into DBUS object(s).
1118 *
1119 * @param[in] aResp Shared pointer for generating response message.
1120 * @param[in] oneTimeEnabled Is "one-time" setting already enabled.
1121 * @param[in] bootSource The boot source to set.
1122 * @param[in] bootEnable The source override "enable" to set.
1123 *
Johnathan Mantey265c1602019-08-08 11:02:51 -07001124 * @return Integer error code.
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301125 */
1126static void setBootModeOrSource(std::shared_ptr<AsyncResp> aResp,
1127 bool oneTimeEnabled,
1128 std::optional<std::string> bootSource,
1129 std::optional<std::string> bootEnable)
1130{
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001131 std::string bootSourceStr =
1132 "xyz.openbmc_project.Control.Boot.Source.Sources.Default";
1133 std::string bootModeStr =
1134 "xyz.openbmc_project.Control.Boot.Mode.Modes.Regular";
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301135 bool oneTimeSetting = oneTimeEnabled;
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001136 bool useBootSource = true;
1137
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301138 // Validate incoming parameters
1139 if (bootEnable)
1140 {
1141 if (*bootEnable == "Once")
1142 {
1143 oneTimeSetting = true;
1144 }
1145 else if (*bootEnable == "Continuous")
1146 {
1147 oneTimeSetting = false;
1148 }
1149 else if (*bootEnable == "Disabled")
1150 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001151 BMCWEB_LOG_DEBUG << "Boot source override will be disabled";
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301152 oneTimeSetting = false;
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001153 useBootSource = false;
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301154 }
1155 else
1156 {
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301157 BMCWEB_LOG_DEBUG << "Unsupported value for "
1158 "BootSourceOverrideEnabled: "
1159 << *bootEnable;
1160 messages::propertyValueNotInList(aResp->res, *bootEnable,
1161 "BootSourceOverrideEnabled");
1162 return;
1163 }
1164 }
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301165
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001166 if (bootSource && useBootSource)
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301167 {
1168 // Source target specified
1169 BMCWEB_LOG_DEBUG << "Boot source: " << *bootSource;
1170 // Figure out which DBUS interface and property to use
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001171 if (assignBootParameters(aResp, *bootSource, bootSourceStr,
1172 bootModeStr))
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301173 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001174 BMCWEB_LOG_DEBUG
1175 << "Invalid property value for BootSourceOverrideTarget: "
1176 << *bootSource;
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301177 messages::propertyValueNotInList(aResp->res, *bootSource,
1178 "BootSourceTargetOverride");
1179 return;
1180 }
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001181 }
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301182
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001183 // Act on validated parameters
1184 BMCWEB_LOG_DEBUG << "DBUS boot source: " << bootSourceStr;
1185 BMCWEB_LOG_DEBUG << "DBUS boot mode: " << bootModeStr;
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001186 const char* bootObj =
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001187 oneTimeSetting ? "/xyz/openbmc_project/control/host0/boot/one_time"
1188 : "/xyz/openbmc_project/control/host0/boot";
1189
1190 crow::connections::systemBus->async_method_call(
1191 [aResp](const boost::system::error_code ec) {
1192 if (ec)
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301193 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001194 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1195 messages::internalError(aResp->res);
1196 return;
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301197 }
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001198 BMCWEB_LOG_DEBUG << "Boot source update done.";
1199 },
1200 "xyz.openbmc_project.Settings", bootObj,
1201 "org.freedesktop.DBus.Properties", "Set",
1202 "xyz.openbmc_project.Control.Boot.Source", "BootSource",
1203 std::variant<std::string>(bootSourceStr));
1204
1205 crow::connections::systemBus->async_method_call(
1206 [aResp](const boost::system::error_code ec) {
1207 if (ec)
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301208 {
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001209 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1210 messages::internalError(aResp->res);
1211 return;
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301212 }
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001213 BMCWEB_LOG_DEBUG << "Boot mode update done.";
1214 },
1215 "xyz.openbmc_project.Settings", bootObj,
1216 "org.freedesktop.DBus.Properties", "Set",
1217 "xyz.openbmc_project.Control.Boot.Mode", "BootMode",
1218 std::variant<std::string>(bootModeStr));
1219
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301220 crow::connections::systemBus->async_method_call(
1221 [aResp{std::move(aResp)}](const boost::system::error_code ec) {
1222 if (ec)
1223 {
1224 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1225 messages::internalError(aResp->res);
1226 return;
1227 }
1228 BMCWEB_LOG_DEBUG << "Boot enable update done.";
1229 },
1230 "xyz.openbmc_project.Settings",
1231 "/xyz/openbmc_project/control/host0/boot/one_time",
1232 "org.freedesktop.DBus.Properties", "Set",
1233 "xyz.openbmc_project.Object.Enable", "Enabled",
1234 std::variant<bool>(oneTimeSetting));
1235}
1236
1237/**
1238 * @brief Retrieves "One time" enabled setting over DBUS and calls function to
1239 * set boot source/boot mode properties.
1240 *
1241 * @param[in] aResp Shared pointer for generating response message.
1242 * @param[in] bootSource The boot source from incoming RF request.
1243 * @param[in] bootEnable The boot override enable from incoming RF request.
1244 *
Johnathan Mantey265c1602019-08-08 11:02:51 -07001245 * @return Integer error code.
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301246 */
Gunnar Mills69f35302020-05-17 16:06:31 -05001247static void setBootSourceProperties(std::shared_ptr<AsyncResp> aResp,
1248 std::optional<std::string> bootSource,
1249 std::optional<std::string> bootEnable)
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301250{
1251 BMCWEB_LOG_DEBUG << "Set boot information.";
1252
1253 crow::connections::systemBus->async_method_call(
Johnathan Mantey265c1602019-08-08 11:02:51 -07001254 [aResp, bootSource{std::move(bootSource)},
Patrick Williams19bd78d2020-05-13 17:38:24 -05001255 bootEnable{std::move(bootEnable)}](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001256 const std::variant<bool>& oneTime) {
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301257 if (ec)
1258 {
1259 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1260 messages::internalError(aResp->res);
1261 return;
1262 }
1263
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001264 const bool* oneTimePtr = std::get_if<bool>(&oneTime);
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301265
1266 if (!oneTimePtr)
1267 {
1268 messages::internalError(aResp->res);
1269 return;
1270 }
1271
1272 BMCWEB_LOG_DEBUG << "Got one time: " << *oneTimePtr;
1273
1274 setBootModeOrSource(aResp, *oneTimePtr, std::move(bootSource),
1275 std::move(bootEnable));
1276 },
1277 "xyz.openbmc_project.Settings",
1278 "/xyz/openbmc_project/control/host0/boot/one_time",
1279 "org.freedesktop.DBus.Properties", "Get",
1280 "xyz.openbmc_project.Object.Enable", "Enabled");
1281}
1282
George Liuc6a620f2020-04-10 17:18:11 +08001283/**
Gunnar Mills69f35302020-05-17 16:06:31 -05001284 * @brief Sets automaticRetry (Auto Reboot)
1285 *
1286 * @param[in] aResp Shared pointer for generating response message.
1287 * @param[in] automaticRetryConfig "AutomaticRetryConfig" from request.
1288 *
1289 * @return None.
1290 */
1291static void setAutomaticRetry(std::shared_ptr<AsyncResp> aResp,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001292 const std::string&& automaticRetryConfig)
Gunnar Mills69f35302020-05-17 16:06:31 -05001293{
1294 BMCWEB_LOG_DEBUG << "Set Automatic Retry.";
1295
1296 // OpenBMC only supports "Disabled" and "RetryAttempts".
1297 bool autoRebootEnabled;
1298
1299 if (automaticRetryConfig == "Disabled")
1300 {
1301 autoRebootEnabled = false;
1302 }
1303 else if (automaticRetryConfig == "RetryAttempts")
1304 {
1305 autoRebootEnabled = true;
1306 }
1307 else
1308 {
1309 BMCWEB_LOG_DEBUG << "Invalid property value for "
1310 "AutomaticRetryConfig: "
1311 << automaticRetryConfig;
1312 messages::propertyValueNotInList(aResp->res, automaticRetryConfig,
1313 "AutomaticRetryConfig");
1314 return;
1315 }
1316
1317 crow::connections::systemBus->async_method_call(
1318 [aResp](const boost::system::error_code ec) {
1319 if (ec)
1320 {
1321 messages::internalError(aResp->res);
1322 return;
1323 }
1324 },
1325 "xyz.openbmc_project.Settings",
1326 "/xyz/openbmc_project/control/host0/auto_reboot",
1327 "org.freedesktop.DBus.Properties", "Set",
1328 "xyz.openbmc_project.Control.Boot.RebootPolicy", "AutoReboot",
1329 std::variant<bool>(autoRebootEnabled));
1330}
1331
1332/**
George Liuc6a620f2020-04-10 17:18:11 +08001333 * @brief Sets power restore policy properties.
1334 *
1335 * @param[in] aResp Shared pointer for generating response message.
1336 * @param[in] policy power restore policy properties from request.
1337 *
1338 * @return None.
1339 */
1340static void setPowerRestorePolicy(std::shared_ptr<AsyncResp> aResp,
1341 std::optional<std::string> policy)
1342{
1343 BMCWEB_LOG_DEBUG << "Set power restore policy.";
1344
1345 const boost::container::flat_map<std::string, std::string> policyMaps = {
1346 {"AlwaysOn", "xyz.openbmc_project.Control.Power.RestorePolicy.Policy."
1347 "AlwaysOn"},
1348 {"AlwaysOff", "xyz.openbmc_project.Control.Power.RestorePolicy.Policy."
1349 "AlwaysOff"},
1350 {"LastState", "xyz.openbmc_project.Control.Power.RestorePolicy.Policy."
1351 "LastState"}};
1352
1353 std::string powerRestorPolicy;
1354
1355 auto policyMapsIt = policyMaps.find(*policy);
1356 if (policyMapsIt == policyMaps.end())
1357 {
1358 messages::internalError(aResp->res);
1359 return;
1360 }
1361
1362 powerRestorPolicy = policyMapsIt->second;
1363
1364 crow::connections::systemBus->async_method_call(
1365 [aResp](const boost::system::error_code ec) {
1366 if (ec)
1367 {
1368 messages::internalError(aResp->res);
1369 return;
1370 }
1371 },
1372 "xyz.openbmc_project.Settings",
1373 "/xyz/openbmc_project/control/host0/power_restore_policy",
1374 "org.freedesktop.DBus.Properties", "Set",
1375 "xyz.openbmc_project.Control.Power.RestorePolicy", "PowerRestorePolicy",
1376 std::variant<std::string>(powerRestorPolicy));
1377}
1378
AppaRao Pulia6349912019-10-18 17:16:08 +05301379#ifdef BMCWEB_ENABLE_REDFISH_PROVISIONING_FEATURE
1380/**
1381 * @brief Retrieves provisioning status
1382 *
1383 * @param[in] aResp Shared pointer for completing asynchronous calls.
1384 *
1385 * @return None.
1386 */
1387void getProvisioningStatus(std::shared_ptr<AsyncResp> aResp)
1388{
1389 BMCWEB_LOG_DEBUG << "Get OEM information.";
1390 crow::connections::systemBus->async_method_call(
1391 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001392 const std::vector<std::pair<std::string, VariantType>>&
1393 propertiesList) {
AppaRao Pulia6349912019-10-18 17:16:08 +05301394 if (ec)
1395 {
1396 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1397 messages::internalError(aResp->res);
1398 return;
1399 }
1400
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001401 const bool* provState = nullptr;
1402 const bool* lockState = nullptr;
1403 for (const std::pair<std::string, VariantType>& property :
AppaRao Pulia6349912019-10-18 17:16:08 +05301404 propertiesList)
1405 {
1406 if (property.first == "UfmProvisioned")
1407 {
1408 provState = std::get_if<bool>(&property.second);
1409 }
1410 else if (property.first == "UfmLocked")
1411 {
1412 lockState = std::get_if<bool>(&property.second);
1413 }
1414 }
1415
1416 if ((provState == nullptr) || (lockState == nullptr))
1417 {
1418 BMCWEB_LOG_DEBUG << "Unable to get PFR attributes.";
1419 messages::internalError(aResp->res);
1420 return;
1421 }
1422
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001423 nlohmann::json& oemPFR =
AppaRao Pulia6349912019-10-18 17:16:08 +05301424 aResp->res.jsonValue["Oem"]["OpenBmc"]["FirmwareProvisioning"];
1425 if (*provState == true)
1426 {
1427 if (*lockState == true)
1428 {
1429 oemPFR["ProvisioningStatus"] = "ProvisionedAndLocked";
1430 }
1431 else
1432 {
1433 oemPFR["ProvisioningStatus"] = "ProvisionedButNotLocked";
1434 }
1435 }
1436 else
1437 {
1438 oemPFR["ProvisioningStatus"] = "NotProvisioned";
1439 }
1440 },
1441 "xyz.openbmc_project.PFR.Manager", "/xyz/openbmc_project/pfr",
1442 "org.freedesktop.DBus.Properties", "GetAll",
1443 "xyz.openbmc_project.PFR.Attributes");
1444}
1445#endif
1446
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301447/**
Yong Li51709ff2019-09-30 14:13:04 +08001448 * @brief Translates watchdog timeout action DBUS property value to redfish.
1449 *
1450 * @param[in] dbusAction The watchdog timeout action in D-BUS.
1451 *
1452 * @return Returns as a string, the timeout action in Redfish terms. If
1453 * translation cannot be done, returns an empty string.
1454 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001455static std::string dbusToRfWatchdogAction(const std::string& dbusAction)
Yong Li51709ff2019-09-30 14:13:04 +08001456{
1457 if (dbusAction == "xyz.openbmc_project.State.Watchdog.Action.None")
1458 {
1459 return "None";
1460 }
1461 else if (dbusAction ==
1462 "xyz.openbmc_project.State.Watchdog.Action.HardReset")
1463 {
1464 return "ResetSystem";
1465 }
1466 else if (dbusAction == "xyz.openbmc_project.State.Watchdog.Action.PowerOff")
1467 {
1468 return "PowerDown";
1469 }
1470 else if (dbusAction ==
1471 "xyz.openbmc_project.State.Watchdog.Action.PowerCycle")
1472 {
1473 return "PowerCycle";
1474 }
1475
1476 return "";
1477}
1478
1479/**
Yong Lic45f0082019-10-10 14:19:01 +08001480 *@brief Translates timeout action from Redfish to DBUS property value.
1481 *
1482 *@param[in] rfAction The timeout action in Redfish.
1483 *
1484 *@return Returns as a string, the time_out action as expected by DBUS.
1485 *If translation cannot be done, returns an empty string.
1486 */
1487
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001488static std::string rfToDbusWDTTimeOutAct(const std::string& rfAction)
Yong Lic45f0082019-10-10 14:19:01 +08001489{
1490 if (rfAction == "None")
1491 {
1492 return "xyz.openbmc_project.State.Watchdog.Action.None";
1493 }
1494 else if (rfAction == "PowerCycle")
1495 {
1496 return "xyz.openbmc_project.State.Watchdog.Action.PowerCycle";
1497 }
1498 else if (rfAction == "PowerDown")
1499 {
1500 return "xyz.openbmc_project.State.Watchdog.Action.PowerOff";
1501 }
1502 else if (rfAction == "ResetSystem")
1503 {
1504 return "xyz.openbmc_project.State.Watchdog.Action.HardReset";
1505 }
1506
1507 return "";
1508}
1509
1510/**
Yong Li51709ff2019-09-30 14:13:04 +08001511 * @brief Retrieves host watchdog timer properties over DBUS
1512 *
1513 * @param[in] aResp Shared pointer for completing asynchronous calls.
1514 *
1515 * @return None.
1516 */
1517void getHostWatchdogTimer(std::shared_ptr<AsyncResp> aResp)
1518{
1519 BMCWEB_LOG_DEBUG << "Get host watchodg";
1520 crow::connections::systemBus->async_method_call(
1521 [aResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001522 PropertiesType& properties) {
Yong Li51709ff2019-09-30 14:13:04 +08001523 if (ec)
1524 {
1525 // watchdog service is stopped
1526 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1527 return;
1528 }
1529
1530 BMCWEB_LOG_DEBUG << "Got " << properties.size() << " wdt prop.";
1531
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001532 nlohmann::json& hostWatchdogTimer =
Yong Li51709ff2019-09-30 14:13:04 +08001533 aResp->res.jsonValue["HostWatchdogTimer"];
1534
1535 // watchdog service is running/enabled
1536 hostWatchdogTimer["Status"]["State"] = "Enabled";
1537
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001538 for (const auto& property : properties)
Yong Li51709ff2019-09-30 14:13:04 +08001539 {
1540 BMCWEB_LOG_DEBUG << "prop=" << property.first;
1541 if (property.first == "Enabled")
1542 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001543 const bool* state = std::get_if<bool>(&property.second);
Yong Li51709ff2019-09-30 14:13:04 +08001544
1545 if (!state)
1546 {
1547 messages::internalError(aResp->res);
1548 continue;
1549 }
1550
1551 hostWatchdogTimer["FunctionEnabled"] = *state;
1552 }
1553 else if (property.first == "ExpireAction")
1554 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001555 const std::string* s =
Yong Li51709ff2019-09-30 14:13:04 +08001556 std::get_if<std::string>(&property.second);
1557 if (!s)
1558 {
1559 messages::internalError(aResp->res);
1560 continue;
1561 }
1562
1563 std::string action = dbusToRfWatchdogAction(*s);
1564 if (action.empty())
1565 {
1566 messages::internalError(aResp->res);
1567 continue;
1568 }
1569 hostWatchdogTimer["TimeoutAction"] = action;
1570 }
1571 }
1572 },
1573 "xyz.openbmc_project.Watchdog", "/xyz/openbmc_project/watchdog/host0",
1574 "org.freedesktop.DBus.Properties", "GetAll",
1575 "xyz.openbmc_project.State.Watchdog");
1576}
1577
1578/**
Yong Lic45f0082019-10-10 14:19:01 +08001579 * @brief Sets Host WatchDog Timer properties.
1580 *
1581 * @param[in] aResp Shared pointer for generating response message.
1582 * @param[in] wdtEnable The WDTimer Enable value (true/false) from incoming
1583 * RF request.
1584 * @param[in] wdtTimeOutAction The WDT Timeout action, from incoming RF request.
1585 *
1586 * @return None.
1587 */
1588static void setWDTProperties(std::shared_ptr<AsyncResp> aResp,
1589 const std::optional<bool> wdtEnable,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001590 const std::optional<std::string>& wdtTimeOutAction)
Yong Lic45f0082019-10-10 14:19:01 +08001591{
1592 BMCWEB_LOG_DEBUG << "Set host watchdog";
1593
1594 if (wdtTimeOutAction)
1595 {
1596 std::string wdtTimeOutActStr = rfToDbusWDTTimeOutAct(*wdtTimeOutAction);
1597 // check if TimeOut Action is Valid
1598 if (wdtTimeOutActStr.empty())
1599 {
1600 BMCWEB_LOG_DEBUG << "Unsupported value for TimeoutAction: "
1601 << *wdtTimeOutAction;
1602 messages::propertyValueNotInList(aResp->res, *wdtTimeOutAction,
1603 "TimeoutAction");
1604 return;
1605 }
1606
1607 crow::connections::systemBus->async_method_call(
1608 [aResp](const boost::system::error_code ec) {
1609 if (ec)
1610 {
1611 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1612 messages::internalError(aResp->res);
1613 return;
1614 }
1615 },
1616 "xyz.openbmc_project.Watchdog",
1617 "/xyz/openbmc_project/watchdog/host0",
1618 "org.freedesktop.DBus.Properties", "Set",
1619 "xyz.openbmc_project.State.Watchdog", "ExpireAction",
1620 std::variant<std::string>(wdtTimeOutActStr));
1621 }
1622
1623 if (wdtEnable)
1624 {
1625 crow::connections::systemBus->async_method_call(
1626 [aResp](const boost::system::error_code ec) {
1627 if (ec)
1628 {
1629 BMCWEB_LOG_DEBUG << "DBUS response error " << ec;
1630 messages::internalError(aResp->res);
1631 return;
1632 }
1633 },
1634 "xyz.openbmc_project.Watchdog",
1635 "/xyz/openbmc_project/watchdog/host0",
1636 "org.freedesktop.DBus.Properties", "Set",
1637 "xyz.openbmc_project.State.Watchdog", "Enabled",
1638 std::variant<bool>(*wdtEnable));
1639 }
1640}
1641
1642/**
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001643 * SystemsCollection derived class for delivering ComputerSystems Collection
1644 * Schema
1645 */
Ed Tanous1abe55e2018-09-05 08:30:59 -07001646class SystemsCollection : public Node
1647{
1648 public:
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001649 SystemsCollection(CrowApp& app) : Node(app, "/redfish/v1/Systems/")
Ed Tanous1abe55e2018-09-05 08:30:59 -07001650 {
Ed Tanous1abe55e2018-09-05 08:30:59 -07001651 entityPrivileges = {
1652 {boost::beast::http::verb::get, {{"Login"}}},
1653 {boost::beast::http::verb::head, {{"Login"}}},
1654 {boost::beast::http::verb::patch, {{"ConfigureComponents"}}},
1655 {boost::beast::http::verb::put, {{"ConfigureComponents"}}},
1656 {boost::beast::http::verb::delete_, {{"ConfigureComponents"}}},
1657 {boost::beast::http::verb::post, {{"ConfigureComponents"}}}};
1658 }
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001659
Ed Tanous1abe55e2018-09-05 08:30:59 -07001660 private:
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001661 void doGet(crow::Response& res, const crow::Request& req,
1662 const std::vector<std::string>& params) override
Ed Tanous1abe55e2018-09-05 08:30:59 -07001663 {
Sunitha Harish462023a2020-02-19 08:34:59 -06001664 std::shared_ptr<AsyncResp> asyncResp = std::make_shared<AsyncResp>(res);
Ed Tanous0f74e642018-11-12 15:17:05 -08001665 res.jsonValue["@odata.type"] =
1666 "#ComputerSystemCollection.ComputerSystemCollection";
1667 res.jsonValue["@odata.id"] = "/redfish/v1/Systems";
Ed Tanous0f74e642018-11-12 15:17:05 -08001668 res.jsonValue["Name"] = "Computer System Collection";
Sunitha Harish462023a2020-02-19 08:34:59 -06001669
1670 crow::connections::systemBus->async_method_call(
1671 [asyncResp](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001672 const std::variant<std::string>& hostName) {
1673 nlohmann::json& iface_array =
Sunitha Harish462023a2020-02-19 08:34:59 -06001674 asyncResp->res.jsonValue["Members"];
1675 iface_array = nlohmann::json::array();
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001676 auto& count = asyncResp->res.jsonValue["Members@odata.count"];
Sunitha Harish462023a2020-02-19 08:34:59 -06001677 count = 0;
1678 if (ec)
1679 {
1680 iface_array.push_back(
1681 {{"@odata.id", "/redfish/v1/Systems/system"}});
1682 count = iface_array.size();
1683 return;
1684 }
1685 BMCWEB_LOG_DEBUG << "Hypervisor is available";
1686 iface_array.push_back(
1687 {{"@odata.id", "/redfish/v1/Systems/system"}});
1688 iface_array.push_back(
1689 {{"@odata.id", "/redfish/v1/Systems/hypervisor"}});
1690 count = iface_array.size();
1691 },
1692 "xyz.openbmc_project.Settings", "/xyz/openbmc_project/network/vmi",
1693 "org.freedesktop.DBus.Properties", "Get",
1694 "xyz.openbmc_project.Network.SystemConfiguration", "HostName");
Ed Tanous1abe55e2018-09-05 08:30:59 -07001695 }
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001696};
1697
1698/**
Ed Tanouscc340dd2018-08-29 13:43:38 -07001699 * SystemActionsReset class supports handle POST method for Reset action.
1700 * The class retrieves and sends data directly to D-Bus.
1701 */
1702class SystemActionsReset : public Node
1703{
1704 public:
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001705 SystemActionsReset(CrowApp& app) :
Ed Tanous029573d2019-02-01 10:57:49 -08001706 Node(app, "/redfish/v1/Systems/system/Actions/ComputerSystem.Reset/")
Ed Tanouscc340dd2018-08-29 13:43:38 -07001707 {
1708 entityPrivileges = {
1709 {boost::beast::http::verb::post, {{"ConfigureComponents"}}}};
1710 }
1711
1712 private:
1713 /**
1714 * Function handles POST method request.
1715 * Analyzes POST body message before sends Reset request data to D-Bus.
1716 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001717 void doPost(crow::Response& res, const crow::Request& req,
1718 const std::vector<std::string>& params) override
Ed Tanouscc340dd2018-08-29 13:43:38 -07001719 {
Ed Tanous9712f8a2018-09-21 13:38:49 -07001720 auto asyncResp = std::make_shared<AsyncResp>(res);
1721
1722 std::string resetType;
1723 if (!json_util::readJson(req, res, "ResetType", resetType))
Ed Tanouscc340dd2018-08-29 13:43:38 -07001724 {
1725 return;
1726 }
1727
Jason M. Billsd22c8392019-06-03 13:59:03 -07001728 // Get the command and host vs. chassis
Ed Tanous9712f8a2018-09-21 13:38:49 -07001729 std::string command;
Jason M. Billsd22c8392019-06-03 13:59:03 -07001730 bool hostCommand;
Ed Tanous9712f8a2018-09-21 13:38:49 -07001731 if (resetType == "On")
1732 {
1733 command = "xyz.openbmc_project.State.Host.Transition.On";
Jason M. Billsd22c8392019-06-03 13:59:03 -07001734 hostCommand = true;
1735 }
1736 else if (resetType == "ForceOff")
1737 {
1738 command = "xyz.openbmc_project.State.Chassis.Transition.Off";
1739 hostCommand = false;
1740 }
1741 else if (resetType == "ForceOn")
1742 {
1743 command = "xyz.openbmc_project.State.Host.Transition.On";
1744 hostCommand = true;
1745 }
1746 else if (resetType == "ForceRestart")
1747 {
Jason M. Bills86a08512020-02-04 13:15:49 -08001748 command =
1749 "xyz.openbmc_project.State.Host.Transition.ForceWarmReboot";
1750 hostCommand = true;
Ed Tanous9712f8a2018-09-21 13:38:49 -07001751 }
1752 else if (resetType == "GracefulShutdown")
1753 {
1754 command = "xyz.openbmc_project.State.Host.Transition.Off";
Jason M. Billsd22c8392019-06-03 13:59:03 -07001755 hostCommand = true;
Ed Tanous9712f8a2018-09-21 13:38:49 -07001756 }
1757 else if (resetType == "GracefulRestart")
1758 {
Jason M. Bills86a08512020-02-04 13:15:49 -08001759 command =
1760 "xyz.openbmc_project.State.Host.Transition.GracefulWarmReboot";
Jason M. Billsd22c8392019-06-03 13:59:03 -07001761 hostCommand = true;
1762 }
1763 else if (resetType == "PowerCycle")
1764 {
Jason M. Bills86a08512020-02-04 13:15:49 -08001765 command = "xyz.openbmc_project.State.Host.Transition.Reboot";
1766 hostCommand = true;
Ed Tanous9712f8a2018-09-21 13:38:49 -07001767 }
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001768 else if (resetType == "Nmi")
1769 {
1770 doNMI(asyncResp);
1771 return;
1772 }
Ed Tanous9712f8a2018-09-21 13:38:49 -07001773 else
1774 {
Jason M. Billsf12894f2018-10-09 12:45:45 -07001775 messages::actionParameterUnknown(res, "Reset", resetType);
Ed Tanous9712f8a2018-09-21 13:38:49 -07001776 return;
1777 }
1778
Jason M. Billsd22c8392019-06-03 13:59:03 -07001779 if (hostCommand)
1780 {
1781 crow::connections::systemBus->async_method_call(
1782 [asyncResp, resetType](const boost::system::error_code ec) {
1783 if (ec)
1784 {
1785 BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec;
1786 if (ec.value() == boost::asio::error::invalid_argument)
1787 {
1788 messages::actionParameterNotSupported(
1789 asyncResp->res, resetType, "Reset");
1790 }
1791 else
1792 {
1793 messages::internalError(asyncResp->res);
1794 }
1795 return;
1796 }
1797 messages::success(asyncResp->res);
1798 },
1799 "xyz.openbmc_project.State.Host",
1800 "/xyz/openbmc_project/state/host0",
1801 "org.freedesktop.DBus.Properties", "Set",
1802 "xyz.openbmc_project.State.Host", "RequestedHostTransition",
1803 std::variant<std::string>{command});
1804 }
1805 else
1806 {
1807 crow::connections::systemBus->async_method_call(
1808 [asyncResp, resetType](const boost::system::error_code ec) {
1809 if (ec)
1810 {
1811 BMCWEB_LOG_ERROR << "D-Bus responses error: " << ec;
1812 if (ec.value() == boost::asio::error::invalid_argument)
1813 {
1814 messages::actionParameterNotSupported(
1815 asyncResp->res, resetType, "Reset");
1816 }
1817 else
1818 {
1819 messages::internalError(asyncResp->res);
1820 }
1821 return;
1822 }
1823 messages::success(asyncResp->res);
1824 },
1825 "xyz.openbmc_project.State.Chassis",
1826 "/xyz/openbmc_project/state/chassis0",
1827 "org.freedesktop.DBus.Properties", "Set",
1828 "xyz.openbmc_project.State.Chassis", "RequestedPowerTransition",
1829 std::variant<std::string>{command});
1830 }
Ed Tanouscc340dd2018-08-29 13:43:38 -07001831 }
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001832 /**
1833 * Function transceives data with dbus directly.
1834 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001835 void doNMI(const std::shared_ptr<AsyncResp>& asyncResp)
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001836 {
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001837 constexpr char const* serviceName =
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001838 "xyz.openbmc_project.Control.Host.NMI";
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001839 constexpr char const* objectPath =
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001840 "/xyz/openbmc_project/control/host0/nmi";
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001841 constexpr char const* interfaceName =
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001842 "xyz.openbmc_project.Control.Host.NMI";
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001843 constexpr char const* method = "NMI";
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001844
1845 crow::connections::systemBus->async_method_call(
1846 [asyncResp](const boost::system::error_code ec) {
1847 if (ec)
1848 {
1849 BMCWEB_LOG_ERROR << " Bad D-Bus request error: " << ec;
1850 messages::internalError(asyncResp->res);
1851 return;
1852 }
1853 messages::success(asyncResp->res);
1854 },
1855 serviceName, objectPath, interfaceName, method);
1856 }
Ed Tanouscc340dd2018-08-29 13:43:38 -07001857};
1858
1859/**
Ed Tanous66173382018-08-15 18:20:59 -07001860 * Systems derived class for delivering Computer Systems Schema.
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001861 */
Ed Tanous1abe55e2018-09-05 08:30:59 -07001862class Systems : public Node
1863{
1864 public:
1865 /*
1866 * Default Constructor
1867 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001868 Systems(CrowApp& app) : Node(app, "/redfish/v1/Systems/system/")
Ed Tanous1abe55e2018-09-05 08:30:59 -07001869 {
Ed Tanous1abe55e2018-09-05 08:30:59 -07001870 entityPrivileges = {
1871 {boost::beast::http::verb::get, {{"Login"}}},
1872 {boost::beast::http::verb::head, {{"Login"}}},
1873 {boost::beast::http::verb::patch, {{"ConfigureComponents"}}},
1874 {boost::beast::http::verb::put, {{"ConfigureComponents"}}},
1875 {boost::beast::http::verb::delete_, {{"ConfigureComponents"}}},
1876 {boost::beast::http::verb::post, {{"ConfigureComponents"}}}};
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001877 }
1878
Ed Tanous1abe55e2018-09-05 08:30:59 -07001879 private:
Ed Tanous1abe55e2018-09-05 08:30:59 -07001880 /**
1881 * Functions triggers appropriate requests on DBus
1882 */
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001883 void doGet(crow::Response& res, const crow::Request& req,
1884 const std::vector<std::string>& params) override
Ed Tanous1abe55e2018-09-05 08:30:59 -07001885 {
Gunnar Millsc0557e12020-06-30 11:26:20 -05001886 res.jsonValue["@odata.type"] = "#ComputerSystem.v1_12_0.ComputerSystem";
Gunnar Mills450a25c2020-04-14 21:34:07 -05001887 res.jsonValue["Name"] = "system";
Ed Tanous029573d2019-02-01 10:57:49 -08001888 res.jsonValue["Id"] = "system";
Ed Tanous0f74e642018-11-12 15:17:05 -08001889 res.jsonValue["SystemType"] = "Physical";
1890 res.jsonValue["Description"] = "Computer System";
Ed Tanous0f74e642018-11-12 15:17:05 -08001891 res.jsonValue["ProcessorSummary"]["Count"] = 0;
1892 res.jsonValue["ProcessorSummary"]["Status"]["State"] = "Disabled";
Cheng C Yang5fd7ba62019-11-28 15:58:08 +08001893 res.jsonValue["MemorySummary"]["TotalSystemMemoryGiB"] = uint64_t(0);
Ed Tanous0f74e642018-11-12 15:17:05 -08001894 res.jsonValue["MemorySummary"]["Status"]["State"] = "Disabled";
Ed Tanous029573d2019-02-01 10:57:49 -08001895 res.jsonValue["@odata.id"] = "/redfish/v1/Systems/system";
Ed Tanous04a258f2018-10-15 08:00:41 -07001896
Rapkiewicz, Pawel443c2932018-10-22 15:08:49 +02001897 res.jsonValue["Processors"] = {
Ed Tanous029573d2019-02-01 10:57:49 -08001898 {"@odata.id", "/redfish/v1/Systems/system/Processors"}};
Rapkiewicz, Pawel443c2932018-10-22 15:08:49 +02001899 res.jsonValue["Memory"] = {
Ed Tanous029573d2019-02-01 10:57:49 -08001900 {"@odata.id", "/redfish/v1/Systems/system/Memory"}};
Nikhil Potadea25aecc2019-08-23 16:35:26 -07001901 res.jsonValue["Storage"] = {
1902 {"@odata.id", "/redfish/v1/Systems/system/Storage"}};
Ed Tanous029573d2019-02-01 10:57:49 -08001903
Ed Tanouscc340dd2018-08-29 13:43:38 -07001904 res.jsonValue["Actions"]["#ComputerSystem.Reset"] = {
1905 {"target",
Ed Tanous029573d2019-02-01 10:57:49 -08001906 "/redfish/v1/Systems/system/Actions/ComputerSystem.Reset"},
Ed Tanouscc340dd2018-08-29 13:43:38 -07001907 {"ResetType@Redfish.AllowableValues",
Jason M. Billsd22c8392019-06-03 13:59:03 -07001908 {"On", "ForceOff", "ForceOn", "ForceRestart", "GracefulRestart",
Lakshminarayana R. Kammathbfd5b822019-06-17 12:11:01 -05001909 "GracefulShutdown", "PowerCycle", "Nmi"}}};
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001910
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001911 res.jsonValue["LogServices"] = {
Ed Tanous029573d2019-02-01 10:57:49 -08001912 {"@odata.id", "/redfish/v1/Systems/system/LogServices"}};
Jason M. Billsc4bf6372018-11-05 13:48:27 -08001913
Carol Wangd82a3ac2019-11-21 13:56:38 +08001914 res.jsonValue["Bios"] = {
1915 {"@odata.id", "/redfish/v1/Systems/system/Bios"}};
1916
Jennifer Leec5d03ff2019-03-08 15:42:58 -08001917 res.jsonValue["Links"]["ManagedBy"] = {
1918 {{"@odata.id", "/redfish/v1/Managers/bmc"}}};
1919
1920 res.jsonValue["Status"] = {
1921 {"Health", "OK"},
1922 {"State", "Enabled"},
1923 };
Ed Tanousa0803ef2018-08-29 13:29:23 -07001924 auto asyncResp = std::make_shared<AsyncResp>(res);
Ed Tanous1abe55e2018-09-05 08:30:59 -07001925
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001926 constexpr const std::array<const char*, 4> inventoryForSystems = {
James Feistb49ac872019-05-21 15:12:01 -07001927 "xyz.openbmc_project.Inventory.Item.Dimm",
James Feist2ad9c2f2019-10-29 16:26:48 -07001928 "xyz.openbmc_project.Inventory.Item.Cpu",
James Feiste284a7c2019-11-20 16:20:23 -08001929 "xyz.openbmc_project.Inventory.Item.Drive",
1930 "xyz.openbmc_project.Inventory.Item.StorageController"};
James Feistb49ac872019-05-21 15:12:01 -07001931
1932 auto health = std::make_shared<HealthPopulate>(asyncResp);
1933 crow::connections::systemBus->async_method_call(
1934 [health](const boost::system::error_code ec,
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001935 std::vector<std::string>& resp) {
James Feistb49ac872019-05-21 15:12:01 -07001936 if (ec)
1937 {
1938 // no inventory
1939 return;
1940 }
1941
1942 health->inventory = std::move(resp);
1943 },
1944 "xyz.openbmc_project.ObjectMapper",
1945 "/xyz/openbmc_project/object_mapper",
1946 "xyz.openbmc_project.ObjectMapper", "GetSubTreePaths", "/",
1947 int32_t(0), inventoryForSystems);
1948
1949 health->populate();
1950
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001951 getMainChassisId(asyncResp, [](const std::string& chassisId,
Jennifer Leec5d03ff2019-03-08 15:42:58 -08001952 std::shared_ptr<AsyncResp> aRsp) {
1953 aRsp->res.jsonValue["Links"]["Chassis"] = {
1954 {{"@odata.id", "/redfish/v1/Chassis/" + chassisId}}};
1955 });
AppaRao Pulia3002222019-11-12 21:32:59 +05301956
1957 getIndicatorLedState(asyncResp);
James Feist5bc2dc82019-10-22 14:33:16 -07001958 getComputerSystem(asyncResp, health);
Ed Tanous6c34de42018-08-29 13:37:36 -07001959 getHostState(asyncResp);
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301960 getBootProperties(asyncResp);
Jason M. Billsadbe1922019-10-14 15:44:35 -07001961 getPCIeDeviceList(asyncResp, "PCIeDevices");
Yong Li51709ff2019-09-30 14:13:04 +08001962 getHostWatchdogTimer(asyncResp);
George Liuc6a620f2020-04-10 17:18:11 +08001963 getPowerRestorePolicy(asyncResp);
Gunnar Mills6bd5a8d2020-05-16 18:49:33 -05001964 getAutomaticRetry(asyncResp);
Gunnar Millsc0557e12020-06-30 11:26:20 -05001965 getLastResetTime(asyncResp);
AppaRao Pulia6349912019-10-18 17:16:08 +05301966#ifdef BMCWEB_ENABLE_REDFISH_PROVISIONING_FEATURE
1967 getProvisioningStatus(asyncResp);
1968#endif
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02001969 }
1970
Gunnar Mills1214b7e2020-06-04 10:11:30 -05001971 void doPatch(crow::Response& res, const crow::Request& req,
1972 const std::vector<std::string>& params) override
Ed Tanous1abe55e2018-09-05 08:30:59 -07001973 {
Santosh Puranikcde19e52019-02-20 00:10:56 +05301974 std::optional<std::string> indicatorLed;
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301975 std::optional<nlohmann::json> bootProps;
Yong Lic45f0082019-10-10 14:19:01 +08001976 std::optional<nlohmann::json> wdtTimerProps;
George Liuc6a620f2020-04-10 17:18:11 +08001977 std::optional<std::string> powerRestorePolicy;
Santosh Puranik41352c22019-07-03 05:35:49 -05001978 auto asyncResp = std::make_shared<AsyncResp>(res);
1979
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001980 if (!json_util::readJson(req, res, "IndicatorLED", indicatorLed, "Boot",
George Liuc6a620f2020-04-10 17:18:11 +08001981 bootProps, "WatchdogTimer", wdtTimerProps,
1982 "PowerRestorePolicy", powerRestorePolicy))
Ed Tanous66173382018-08-15 18:20:59 -07001983 {
Ed Tanous9712f8a2018-09-21 13:38:49 -07001984 return;
1985 }
Santosh Puranik491d8ee2019-02-06 19:46:56 +05301986
Johnathan Mantey944ffaf2019-08-12 16:16:14 -07001987 res.result(boost::beast::http::status::no_content);
Yong Lic45f0082019-10-10 14:19:01 +08001988
1989 if (wdtTimerProps)
1990 {
1991 std::optional<bool> wdtEnable;
1992 std::optional<std::string> wdtTimeOutAction;
1993
1994 if (!json_util::readJson(*wdtTimerProps, asyncResp->res,
1995 "FunctionEnabled", wdtEnable,
1996 "TimeoutAction", wdtTimeOutAction))
1997 {
1998 return;
1999 }
2000 setWDTProperties(asyncResp, std::move(wdtEnable),
2001 std::move(wdtTimeOutAction));
2002 }
2003
Santosh Puranik491d8ee2019-02-06 19:46:56 +05302004 if (bootProps)
2005 {
2006 std::optional<std::string> bootSource;
2007 std::optional<std::string> bootEnable;
Gunnar Mills69f35302020-05-17 16:06:31 -05002008 std::optional<std::string> automaticRetryConfig;
Santosh Puranik491d8ee2019-02-06 19:46:56 +05302009
Gunnar Mills69f35302020-05-17 16:06:31 -05002010 if (!json_util::readJson(
2011 *bootProps, asyncResp->res, "BootSourceOverrideTarget",
2012 bootSource, "BootSourceOverrideEnabled", bootEnable,
2013 "AutomaticRetryConfig", automaticRetryConfig))
Santosh Puranik491d8ee2019-02-06 19:46:56 +05302014 {
2015 return;
2016 }
Gunnar Mills69f35302020-05-17 16:06:31 -05002017 if (bootSource || bootEnable)
2018 {
2019 setBootSourceProperties(asyncResp, std::move(bootSource),
2020 std::move(bootEnable));
2021 }
2022 if (automaticRetryConfig)
2023 {
2024 setAutomaticRetry(asyncResp, std::move(*automaticRetryConfig));
2025 }
Santosh Puranik491d8ee2019-02-06 19:46:56 +05302026 }
Johnathan Mantey265c1602019-08-08 11:02:51 -07002027
Ed Tanous9712f8a2018-09-21 13:38:49 -07002028 if (indicatorLed)
2029 {
AppaRao Pulia3002222019-11-12 21:32:59 +05302030 setIndicatorLedState(asyncResp, std::move(*indicatorLed));
Ed Tanous1abe55e2018-09-05 08:30:59 -07002031 }
George Liuc6a620f2020-04-10 17:18:11 +08002032
2033 if (powerRestorePolicy)
2034 {
2035 setPowerRestorePolicy(asyncResp, std::move(*powerRestorePolicy));
2036 }
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02002037 }
Lewanczyk, Dawidc5b2abe2018-05-30 16:59:42 +02002038};
Ed Tanous1abe55e2018-09-05 08:30:59 -07002039} // namespace redfish