1 /* 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 18 #include "bmcweb_config.h" 19 20 #include "app.hpp" 21 #include "dbus_utility.hpp" 22 #include "health.hpp" 23 #include "query.hpp" 24 #include "redfish_util.hpp" 25 #include "registries/privilege_registry.hpp" 26 #include "utils/dbus_utils.hpp" 27 #include "utils/json_utils.hpp" 28 #include "utils/sw_utils.hpp" 29 #include "utils/systemd_utils.hpp" 30 #include "utils/time_utils.hpp" 31 32 #include <boost/system/error_code.hpp> 33 #include <boost/url/format.hpp> 34 #include <sdbusplus/asio/property.hpp> 35 #include <sdbusplus/unpack_properties.hpp> 36 37 #include <algorithm> 38 #include <array> 39 #include <cstdint> 40 #include <memory> 41 #include <optional> 42 #include <ranges> 43 #include <sstream> 44 #include <string> 45 #include <string_view> 46 #include <variant> 47 48 namespace redfish 49 { 50 51 /** 52 * Function reboots the BMC. 53 * 54 * @param[in] asyncResp - Shared pointer for completing asynchronous calls 55 */ 56 inline void 57 doBMCGracefulRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 58 { 59 const char* processName = "xyz.openbmc_project.State.BMC"; 60 const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 61 const char* interfaceName = "xyz.openbmc_project.State.BMC"; 62 const std::string& propertyValue = 63 "xyz.openbmc_project.State.BMC.Transition.Reboot"; 64 const char* destProperty = "RequestedBMCTransition"; 65 66 // Create the D-Bus variant for D-Bus call. 67 sdbusplus::asio::setProperty( 68 *crow::connections::systemBus, processName, objectPath, interfaceName, 69 destProperty, propertyValue, 70 [asyncResp](const boost::system::error_code& ec) { 71 // Use "Set" method to set the property value. 72 if (ec) 73 { 74 BMCWEB_LOG_DEBUG("[Set] Bad D-Bus request error: {}", ec); 75 messages::internalError(asyncResp->res); 76 return; 77 } 78 79 messages::success(asyncResp->res); 80 }); 81 } 82 83 inline void 84 doBMCForceRestart(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 85 { 86 const char* processName = "xyz.openbmc_project.State.BMC"; 87 const char* objectPath = "/xyz/openbmc_project/state/bmc0"; 88 const char* interfaceName = "xyz.openbmc_project.State.BMC"; 89 const std::string& propertyValue = 90 "xyz.openbmc_project.State.BMC.Transition.HardReboot"; 91 const char* destProperty = "RequestedBMCTransition"; 92 93 // Create the D-Bus variant for D-Bus call. 94 sdbusplus::asio::setProperty( 95 *crow::connections::systemBus, processName, objectPath, interfaceName, 96 destProperty, propertyValue, 97 [asyncResp](const boost::system::error_code& ec) { 98 // Use "Set" method to set the property value. 99 if (ec) 100 { 101 BMCWEB_LOG_DEBUG("[Set] Bad D-Bus request error: {}", ec); 102 messages::internalError(asyncResp->res); 103 return; 104 } 105 106 messages::success(asyncResp->res); 107 }); 108 } 109 110 /** 111 * ManagerResetAction class supports the POST method for the Reset (reboot) 112 * action. 113 */ 114 inline void requestRoutesManagerResetAction(App& app) 115 { 116 /** 117 * Function handles POST method request. 118 * Analyzes POST body before sending Reset (Reboot) request data to D-Bus. 119 * OpenBMC supports ResetType "GracefulRestart" and "ForceRestart". 120 */ 121 122 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/Actions/Manager.Reset/") 123 .privileges(redfish::privileges::postManager) 124 .methods(boost::beast::http::verb::post)( 125 [&app](const crow::Request& req, 126 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 127 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 128 { 129 return; 130 } 131 BMCWEB_LOG_DEBUG("Post Manager Reset."); 132 133 std::string resetType; 134 135 if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", 136 resetType)) 137 { 138 return; 139 } 140 141 if (resetType == "GracefulRestart") 142 { 143 BMCWEB_LOG_DEBUG("Proceeding with {}", resetType); 144 doBMCGracefulRestart(asyncResp); 145 return; 146 } 147 if (resetType == "ForceRestart") 148 { 149 BMCWEB_LOG_DEBUG("Proceeding with {}", resetType); 150 doBMCForceRestart(asyncResp); 151 return; 152 } 153 BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", resetType); 154 messages::actionParameterNotSupported(asyncResp->res, resetType, 155 "ResetType"); 156 157 return; 158 }); 159 } 160 161 /** 162 * ManagerResetToDefaultsAction class supports POST method for factory reset 163 * action. 164 */ 165 inline void requestRoutesManagerResetToDefaultsAction(App& app) 166 { 167 /** 168 * Function handles ResetToDefaults POST method request. 169 * 170 * Analyzes POST body message and factory resets BMC by calling 171 * BMC code updater factory reset followed by a BMC reboot. 172 * 173 * BMC code updater factory reset wipes the whole BMC read-write 174 * filesystem which includes things like the network settings. 175 * 176 * OpenBMC only supports ResetToDefaultsType "ResetAll". 177 */ 178 179 BMCWEB_ROUTE(app, 180 "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults/") 181 .privileges(redfish::privileges::postManager) 182 .methods(boost::beast::http::verb::post)( 183 [&app](const crow::Request& req, 184 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 185 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 186 { 187 return; 188 } 189 BMCWEB_LOG_DEBUG("Post ResetToDefaults."); 190 191 std::optional<std::string> resetType; 192 std::optional<std::string> resetToDefaultsType; 193 194 if (!json_util::readJsonAction(req, asyncResp->res, "ResetType", 195 resetType, "ResetToDefaultsType", 196 resetToDefaultsType)) 197 { 198 BMCWEB_LOG_DEBUG("Missing property ResetType."); 199 200 messages::actionParameterMissing(asyncResp->res, "ResetToDefaults", 201 "ResetType"); 202 return; 203 } 204 205 if (resetToDefaultsType && !resetType) 206 { 207 BMCWEB_LOG_WARNING( 208 "Using deprecated ResetToDefaultsType, should be ResetType." 209 "Support for the ResetToDefaultsType will be dropped in 2Q24"); 210 resetType = resetToDefaultsType; 211 } 212 213 if (resetType != "ResetAll") 214 { 215 BMCWEB_LOG_DEBUG("Invalid property value for ResetType: {}", 216 *resetType); 217 messages::actionParameterNotSupported(asyncResp->res, *resetType, 218 "ResetType"); 219 return; 220 } 221 222 crow::connections::systemBus->async_method_call( 223 [asyncResp](const boost::system::error_code& ec) { 224 if (ec) 225 { 226 BMCWEB_LOG_DEBUG("Failed to ResetToDefaults: {}", ec); 227 messages::internalError(asyncResp->res); 228 return; 229 } 230 // Factory Reset doesn't actually happen until a reboot 231 // Can't erase what the BMC is running on 232 doBMCGracefulRestart(asyncResp); 233 }, 234 "xyz.openbmc_project.Software.BMC.Updater", 235 "/xyz/openbmc_project/software", 236 "xyz.openbmc_project.Common.FactoryReset", "Reset"); 237 }); 238 } 239 240 /** 241 * ManagerResetActionInfo derived class for delivering Manager 242 * ResetType AllowableValues using ResetInfo schema. 243 */ 244 inline void requestRoutesManagerResetActionInfo(App& app) 245 { 246 /** 247 * Functions triggers appropriate requests on DBus 248 */ 249 250 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/ResetActionInfo/") 251 .privileges(redfish::privileges::getActionInfo) 252 .methods(boost::beast::http::verb::get)( 253 [&app](const crow::Request& req, 254 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 255 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 256 { 257 return; 258 } 259 260 asyncResp->res.jsonValue["@odata.type"] = 261 "#ActionInfo.v1_1_2.ActionInfo"; 262 asyncResp->res.jsonValue["@odata.id"] = 263 "/redfish/v1/Managers/bmc/ResetActionInfo"; 264 asyncResp->res.jsonValue["Name"] = "Reset Action Info"; 265 asyncResp->res.jsonValue["Id"] = "ResetActionInfo"; 266 nlohmann::json::object_t parameter; 267 parameter["Name"] = "ResetType"; 268 parameter["Required"] = true; 269 parameter["DataType"] = "String"; 270 271 nlohmann::json::array_t allowableValues; 272 allowableValues.emplace_back("GracefulRestart"); 273 allowableValues.emplace_back("ForceRestart"); 274 parameter["AllowableValues"] = std::move(allowableValues); 275 276 nlohmann::json::array_t parameters; 277 parameters.emplace_back(std::move(parameter)); 278 279 asyncResp->res.jsonValue["Parameters"] = std::move(parameters); 280 }); 281 } 282 283 static constexpr const char* objectManagerIface = 284 "org.freedesktop.DBus.ObjectManager"; 285 static constexpr const char* pidConfigurationIface = 286 "xyz.openbmc_project.Configuration.Pid"; 287 static constexpr const char* pidZoneConfigurationIface = 288 "xyz.openbmc_project.Configuration.Pid.Zone"; 289 static constexpr const char* stepwiseConfigurationIface = 290 "xyz.openbmc_project.Configuration.Stepwise"; 291 static constexpr const char* thermalModeIface = 292 "xyz.openbmc_project.Control.ThermalMode"; 293 294 inline void 295 asyncPopulatePid(const std::string& connection, const std::string& path, 296 const std::string& currentProfile, 297 const std::vector<std::string>& supportedProfiles, 298 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 299 { 300 sdbusplus::message::object_path objPath(path); 301 dbus::utility::getManagedObjects( 302 connection, objPath, 303 [asyncResp, currentProfile, supportedProfiles]( 304 const boost::system::error_code& ec, 305 const dbus::utility::ManagedObjectType& managedObj) { 306 if (ec) 307 { 308 BMCWEB_LOG_ERROR("{}", ec); 309 messages::internalError(asyncResp->res); 310 return; 311 } 312 nlohmann::json& configRoot = 313 asyncResp->res.jsonValue["Oem"]["OpenBmc"]["Fan"]; 314 nlohmann::json& fans = configRoot["FanControllers"]; 315 fans["@odata.type"] = "#OemManager.FanControllers"; 316 fans["@odata.id"] = 317 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanControllers"; 318 319 nlohmann::json& pids = configRoot["PidControllers"]; 320 pids["@odata.type"] = "#OemManager.PidControllers"; 321 pids["@odata.id"] = 322 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/PidControllers"; 323 324 nlohmann::json& stepwise = configRoot["StepwiseControllers"]; 325 stepwise["@odata.type"] = "#OemManager.StepwiseControllers"; 326 stepwise["@odata.id"] = 327 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/StepwiseControllers"; 328 329 nlohmann::json& zones = configRoot["FanZones"]; 330 zones["@odata.id"] = 331 "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones"; 332 zones["@odata.type"] = "#OemManager.FanZones"; 333 configRoot["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan"; 334 configRoot["@odata.type"] = "#OemManager.Fan"; 335 configRoot["Profile@Redfish.AllowableValues"] = supportedProfiles; 336 337 if (!currentProfile.empty()) 338 { 339 configRoot["Profile"] = currentProfile; 340 } 341 BMCWEB_LOG_DEBUG("profile = {} !", currentProfile); 342 343 for (const auto& pathPair : managedObj) 344 { 345 for (const auto& intfPair : pathPair.second) 346 { 347 if (intfPair.first != pidConfigurationIface && 348 intfPair.first != pidZoneConfigurationIface && 349 intfPair.first != stepwiseConfigurationIface) 350 { 351 continue; 352 } 353 354 std::string name; 355 356 for (const std::pair<std::string, 357 dbus::utility::DbusVariantType>& propPair : 358 intfPair.second) 359 { 360 if (propPair.first == "Name") 361 { 362 const std::string* namePtr = 363 std::get_if<std::string>(&propPair.second); 364 if (namePtr == nullptr) 365 { 366 BMCWEB_LOG_ERROR("Pid Name Field illegal"); 367 messages::internalError(asyncResp->res); 368 return; 369 } 370 name = *namePtr; 371 dbus::utility::escapePathForDbus(name); 372 } 373 else if (propPair.first == "Profiles") 374 { 375 const std::vector<std::string>* profiles = 376 std::get_if<std::vector<std::string>>( 377 &propPair.second); 378 if (profiles == nullptr) 379 { 380 BMCWEB_LOG_ERROR("Pid Profiles Field illegal"); 381 messages::internalError(asyncResp->res); 382 return; 383 } 384 if (std::find(profiles->begin(), profiles->end(), 385 currentProfile) == profiles->end()) 386 { 387 BMCWEB_LOG_INFO( 388 "{} not supported in current profile", name); 389 continue; 390 } 391 } 392 } 393 nlohmann::json* config = nullptr; 394 const std::string* classPtr = nullptr; 395 396 for (const std::pair<std::string, 397 dbus::utility::DbusVariantType>& propPair : 398 intfPair.second) 399 { 400 if (propPair.first == "Class") 401 { 402 classPtr = std::get_if<std::string>(&propPair.second); 403 } 404 } 405 406 boost::urls::url url("/redfish/v1/Managers/bmc"); 407 if (intfPair.first == pidZoneConfigurationIface) 408 { 409 std::string chassis; 410 if (!dbus::utility::getNthStringFromPath(pathPair.first.str, 411 5, chassis)) 412 { 413 chassis = "#IllegalValue"; 414 } 415 nlohmann::json& zone = zones[name]; 416 zone["Chassis"]["@odata.id"] = 417 boost::urls::format("/redfish/v1/Chassis/{}", chassis); 418 url.set_fragment( 419 ("/Oem/OpenBmc/Fan/FanZones"_json_pointer / name) 420 .to_string()); 421 zone["@odata.id"] = std::move(url); 422 zone["@odata.type"] = "#OemManager.FanZone"; 423 config = &zone; 424 } 425 426 else if (intfPair.first == stepwiseConfigurationIface) 427 { 428 if (classPtr == nullptr) 429 { 430 BMCWEB_LOG_ERROR("Pid Class Field illegal"); 431 messages::internalError(asyncResp->res); 432 return; 433 } 434 435 nlohmann::json& controller = stepwise[name]; 436 config = &controller; 437 url.set_fragment( 438 ("/Oem/OpenBmc/Fan/StepwiseControllers"_json_pointer / 439 name) 440 .to_string()); 441 controller["@odata.id"] = std::move(url); 442 controller["@odata.type"] = 443 "#OemManager.StepwiseController"; 444 445 controller["Direction"] = *classPtr; 446 } 447 448 // pid and fans are off the same configuration 449 else if (intfPair.first == pidConfigurationIface) 450 { 451 if (classPtr == nullptr) 452 { 453 BMCWEB_LOG_ERROR("Pid Class Field illegal"); 454 messages::internalError(asyncResp->res); 455 return; 456 } 457 bool isFan = *classPtr == "fan"; 458 nlohmann::json& element = isFan ? fans[name] : pids[name]; 459 config = &element; 460 if (isFan) 461 { 462 url.set_fragment( 463 ("/Oem/OpenBmc/Fan/FanControllers"_json_pointer / 464 name) 465 .to_string()); 466 element["@odata.id"] = std::move(url); 467 element["@odata.type"] = "#OemManager.FanController"; 468 } 469 else 470 { 471 url.set_fragment( 472 ("/Oem/OpenBmc/Fan/PidControllers"_json_pointer / 473 name) 474 .to_string()); 475 element["@odata.id"] = std::move(url); 476 element["@odata.type"] = "#OemManager.PidController"; 477 } 478 } 479 else 480 { 481 BMCWEB_LOG_ERROR("Unexpected configuration"); 482 messages::internalError(asyncResp->res); 483 return; 484 } 485 486 // used for making maps out of 2 vectors 487 const std::vector<double>* keys = nullptr; 488 const std::vector<double>* values = nullptr; 489 490 for (const auto& propertyPair : intfPair.second) 491 { 492 if (propertyPair.first == "Type" || 493 propertyPair.first == "Class" || 494 propertyPair.first == "Name") 495 { 496 continue; 497 } 498 499 // zones 500 if (intfPair.first == pidZoneConfigurationIface) 501 { 502 const double* ptr = 503 std::get_if<double>(&propertyPair.second); 504 if (ptr == nullptr) 505 { 506 BMCWEB_LOG_ERROR("Field Illegal {}", 507 propertyPair.first); 508 messages::internalError(asyncResp->res); 509 return; 510 } 511 (*config)[propertyPair.first] = *ptr; 512 } 513 514 if (intfPair.first == stepwiseConfigurationIface) 515 { 516 if (propertyPair.first == "Reading" || 517 propertyPair.first == "Output") 518 { 519 const std::vector<double>* ptr = 520 std::get_if<std::vector<double>>( 521 &propertyPair.second); 522 523 if (ptr == nullptr) 524 { 525 BMCWEB_LOG_ERROR("Field Illegal {}", 526 propertyPair.first); 527 messages::internalError(asyncResp->res); 528 return; 529 } 530 531 if (propertyPair.first == "Reading") 532 { 533 keys = ptr; 534 } 535 else 536 { 537 values = ptr; 538 } 539 if (keys != nullptr && values != nullptr) 540 { 541 if (keys->size() != values->size()) 542 { 543 BMCWEB_LOG_ERROR( 544 "Reading and Output size don't match "); 545 messages::internalError(asyncResp->res); 546 return; 547 } 548 nlohmann::json& steps = (*config)["Steps"]; 549 steps = nlohmann::json::array(); 550 for (size_t ii = 0; ii < keys->size(); ii++) 551 { 552 nlohmann::json::object_t step; 553 step["Target"] = (*keys)[ii]; 554 step["Output"] = (*values)[ii]; 555 steps.emplace_back(std::move(step)); 556 } 557 } 558 } 559 if (propertyPair.first == "NegativeHysteresis" || 560 propertyPair.first == "PositiveHysteresis") 561 { 562 const double* ptr = 563 std::get_if<double>(&propertyPair.second); 564 if (ptr == nullptr) 565 { 566 BMCWEB_LOG_ERROR("Field Illegal {}", 567 propertyPair.first); 568 messages::internalError(asyncResp->res); 569 return; 570 } 571 (*config)[propertyPair.first] = *ptr; 572 } 573 } 574 575 // pid and fans are off the same configuration 576 if (intfPair.first == pidConfigurationIface || 577 intfPair.first == stepwiseConfigurationIface) 578 { 579 if (propertyPair.first == "Zones") 580 { 581 const std::vector<std::string>* inputs = 582 std::get_if<std::vector<std::string>>( 583 &propertyPair.second); 584 585 if (inputs == nullptr) 586 { 587 BMCWEB_LOG_ERROR("Zones Pid Field Illegal"); 588 messages::internalError(asyncResp->res); 589 return; 590 } 591 auto& data = (*config)[propertyPair.first]; 592 data = nlohmann::json::array(); 593 for (std::string itemCopy : *inputs) 594 { 595 dbus::utility::escapePathForDbus(itemCopy); 596 nlohmann::json::object_t input; 597 boost::urls::url managerUrl = boost::urls::format( 598 "/redfish/v1/Managers/bmc#{}", 599 ("/Oem/OpenBmc/Fan/FanZones"_json_pointer / 600 itemCopy) 601 .to_string()); 602 input["@odata.id"] = std::move(managerUrl); 603 data.emplace_back(std::move(input)); 604 } 605 } 606 // todo(james): may never happen, but this 607 // assumes configuration data referenced in the 608 // PID config is provided by the same daemon, we 609 // could add another loop to cover all cases, 610 // but I'm okay kicking this can down the road a 611 // bit 612 613 else if (propertyPair.first == "Inputs" || 614 propertyPair.first == "Outputs") 615 { 616 auto& data = (*config)[propertyPair.first]; 617 const std::vector<std::string>* inputs = 618 std::get_if<std::vector<std::string>>( 619 &propertyPair.second); 620 621 if (inputs == nullptr) 622 { 623 BMCWEB_LOG_ERROR("Field Illegal {}", 624 propertyPair.first); 625 messages::internalError(asyncResp->res); 626 return; 627 } 628 data = *inputs; 629 } 630 else if (propertyPair.first == "SetPointOffset") 631 { 632 const std::string* ptr = 633 std::get_if<std::string>(&propertyPair.second); 634 635 if (ptr == nullptr) 636 { 637 BMCWEB_LOG_ERROR("Field Illegal {}", 638 propertyPair.first); 639 messages::internalError(asyncResp->res); 640 return; 641 } 642 // translate from dbus to redfish 643 if (*ptr == "WarningHigh") 644 { 645 (*config)["SetPointOffset"] = 646 "UpperThresholdNonCritical"; 647 } 648 else if (*ptr == "WarningLow") 649 { 650 (*config)["SetPointOffset"] = 651 "LowerThresholdNonCritical"; 652 } 653 else if (*ptr == "CriticalHigh") 654 { 655 (*config)["SetPointOffset"] = 656 "UpperThresholdCritical"; 657 } 658 else if (*ptr == "CriticalLow") 659 { 660 (*config)["SetPointOffset"] = 661 "LowerThresholdCritical"; 662 } 663 else 664 { 665 BMCWEB_LOG_ERROR("Value Illegal {}", *ptr); 666 messages::internalError(asyncResp->res); 667 return; 668 } 669 } 670 // doubles 671 else if (propertyPair.first == "FFGainCoefficient" || 672 propertyPair.first == "FFOffCoefficient" || 673 propertyPair.first == "ICoefficient" || 674 propertyPair.first == "ILimitMax" || 675 propertyPair.first == "ILimitMin" || 676 propertyPair.first == "PositiveHysteresis" || 677 propertyPair.first == "NegativeHysteresis" || 678 propertyPair.first == "OutLimitMax" || 679 propertyPair.first == "OutLimitMin" || 680 propertyPair.first == "PCoefficient" || 681 propertyPair.first == "SetPoint" || 682 propertyPair.first == "SlewNeg" || 683 propertyPair.first == "SlewPos") 684 { 685 const double* ptr = 686 std::get_if<double>(&propertyPair.second); 687 if (ptr == nullptr) 688 { 689 BMCWEB_LOG_ERROR("Field Illegal {}", 690 propertyPair.first); 691 messages::internalError(asyncResp->res); 692 return; 693 } 694 (*config)[propertyPair.first] = *ptr; 695 } 696 } 697 } 698 } 699 } 700 }); 701 } 702 703 enum class CreatePIDRet 704 { 705 fail, 706 del, 707 patch 708 }; 709 710 inline bool 711 getZonesFromJsonReq(const std::shared_ptr<bmcweb::AsyncResp>& response, 712 std::vector<nlohmann::json>& config, 713 std::vector<std::string>& zones) 714 { 715 if (config.empty()) 716 { 717 BMCWEB_LOG_ERROR("Empty Zones"); 718 messages::propertyValueFormatError(response->res, config, "Zones"); 719 return false; 720 } 721 for (auto& odata : config) 722 { 723 std::string path; 724 if (!redfish::json_util::readJson(odata, response->res, "@odata.id", 725 path)) 726 { 727 return false; 728 } 729 std::string input; 730 731 // 8 below comes from 732 // /redfish/v1/Managers/bmc#/Oem/OpenBmc/Fan/FanZones/Left 733 // 0 1 2 3 4 5 6 7 8 734 if (!dbus::utility::getNthStringFromPath(path, 8, input)) 735 { 736 BMCWEB_LOG_ERROR("Got invalid path {}", path); 737 BMCWEB_LOG_ERROR("Illegal Type Zones"); 738 messages::propertyValueFormatError(response->res, odata, "Zones"); 739 return false; 740 } 741 std::replace(input.begin(), input.end(), '_', ' '); 742 zones.emplace_back(std::move(input)); 743 } 744 return true; 745 } 746 747 inline const dbus::utility::ManagedObjectType::value_type* 748 findChassis(const dbus::utility::ManagedObjectType& managedObj, 749 const std::string& value, std::string& chassis) 750 { 751 BMCWEB_LOG_DEBUG("Find Chassis: {}", value); 752 753 std::string escaped = value; 754 std::replace(escaped.begin(), escaped.end(), ' ', '_'); 755 escaped = "/" + escaped; 756 auto it = std::ranges::find_if(managedObj, [&escaped](const auto& obj) { 757 if (obj.first.str.ends_with(escaped)) 758 { 759 BMCWEB_LOG_DEBUG("Matched {}", obj.first.str); 760 return true; 761 } 762 return false; 763 }); 764 765 if (it == managedObj.end()) 766 { 767 return nullptr; 768 } 769 // 5 comes from <chassis-name> being the 5th element 770 // /xyz/openbmc_project/inventory/system/chassis/<chassis-name> 771 if (dbus::utility::getNthStringFromPath(it->first.str, 5, chassis)) 772 { 773 return &(*it); 774 } 775 776 return nullptr; 777 } 778 779 inline CreatePIDRet createPidInterface( 780 const std::shared_ptr<bmcweb::AsyncResp>& response, const std::string& type, 781 const nlohmann::json::iterator& it, const std::string& path, 782 const dbus::utility::ManagedObjectType& managedObj, bool createNewObject, 783 dbus::utility::DBusPropertiesMap& output, std::string& chassis, 784 const std::string& profile) 785 { 786 // common deleter 787 if (it.value() == nullptr) 788 { 789 std::string iface; 790 if (type == "PidControllers" || type == "FanControllers") 791 { 792 iface = pidConfigurationIface; 793 } 794 else if (type == "FanZones") 795 { 796 iface = pidZoneConfigurationIface; 797 } 798 else if (type == "StepwiseControllers") 799 { 800 iface = stepwiseConfigurationIface; 801 } 802 else 803 { 804 BMCWEB_LOG_ERROR("Illegal Type {}", type); 805 messages::propertyUnknown(response->res, type); 806 return CreatePIDRet::fail; 807 } 808 809 BMCWEB_LOG_DEBUG("del {} {}", path, iface); 810 // delete interface 811 crow::connections::systemBus->async_method_call( 812 [response, path](const boost::system::error_code& ec) { 813 if (ec) 814 { 815 BMCWEB_LOG_ERROR("Error patching {}: {}", path, ec); 816 messages::internalError(response->res); 817 return; 818 } 819 messages::success(response->res); 820 }, 821 "xyz.openbmc_project.EntityManager", path, iface, "Delete"); 822 return CreatePIDRet::del; 823 } 824 825 const dbus::utility::ManagedObjectType::value_type* managedItem = nullptr; 826 if (!createNewObject) 827 { 828 // if we aren't creating a new object, we should be able to find it on 829 // d-bus 830 managedItem = findChassis(managedObj, it.key(), chassis); 831 if (managedItem == nullptr) 832 { 833 BMCWEB_LOG_ERROR("Failed to get chassis from config patch"); 834 messages::invalidObject( 835 response->res, 836 boost::urls::format("/redfish/v1/Chassis/{}", chassis)); 837 return CreatePIDRet::fail; 838 } 839 } 840 841 if (!profile.empty() && 842 (type == "PidControllers" || type == "FanControllers" || 843 type == "StepwiseControllers")) 844 { 845 if (managedItem == nullptr) 846 { 847 output.emplace_back("Profiles", std::vector<std::string>{profile}); 848 } 849 else 850 { 851 std::string interface; 852 if (type == "StepwiseControllers") 853 { 854 interface = stepwiseConfigurationIface; 855 } 856 else 857 { 858 interface = pidConfigurationIface; 859 } 860 bool ifaceFound = false; 861 for (const auto& iface : managedItem->second) 862 { 863 if (iface.first == interface) 864 { 865 ifaceFound = true; 866 for (const auto& prop : iface.second) 867 { 868 if (prop.first == "Profiles") 869 { 870 const std::vector<std::string>* curProfiles = 871 std::get_if<std::vector<std::string>>( 872 &(prop.second)); 873 if (curProfiles == nullptr) 874 { 875 BMCWEB_LOG_ERROR( 876 "Illegal profiles in managed object"); 877 messages::internalError(response->res); 878 return CreatePIDRet::fail; 879 } 880 if (std::find(curProfiles->begin(), 881 curProfiles->end(), 882 profile) == curProfiles->end()) 883 { 884 std::vector<std::string> newProfiles = 885 *curProfiles; 886 newProfiles.push_back(profile); 887 output.emplace_back("Profiles", newProfiles); 888 } 889 } 890 } 891 } 892 } 893 894 if (!ifaceFound) 895 { 896 BMCWEB_LOG_ERROR("Failed to find interface in managed object"); 897 messages::internalError(response->res); 898 return CreatePIDRet::fail; 899 } 900 } 901 } 902 903 if (type == "PidControllers" || type == "FanControllers") 904 { 905 if (createNewObject) 906 { 907 output.emplace_back("Class", 908 type == "PidControllers" ? "temp" : "fan"); 909 output.emplace_back("Type", "Pid"); 910 } 911 912 std::optional<std::vector<nlohmann::json>> zones; 913 std::optional<std::vector<std::string>> inputs; 914 std::optional<std::vector<std::string>> outputs; 915 std::map<std::string, std::optional<double>> doubles; 916 std::optional<std::string> setpointOffset; 917 if (!redfish::json_util::readJson( 918 it.value(), response->res, "Inputs", inputs, "Outputs", outputs, 919 "Zones", zones, "FFGainCoefficient", 920 doubles["FFGainCoefficient"], "FFOffCoefficient", 921 doubles["FFOffCoefficient"], "ICoefficient", 922 doubles["ICoefficient"], "ILimitMax", doubles["ILimitMax"], 923 "ILimitMin", doubles["ILimitMin"], "OutLimitMax", 924 doubles["OutLimitMax"], "OutLimitMin", doubles["OutLimitMin"], 925 "PCoefficient", doubles["PCoefficient"], "SetPoint", 926 doubles["SetPoint"], "SetPointOffset", setpointOffset, 927 "SlewNeg", doubles["SlewNeg"], "SlewPos", doubles["SlewPos"], 928 "PositiveHysteresis", doubles["PositiveHysteresis"], 929 "NegativeHysteresis", doubles["NegativeHysteresis"])) 930 { 931 return CreatePIDRet::fail; 932 } 933 if (zones) 934 { 935 std::vector<std::string> zonesStr; 936 if (!getZonesFromJsonReq(response, *zones, zonesStr)) 937 { 938 BMCWEB_LOG_ERROR("Illegal Zones"); 939 return CreatePIDRet::fail; 940 } 941 if (chassis.empty() && 942 findChassis(managedObj, zonesStr[0], chassis) == nullptr) 943 { 944 BMCWEB_LOG_ERROR("Failed to get chassis from config patch"); 945 messages::invalidObject( 946 response->res, 947 boost::urls::format("/redfish/v1/Chassis/{}", chassis)); 948 return CreatePIDRet::fail; 949 } 950 output.emplace_back("Zones", std::move(zonesStr)); 951 } 952 953 if (inputs) 954 { 955 for (std::string& value : *inputs) 956 { 957 std::replace(value.begin(), value.end(), '_', ' '); 958 } 959 output.emplace_back("Inputs", *inputs); 960 } 961 962 if (outputs) 963 { 964 for (std::string& value : *outputs) 965 { 966 std::replace(value.begin(), value.end(), '_', ' '); 967 } 968 output.emplace_back("Outputs", *outputs); 969 } 970 971 if (setpointOffset) 972 { 973 // translate between redfish and dbus names 974 if (*setpointOffset == "UpperThresholdNonCritical") 975 { 976 output.emplace_back("SetPointOffset", "WarningLow"); 977 } 978 else if (*setpointOffset == "LowerThresholdNonCritical") 979 { 980 output.emplace_back("SetPointOffset", "WarningHigh"); 981 } 982 else if (*setpointOffset == "LowerThresholdCritical") 983 { 984 output.emplace_back("SetPointOffset", "CriticalLow"); 985 } 986 else if (*setpointOffset == "UpperThresholdCritical") 987 { 988 output.emplace_back("SetPointOffset", "CriticalHigh"); 989 } 990 else 991 { 992 BMCWEB_LOG_ERROR("Invalid setpointoffset {}", *setpointOffset); 993 messages::propertyValueNotInList(response->res, it.key(), 994 "SetPointOffset"); 995 return CreatePIDRet::fail; 996 } 997 } 998 999 // doubles 1000 for (const auto& pairs : doubles) 1001 { 1002 if (!pairs.second) 1003 { 1004 continue; 1005 } 1006 BMCWEB_LOG_DEBUG("{} = {}", pairs.first, *pairs.second); 1007 output.emplace_back(pairs.first, *pairs.second); 1008 } 1009 } 1010 1011 else if (type == "FanZones") 1012 { 1013 output.emplace_back("Type", "Pid.Zone"); 1014 1015 std::optional<nlohmann::json> chassisContainer; 1016 std::optional<double> failSafePercent; 1017 std::optional<double> minThermalOutput; 1018 if (!redfish::json_util::readJson(it.value(), response->res, "Chassis", 1019 chassisContainer, "FailSafePercent", 1020 failSafePercent, "MinThermalOutput", 1021 minThermalOutput)) 1022 { 1023 return CreatePIDRet::fail; 1024 } 1025 1026 if (chassisContainer) 1027 { 1028 std::string chassisId; 1029 if (!redfish::json_util::readJson(*chassisContainer, response->res, 1030 "@odata.id", chassisId)) 1031 { 1032 return CreatePIDRet::fail; 1033 } 1034 1035 // /redfish/v1/chassis/chassis_name/ 1036 if (!dbus::utility::getNthStringFromPath(chassisId, 3, chassis)) 1037 { 1038 BMCWEB_LOG_ERROR("Got invalid path {}", chassisId); 1039 messages::invalidObject( 1040 response->res, 1041 boost::urls::format("/redfish/v1/Chassis/{}", chassisId)); 1042 return CreatePIDRet::fail; 1043 } 1044 } 1045 if (minThermalOutput) 1046 { 1047 output.emplace_back("MinThermalOutput", *minThermalOutput); 1048 } 1049 if (failSafePercent) 1050 { 1051 output.emplace_back("FailSafePercent", *failSafePercent); 1052 } 1053 } 1054 else if (type == "StepwiseControllers") 1055 { 1056 output.emplace_back("Type", "Stepwise"); 1057 1058 std::optional<std::vector<nlohmann::json>> zones; 1059 std::optional<std::vector<nlohmann::json>> steps; 1060 std::optional<std::vector<std::string>> inputs; 1061 std::optional<double> positiveHysteresis; 1062 std::optional<double> negativeHysteresis; 1063 std::optional<std::string> direction; // upper clipping curve vs lower 1064 if (!redfish::json_util::readJson( 1065 it.value(), response->res, "Zones", zones, "Steps", steps, 1066 "Inputs", inputs, "PositiveHysteresis", positiveHysteresis, 1067 "NegativeHysteresis", negativeHysteresis, "Direction", 1068 direction)) 1069 { 1070 return CreatePIDRet::fail; 1071 } 1072 1073 if (zones) 1074 { 1075 std::vector<std::string> zonesStrs; 1076 if (!getZonesFromJsonReq(response, *zones, zonesStrs)) 1077 { 1078 BMCWEB_LOG_ERROR("Illegal Zones"); 1079 return CreatePIDRet::fail; 1080 } 1081 if (chassis.empty() && 1082 findChassis(managedObj, zonesStrs[0], chassis) == nullptr) 1083 { 1084 BMCWEB_LOG_ERROR("Failed to get chassis from config patch"); 1085 messages::invalidObject( 1086 response->res, 1087 boost::urls::format("/redfish/v1/Chassis/{}", chassis)); 1088 return CreatePIDRet::fail; 1089 } 1090 output.emplace_back("Zones", std::move(zonesStrs)); 1091 } 1092 if (steps) 1093 { 1094 std::vector<double> readings; 1095 std::vector<double> outputs; 1096 for (auto& step : *steps) 1097 { 1098 double target = 0.0; 1099 double out = 0.0; 1100 1101 if (!redfish::json_util::readJson(step, response->res, "Target", 1102 target, "Output", out)) 1103 { 1104 return CreatePIDRet::fail; 1105 } 1106 readings.emplace_back(target); 1107 outputs.emplace_back(out); 1108 } 1109 output.emplace_back("Reading", std::move(readings)); 1110 output.emplace_back("Output", std::move(outputs)); 1111 } 1112 if (inputs) 1113 { 1114 for (std::string& value : *inputs) 1115 { 1116 std::replace(value.begin(), value.end(), '_', ' '); 1117 } 1118 output.emplace_back("Inputs", std::move(*inputs)); 1119 } 1120 if (negativeHysteresis) 1121 { 1122 output.emplace_back("NegativeHysteresis", *negativeHysteresis); 1123 } 1124 if (positiveHysteresis) 1125 { 1126 output.emplace_back("PositiveHysteresis", *positiveHysteresis); 1127 } 1128 if (direction) 1129 { 1130 constexpr const std::array<const char*, 2> allowedDirections = { 1131 "Ceiling", "Floor"}; 1132 if (std::ranges::find(allowedDirections, *direction) == 1133 allowedDirections.end()) 1134 { 1135 messages::propertyValueTypeError(response->res, "Direction", 1136 *direction); 1137 return CreatePIDRet::fail; 1138 } 1139 output.emplace_back("Class", *direction); 1140 } 1141 } 1142 else 1143 { 1144 BMCWEB_LOG_ERROR("Illegal Type {}", type); 1145 messages::propertyUnknown(response->res, type); 1146 return CreatePIDRet::fail; 1147 } 1148 return CreatePIDRet::patch; 1149 } 1150 struct GetPIDValues : std::enable_shared_from_this<GetPIDValues> 1151 { 1152 struct CompletionValues 1153 { 1154 std::vector<std::string> supportedProfiles; 1155 std::string currentProfile; 1156 dbus::utility::MapperGetSubTreeResponse subtree; 1157 }; 1158 1159 explicit GetPIDValues( 1160 const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn) : 1161 asyncResp(asyncRespIn) 1162 1163 {} 1164 1165 void run() 1166 { 1167 std::shared_ptr<GetPIDValues> self = shared_from_this(); 1168 1169 // get all configurations 1170 constexpr std::array<std::string_view, 4> interfaces = { 1171 pidConfigurationIface, pidZoneConfigurationIface, 1172 objectManagerIface, stepwiseConfigurationIface}; 1173 dbus::utility::getSubTree( 1174 "/", 0, interfaces, 1175 [self]( 1176 const boost::system::error_code& ec, 1177 const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) { 1178 if (ec) 1179 { 1180 BMCWEB_LOG_ERROR("{}", ec); 1181 messages::internalError(self->asyncResp->res); 1182 return; 1183 } 1184 self->complete.subtree = subtreeLocal; 1185 }); 1186 1187 // at the same time get the selected profile 1188 constexpr std::array<std::string_view, 1> thermalModeIfaces = { 1189 thermalModeIface}; 1190 dbus::utility::getSubTree( 1191 "/", 0, thermalModeIfaces, 1192 [self]( 1193 const boost::system::error_code& ec, 1194 const dbus::utility::MapperGetSubTreeResponse& subtreeLocal) { 1195 if (ec || subtreeLocal.empty()) 1196 { 1197 return; 1198 } 1199 if (subtreeLocal[0].second.size() != 1) 1200 { 1201 // invalid mapper response, should never happen 1202 BMCWEB_LOG_ERROR("GetPIDValues: Mapper Error"); 1203 messages::internalError(self->asyncResp->res); 1204 return; 1205 } 1206 1207 const std::string& path = subtreeLocal[0].first; 1208 const std::string& owner = subtreeLocal[0].second[0].first; 1209 1210 sdbusplus::asio::getAllProperties( 1211 *crow::connections::systemBus, owner, path, thermalModeIface, 1212 [path, owner, 1213 self](const boost::system::error_code& ec2, 1214 const dbus::utility::DBusPropertiesMap& resp) { 1215 if (ec2) 1216 { 1217 BMCWEB_LOG_ERROR( 1218 "GetPIDValues: Can't get thermalModeIface {}", path); 1219 messages::internalError(self->asyncResp->res); 1220 return; 1221 } 1222 1223 const std::string* current = nullptr; 1224 const std::vector<std::string>* supported = nullptr; 1225 1226 const bool success = sdbusplus::unpackPropertiesNoThrow( 1227 dbus_utils::UnpackErrorPrinter(), resp, "Current", current, 1228 "Supported", supported); 1229 1230 if (!success) 1231 { 1232 messages::internalError(self->asyncResp->res); 1233 return; 1234 } 1235 1236 if (current == nullptr || supported == nullptr) 1237 { 1238 BMCWEB_LOG_ERROR( 1239 "GetPIDValues: thermal mode iface invalid {}", path); 1240 messages::internalError(self->asyncResp->res); 1241 return; 1242 } 1243 self->complete.currentProfile = *current; 1244 self->complete.supportedProfiles = *supported; 1245 }); 1246 }); 1247 } 1248 1249 static void 1250 processingComplete(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1251 const CompletionValues& completion) 1252 { 1253 if (asyncResp->res.result() != boost::beast::http::status::ok) 1254 { 1255 return; 1256 } 1257 // create map of <connection, path to objMgr>> 1258 boost::container::flat_map< 1259 std::string, std::string, std::less<>, 1260 std::vector<std::pair<std::string, std::string>>> 1261 objectMgrPaths; 1262 boost::container::flat_set<std::string, std::less<>, 1263 std::vector<std::string>> 1264 calledConnections; 1265 for (const auto& pathGroup : completion.subtree) 1266 { 1267 for (const auto& connectionGroup : pathGroup.second) 1268 { 1269 auto findConnection = 1270 calledConnections.find(connectionGroup.first); 1271 if (findConnection != calledConnections.end()) 1272 { 1273 break; 1274 } 1275 for (const std::string& interface : connectionGroup.second) 1276 { 1277 if (interface == objectManagerIface) 1278 { 1279 objectMgrPaths[connectionGroup.first] = pathGroup.first; 1280 } 1281 // this list is alphabetical, so we 1282 // should have found the objMgr by now 1283 if (interface == pidConfigurationIface || 1284 interface == pidZoneConfigurationIface || 1285 interface == stepwiseConfigurationIface) 1286 { 1287 auto findObjMgr = 1288 objectMgrPaths.find(connectionGroup.first); 1289 if (findObjMgr == objectMgrPaths.end()) 1290 { 1291 BMCWEB_LOG_DEBUG("{}Has no Object Manager", 1292 connectionGroup.first); 1293 continue; 1294 } 1295 1296 calledConnections.insert(connectionGroup.first); 1297 1298 asyncPopulatePid(findObjMgr->first, findObjMgr->second, 1299 completion.currentProfile, 1300 completion.supportedProfiles, 1301 asyncResp); 1302 break; 1303 } 1304 } 1305 } 1306 } 1307 } 1308 1309 ~GetPIDValues() 1310 { 1311 boost::asio::post(crow::connections::systemBus->get_io_context(), 1312 std::bind_front(&processingComplete, asyncResp, 1313 std::move(complete))); 1314 } 1315 1316 GetPIDValues(const GetPIDValues&) = delete; 1317 GetPIDValues(GetPIDValues&&) = delete; 1318 GetPIDValues& operator=(const GetPIDValues&) = delete; 1319 GetPIDValues& operator=(GetPIDValues&&) = delete; 1320 1321 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 1322 CompletionValues complete; 1323 }; 1324 1325 struct SetPIDValues : std::enable_shared_from_this<SetPIDValues> 1326 { 1327 SetPIDValues(const std::shared_ptr<bmcweb::AsyncResp>& asyncRespIn, 1328 nlohmann::json& data) : 1329 asyncResp(asyncRespIn) 1330 { 1331 std::optional<nlohmann::json> pidControllers; 1332 std::optional<nlohmann::json> fanControllers; 1333 std::optional<nlohmann::json> fanZones; 1334 std::optional<nlohmann::json> stepwiseControllers; 1335 1336 if (!redfish::json_util::readJson( 1337 data, asyncResp->res, "PidControllers", pidControllers, 1338 "FanControllers", fanControllers, "FanZones", fanZones, 1339 "StepwiseControllers", stepwiseControllers, "Profile", profile)) 1340 { 1341 return; 1342 } 1343 configuration.emplace_back("PidControllers", std::move(pidControllers)); 1344 configuration.emplace_back("FanControllers", std::move(fanControllers)); 1345 configuration.emplace_back("FanZones", std::move(fanZones)); 1346 configuration.emplace_back("StepwiseControllers", 1347 std::move(stepwiseControllers)); 1348 } 1349 1350 SetPIDValues(const SetPIDValues&) = delete; 1351 SetPIDValues(SetPIDValues&&) = delete; 1352 SetPIDValues& operator=(const SetPIDValues&) = delete; 1353 SetPIDValues& operator=(SetPIDValues&&) = delete; 1354 1355 void run() 1356 { 1357 if (asyncResp->res.result() != boost::beast::http::status::ok) 1358 { 1359 return; 1360 } 1361 1362 std::shared_ptr<SetPIDValues> self = shared_from_this(); 1363 1364 // todo(james): might make sense to do a mapper call here if this 1365 // interface gets more traction 1366 sdbusplus::message::object_path objPath( 1367 "/xyz/openbmc_project/inventory"); 1368 dbus::utility::getManagedObjects( 1369 "xyz.openbmc_project.EntityManager", objPath, 1370 [self](const boost::system::error_code& ec, 1371 const dbus::utility::ManagedObjectType& mObj) { 1372 if (ec) 1373 { 1374 BMCWEB_LOG_ERROR("Error communicating to Entity Manager"); 1375 messages::internalError(self->asyncResp->res); 1376 return; 1377 } 1378 const std::array<const char*, 3> configurations = { 1379 pidConfigurationIface, pidZoneConfigurationIface, 1380 stepwiseConfigurationIface}; 1381 1382 for (const auto& [path, object] : mObj) 1383 { 1384 for (const auto& [interface, _] : object) 1385 { 1386 if (std::ranges::find(configurations, interface) != 1387 configurations.end()) 1388 { 1389 self->objectCount++; 1390 break; 1391 } 1392 } 1393 } 1394 self->managedObj = mObj; 1395 }); 1396 1397 // at the same time get the profile information 1398 constexpr std::array<std::string_view, 1> thermalModeIfaces = { 1399 thermalModeIface}; 1400 dbus::utility::getSubTree( 1401 "/", 0, thermalModeIfaces, 1402 [self](const boost::system::error_code& ec, 1403 const dbus::utility::MapperGetSubTreeResponse& subtree) { 1404 if (ec || subtree.empty()) 1405 { 1406 return; 1407 } 1408 if (subtree[0].second.empty()) 1409 { 1410 // invalid mapper response, should never happen 1411 BMCWEB_LOG_ERROR("SetPIDValues: Mapper Error"); 1412 messages::internalError(self->asyncResp->res); 1413 return; 1414 } 1415 1416 const std::string& path = subtree[0].first; 1417 const std::string& owner = subtree[0].second[0].first; 1418 sdbusplus::asio::getAllProperties( 1419 *crow::connections::systemBus, owner, path, thermalModeIface, 1420 [self, path, owner](const boost::system::error_code& ec2, 1421 const dbus::utility::DBusPropertiesMap& r) { 1422 if (ec2) 1423 { 1424 BMCWEB_LOG_ERROR( 1425 "SetPIDValues: Can't get thermalModeIface {}", path); 1426 messages::internalError(self->asyncResp->res); 1427 return; 1428 } 1429 const std::string* current = nullptr; 1430 const std::vector<std::string>* supported = nullptr; 1431 1432 const bool success = sdbusplus::unpackPropertiesNoThrow( 1433 dbus_utils::UnpackErrorPrinter(), r, "Current", current, 1434 "Supported", supported); 1435 1436 if (!success) 1437 { 1438 messages::internalError(self->asyncResp->res); 1439 return; 1440 } 1441 1442 if (current == nullptr || supported == nullptr) 1443 { 1444 BMCWEB_LOG_ERROR( 1445 "SetPIDValues: thermal mode iface invalid {}", path); 1446 messages::internalError(self->asyncResp->res); 1447 return; 1448 } 1449 self->currentProfile = *current; 1450 self->supportedProfiles = *supported; 1451 self->profileConnection = owner; 1452 self->profilePath = path; 1453 }); 1454 }); 1455 } 1456 void pidSetDone() 1457 { 1458 if (asyncResp->res.result() != boost::beast::http::status::ok) 1459 { 1460 return; 1461 } 1462 std::shared_ptr<bmcweb::AsyncResp> response = asyncResp; 1463 if (profile) 1464 { 1465 if (std::ranges::find(supportedProfiles, *profile) == 1466 supportedProfiles.end()) 1467 { 1468 messages::actionParameterUnknown(response->res, "Profile", 1469 *profile); 1470 return; 1471 } 1472 currentProfile = *profile; 1473 sdbusplus::asio::setProperty( 1474 *crow::connections::systemBus, profileConnection, profilePath, 1475 thermalModeIface, "Current", *profile, 1476 [response](const boost::system::error_code& ec) { 1477 if (ec) 1478 { 1479 BMCWEB_LOG_ERROR("Error patching profile{}", ec); 1480 messages::internalError(response->res); 1481 } 1482 }); 1483 } 1484 1485 for (auto& containerPair : configuration) 1486 { 1487 auto& container = containerPair.second; 1488 if (!container) 1489 { 1490 continue; 1491 } 1492 BMCWEB_LOG_DEBUG("{}", *container); 1493 1494 const std::string& type = containerPair.first; 1495 1496 for (nlohmann::json::iterator it = container->begin(); 1497 it != container->end(); ++it) 1498 { 1499 const auto& name = it.key(); 1500 std::string dbusObjName = name; 1501 std::replace(dbusObjName.begin(), dbusObjName.end(), ' ', '_'); 1502 BMCWEB_LOG_DEBUG("looking for {}", name); 1503 1504 auto pathItr = std::ranges::find_if( 1505 managedObj, [&dbusObjName](const auto& obj) { 1506 return obj.first.parent_path() == dbusObjName; 1507 }); 1508 dbus::utility::DBusPropertiesMap output; 1509 1510 output.reserve(16); // The pid interface length 1511 1512 // determines if we're patching entity-manager or 1513 // creating a new object 1514 bool createNewObject = (pathItr == managedObj.end()); 1515 BMCWEB_LOG_DEBUG("Found = {}", !createNewObject); 1516 1517 std::string iface; 1518 if (!createNewObject) 1519 { 1520 bool findInterface = false; 1521 for (const auto& interface : pathItr->second) 1522 { 1523 if (interface.first == pidConfigurationIface) 1524 { 1525 if (type == "PidControllers" || 1526 type == "FanControllers") 1527 { 1528 iface = pidConfigurationIface; 1529 findInterface = true; 1530 break; 1531 } 1532 } 1533 else if (interface.first == pidZoneConfigurationIface) 1534 { 1535 if (type == "FanZones") 1536 { 1537 iface = pidZoneConfigurationIface; 1538 findInterface = true; 1539 break; 1540 } 1541 } 1542 else if (interface.first == stepwiseConfigurationIface) 1543 { 1544 if (type == "StepwiseControllers") 1545 { 1546 iface = stepwiseConfigurationIface; 1547 findInterface = true; 1548 break; 1549 } 1550 } 1551 } 1552 1553 // create new object if interface not found 1554 if (!findInterface) 1555 { 1556 createNewObject = true; 1557 } 1558 } 1559 1560 if (createNewObject && it.value() == nullptr) 1561 { 1562 // can't delete a non-existent object 1563 messages::propertyValueNotInList(response->res, it.value(), 1564 name); 1565 continue; 1566 } 1567 1568 std::string path; 1569 if (pathItr != managedObj.end()) 1570 { 1571 path = pathItr->first.str; 1572 } 1573 1574 BMCWEB_LOG_DEBUG("Create new = {}", createNewObject); 1575 1576 // arbitrary limit to avoid attacks 1577 constexpr const size_t controllerLimit = 500; 1578 if (createNewObject && objectCount >= controllerLimit) 1579 { 1580 messages::resourceExhaustion(response->res, type); 1581 continue; 1582 } 1583 std::string escaped = name; 1584 std::replace(escaped.begin(), escaped.end(), '_', ' '); 1585 output.emplace_back("Name", escaped); 1586 1587 std::string chassis; 1588 CreatePIDRet ret = createPidInterface( 1589 response, type, it, path, managedObj, createNewObject, 1590 output, chassis, currentProfile); 1591 if (ret == CreatePIDRet::fail) 1592 { 1593 return; 1594 } 1595 if (ret == CreatePIDRet::del) 1596 { 1597 continue; 1598 } 1599 1600 if (!createNewObject) 1601 { 1602 for (const auto& property : output) 1603 { 1604 crow::connections::systemBus->async_method_call( 1605 [response, 1606 propertyName{std::string(property.first)}]( 1607 const boost::system::error_code& ec) { 1608 if (ec) 1609 { 1610 BMCWEB_LOG_ERROR("Error patching {}: {}", 1611 propertyName, ec); 1612 messages::internalError(response->res); 1613 return; 1614 } 1615 messages::success(response->res); 1616 }, 1617 "xyz.openbmc_project.EntityManager", path, 1618 "org.freedesktop.DBus.Properties", "Set", iface, 1619 property.first, property.second); 1620 } 1621 } 1622 else 1623 { 1624 if (chassis.empty()) 1625 { 1626 BMCWEB_LOG_ERROR("Failed to get chassis from config"); 1627 messages::internalError(response->res); 1628 return; 1629 } 1630 1631 bool foundChassis = false; 1632 for (const auto& obj : managedObj) 1633 { 1634 if (obj.first.parent_path() == chassis) 1635 { 1636 chassis = obj.first.str; 1637 foundChassis = true; 1638 break; 1639 } 1640 } 1641 if (!foundChassis) 1642 { 1643 BMCWEB_LOG_ERROR("Failed to find chassis on dbus"); 1644 messages::resourceMissingAtURI( 1645 response->res, 1646 boost::urls::format("/redfish/v1/Chassis/{}", 1647 chassis)); 1648 return; 1649 } 1650 1651 crow::connections::systemBus->async_method_call( 1652 [response](const boost::system::error_code& ec) { 1653 if (ec) 1654 { 1655 BMCWEB_LOG_ERROR("Error Adding Pid Object {}", ec); 1656 messages::internalError(response->res); 1657 return; 1658 } 1659 messages::success(response->res); 1660 }, 1661 "xyz.openbmc_project.EntityManager", chassis, 1662 "xyz.openbmc_project.AddObject", "AddObject", output); 1663 } 1664 } 1665 } 1666 } 1667 1668 ~SetPIDValues() 1669 { 1670 try 1671 { 1672 pidSetDone(); 1673 } 1674 catch (...) 1675 { 1676 BMCWEB_LOG_CRITICAL("pidSetDone threw exception"); 1677 } 1678 } 1679 1680 std::shared_ptr<bmcweb::AsyncResp> asyncResp; 1681 std::vector<std::pair<std::string, std::optional<nlohmann::json>>> 1682 configuration; 1683 std::optional<std::string> profile; 1684 dbus::utility::ManagedObjectType managedObj; 1685 std::vector<std::string> supportedProfiles; 1686 std::string currentProfile; 1687 std::string profileConnection; 1688 std::string profilePath; 1689 size_t objectCount = 0; 1690 }; 1691 1692 /** 1693 * @brief Retrieves BMC manager location data over DBus 1694 * 1695 * @param[in] asyncResp Shared pointer for completing asynchronous calls 1696 * @param[in] connectionName - service name 1697 * @param[in] path - object path 1698 * @return none 1699 */ 1700 inline void getLocation(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1701 const std::string& connectionName, 1702 const std::string& path) 1703 { 1704 BMCWEB_LOG_DEBUG("Get BMC manager Location data."); 1705 1706 sdbusplus::asio::getProperty<std::string>( 1707 *crow::connections::systemBus, connectionName, path, 1708 "xyz.openbmc_project.Inventory.Decorator.LocationCode", "LocationCode", 1709 [asyncResp](const boost::system::error_code& ec, 1710 const std::string& property) { 1711 if (ec) 1712 { 1713 BMCWEB_LOG_DEBUG("DBUS response error for " 1714 "Location"); 1715 messages::internalError(asyncResp->res); 1716 return; 1717 } 1718 1719 asyncResp->res.jsonValue["Location"]["PartLocation"]["ServiceLabel"] = 1720 property; 1721 }); 1722 } 1723 // avoid name collision systems.hpp 1724 inline void 1725 managerGetLastResetTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1726 { 1727 BMCWEB_LOG_DEBUG("Getting Manager Last Reset Time"); 1728 1729 sdbusplus::asio::getProperty<uint64_t>( 1730 *crow::connections::systemBus, "xyz.openbmc_project.State.BMC", 1731 "/xyz/openbmc_project/state/bmc0", "xyz.openbmc_project.State.BMC", 1732 "LastRebootTime", 1733 [asyncResp](const boost::system::error_code& ec, 1734 const uint64_t lastResetTime) { 1735 if (ec) 1736 { 1737 BMCWEB_LOG_DEBUG("D-BUS response error {}", ec); 1738 return; 1739 } 1740 1741 // LastRebootTime is epoch time, in milliseconds 1742 // https://github.com/openbmc/phosphor-dbus-interfaces/blob/7f9a128eb9296e926422ddc312c148b625890bb6/xyz/openbmc_project/State/BMC.interface.yaml#L19 1743 uint64_t lastResetTimeStamp = lastResetTime / 1000; 1744 1745 // Convert to ISO 8601 standard 1746 asyncResp->res.jsonValue["LastResetTime"] = 1747 redfish::time_utils::getDateTimeUint(lastResetTimeStamp); 1748 }); 1749 } 1750 1751 /** 1752 * @brief Set the running firmware image 1753 * 1754 * @param[i,o] asyncResp - Async response object 1755 * @param[i] runningFirmwareTarget - Image to make the running image 1756 * 1757 * @return void 1758 */ 1759 inline void 1760 setActiveFirmwareImage(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1761 const std::string& runningFirmwareTarget) 1762 { 1763 // Get the Id from /redfish/v1/UpdateService/FirmwareInventory/<Id> 1764 std::string::size_type idPos = runningFirmwareTarget.rfind('/'); 1765 if (idPos == std::string::npos) 1766 { 1767 messages::propertyValueNotInList(asyncResp->res, runningFirmwareTarget, 1768 "@odata.id"); 1769 BMCWEB_LOG_DEBUG("Can't parse firmware ID!"); 1770 return; 1771 } 1772 idPos++; 1773 if (idPos >= runningFirmwareTarget.size()) 1774 { 1775 messages::propertyValueNotInList(asyncResp->res, runningFirmwareTarget, 1776 "@odata.id"); 1777 BMCWEB_LOG_DEBUG("Invalid firmware ID."); 1778 return; 1779 } 1780 std::string firmwareId = runningFirmwareTarget.substr(idPos); 1781 1782 // Make sure the image is valid before setting priority 1783 sdbusplus::message::object_path objPath("/xyz/openbmc_project/software"); 1784 dbus::utility::getManagedObjects( 1785 "xyz.openbmc_project.Software.BMC.Updater", objPath, 1786 [asyncResp, firmwareId, runningFirmwareTarget]( 1787 const boost::system::error_code& ec, 1788 const dbus::utility::ManagedObjectType& subtree) { 1789 if (ec) 1790 { 1791 BMCWEB_LOG_DEBUG("D-Bus response error getting objects."); 1792 messages::internalError(asyncResp->res); 1793 return; 1794 } 1795 1796 if (subtree.empty()) 1797 { 1798 BMCWEB_LOG_DEBUG("Can't find image!"); 1799 messages::internalError(asyncResp->res); 1800 return; 1801 } 1802 1803 bool foundImage = false; 1804 for (const auto& object : subtree) 1805 { 1806 const std::string& path = 1807 static_cast<const std::string&>(object.first); 1808 std::size_t idPos2 = path.rfind('/'); 1809 1810 if (idPos2 == std::string::npos) 1811 { 1812 continue; 1813 } 1814 1815 idPos2++; 1816 if (idPos2 >= path.size()) 1817 { 1818 continue; 1819 } 1820 1821 if (path.substr(idPos2) == firmwareId) 1822 { 1823 foundImage = true; 1824 break; 1825 } 1826 } 1827 1828 if (!foundImage) 1829 { 1830 messages::propertyValueNotInList( 1831 asyncResp->res, runningFirmwareTarget, "@odata.id"); 1832 BMCWEB_LOG_DEBUG("Invalid firmware ID."); 1833 return; 1834 } 1835 1836 BMCWEB_LOG_DEBUG("Setting firmware version {} to priority 0.", 1837 firmwareId); 1838 1839 // Only support Immediate 1840 // An addition could be a Redfish Setting like 1841 // ActiveSoftwareImageApplyTime and support OnReset 1842 sdbusplus::asio::setProperty( 1843 *crow::connections::systemBus, 1844 "xyz.openbmc_project.Software.BMC.Updater", 1845 "/xyz/openbmc_project/software/" + firmwareId, 1846 "xyz.openbmc_project.Software.RedundancyPriority", "Priority", 1847 static_cast<uint8_t>(0), 1848 [asyncResp](const boost::system::error_code& ec2) { 1849 if (ec2) 1850 { 1851 BMCWEB_LOG_DEBUG("D-Bus response error setting."); 1852 messages::internalError(asyncResp->res); 1853 return; 1854 } 1855 doBMCGracefulRestart(asyncResp); 1856 }); 1857 }); 1858 } 1859 1860 inline void 1861 afterSetDateTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1862 const boost::system::error_code& ec, 1863 const sdbusplus::message_t& msg) 1864 { 1865 if (ec) 1866 { 1867 BMCWEB_LOG_DEBUG("Failed to set elapsed time. DBUS response error {}", 1868 ec); 1869 const sd_bus_error* dbusError = msg.get_error(); 1870 if (dbusError != nullptr) 1871 { 1872 std::string_view errorName(dbusError->name); 1873 if (errorName == 1874 "org.freedesktop.timedate1.AutomaticTimeSyncEnabled") 1875 { 1876 BMCWEB_LOG_DEBUG("Setting conflict"); 1877 messages::propertyValueConflict( 1878 asyncResp->res, "DateTime", 1879 "Managers/NetworkProtocol/NTPProcotolEnabled"); 1880 return; 1881 } 1882 } 1883 messages::internalError(asyncResp->res); 1884 return; 1885 } 1886 asyncResp->res.result(boost::beast::http::status::no_content); 1887 } 1888 1889 inline void setDateTime(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp, 1890 const std::string& datetime) 1891 { 1892 BMCWEB_LOG_DEBUG("Set date time: {}", datetime); 1893 1894 std::optional<redfish::time_utils::usSinceEpoch> us = 1895 redfish::time_utils::dateStringToEpoch(datetime); 1896 if (!us) 1897 { 1898 messages::propertyValueFormatError(asyncResp->res, datetime, 1899 "DateTime"); 1900 return; 1901 } 1902 // Set the absolute datetime 1903 bool relative = false; 1904 bool interactive = false; 1905 crow::connections::systemBus->async_method_call( 1906 [asyncResp](const boost::system::error_code& ec, 1907 const sdbusplus::message_t& msg) { 1908 afterSetDateTime(asyncResp, ec, msg); 1909 }, 1910 "org.freedesktop.timedate1", "/org/freedesktop/timedate1", 1911 "org.freedesktop.timedate1", "SetTime", us->count(), relative, 1912 interactive); 1913 } 1914 1915 inline void 1916 checkForQuiesced(const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) 1917 { 1918 sdbusplus::asio::getProperty<std::string>( 1919 *crow::connections::systemBus, "org.freedesktop.systemd1", 1920 "/org/freedesktop/systemd1/unit/obmc-bmc-service-quiesce@0.target", 1921 "org.freedesktop.systemd1.Unit", "ActiveState", 1922 [asyncResp](const boost::system::error_code& ec, 1923 const std::string& val) { 1924 if (!ec) 1925 { 1926 if (val == "active") 1927 { 1928 asyncResp->res.jsonValue["Status"]["Health"] = "Critical"; 1929 asyncResp->res.jsonValue["Status"]["State"] = "Quiesced"; 1930 return; 1931 } 1932 } 1933 asyncResp->res.jsonValue["Status"]["Health"] = "OK"; 1934 asyncResp->res.jsonValue["Status"]["State"] = "Enabled"; 1935 }); 1936 } 1937 1938 inline void requestRoutesManager(App& app) 1939 { 1940 std::string uuid = persistent_data::getConfig().systemUuid; 1941 1942 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 1943 .privileges(redfish::privileges::getManager) 1944 .methods(boost::beast::http::verb::get)( 1945 [&app, uuid](const crow::Request& req, 1946 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 1947 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 1948 { 1949 return; 1950 } 1951 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers/bmc"; 1952 asyncResp->res.jsonValue["@odata.type"] = "#Manager.v1_14_0.Manager"; 1953 asyncResp->res.jsonValue["Id"] = "bmc"; 1954 asyncResp->res.jsonValue["Name"] = "OpenBmc Manager"; 1955 asyncResp->res.jsonValue["Description"] = 1956 "Baseboard Management Controller"; 1957 asyncResp->res.jsonValue["PowerState"] = "On"; 1958 1959 asyncResp->res.jsonValue["ManagerType"] = "BMC"; 1960 asyncResp->res.jsonValue["UUID"] = systemd_utils::getUuid(); 1961 asyncResp->res.jsonValue["ServiceEntryPointUUID"] = uuid; 1962 asyncResp->res.jsonValue["Model"] = "OpenBmc"; // TODO(ed), get model 1963 1964 asyncResp->res.jsonValue["LogServices"]["@odata.id"] = 1965 "/redfish/v1/Managers/bmc/LogServices"; 1966 asyncResp->res.jsonValue["NetworkProtocol"]["@odata.id"] = 1967 "/redfish/v1/Managers/bmc/NetworkProtocol"; 1968 asyncResp->res.jsonValue["EthernetInterfaces"]["@odata.id"] = 1969 "/redfish/v1/Managers/bmc/EthernetInterfaces"; 1970 1971 #ifdef BMCWEB_ENABLE_VM_NBDPROXY 1972 asyncResp->res.jsonValue["VirtualMedia"]["@odata.id"] = 1973 "/redfish/v1/Managers/bmc/VirtualMedia"; 1974 #endif // BMCWEB_ENABLE_VM_NBDPROXY 1975 1976 // default oem data 1977 nlohmann::json& oem = asyncResp->res.jsonValue["Oem"]; 1978 nlohmann::json& oemOpenbmc = oem["OpenBmc"]; 1979 oem["@odata.type"] = "#OemManager.Oem"; 1980 oem["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem"; 1981 oemOpenbmc["@odata.type"] = "#OemManager.OpenBmc"; 1982 oemOpenbmc["@odata.id"] = "/redfish/v1/Managers/bmc#/Oem/OpenBmc"; 1983 1984 nlohmann::json::object_t certificates; 1985 certificates["@odata.id"] = 1986 "/redfish/v1/Managers/bmc/Truststore/Certificates"; 1987 oemOpenbmc["Certificates"] = std::move(certificates); 1988 1989 // Manager.Reset (an action) can be many values, OpenBMC only 1990 // supports BMC reboot. 1991 nlohmann::json& managerReset = 1992 asyncResp->res.jsonValue["Actions"]["#Manager.Reset"]; 1993 managerReset["target"] = 1994 "/redfish/v1/Managers/bmc/Actions/Manager.Reset"; 1995 managerReset["@Redfish.ActionInfo"] = 1996 "/redfish/v1/Managers/bmc/ResetActionInfo"; 1997 1998 // ResetToDefaults (Factory Reset) has values like 1999 // PreserveNetworkAndUsers and PreserveNetwork that aren't supported 2000 // on OpenBMC 2001 nlohmann::json& resetToDefaults = 2002 asyncResp->res.jsonValue["Actions"]["#Manager.ResetToDefaults"]; 2003 resetToDefaults["target"] = 2004 "/redfish/v1/Managers/bmc/Actions/Manager.ResetToDefaults"; 2005 resetToDefaults["ResetType@Redfish.AllowableValues"] = 2006 nlohmann::json::array_t({"ResetAll"}); 2007 2008 std::pair<std::string, std::string> redfishDateTimeOffset = 2009 redfish::time_utils::getDateTimeOffsetNow(); 2010 2011 asyncResp->res.jsonValue["DateTime"] = redfishDateTimeOffset.first; 2012 asyncResp->res.jsonValue["DateTimeLocalOffset"] = 2013 redfishDateTimeOffset.second; 2014 2015 // TODO (Gunnar): Remove these one day since moved to ComputerSystem 2016 // Still used by OCP profiles 2017 // https://github.com/opencomputeproject/OCP-Profiles/issues/23 2018 // Fill in SerialConsole info 2019 asyncResp->res.jsonValue["SerialConsole"]["ServiceEnabled"] = true; 2020 asyncResp->res.jsonValue["SerialConsole"]["MaxConcurrentSessions"] = 15; 2021 asyncResp->res.jsonValue["SerialConsole"]["ConnectTypesSupported"] = 2022 nlohmann::json::array_t({"IPMI", "SSH"}); 2023 #ifdef BMCWEB_ENABLE_KVM 2024 // Fill in GraphicalConsole info 2025 asyncResp->res.jsonValue["GraphicalConsole"]["ServiceEnabled"] = true; 2026 asyncResp->res.jsonValue["GraphicalConsole"]["MaxConcurrentSessions"] = 2027 4; 2028 asyncResp->res.jsonValue["GraphicalConsole"]["ConnectTypesSupported"] = 2029 nlohmann::json::array_t({"KVMIP"}); 2030 #endif // BMCWEB_ENABLE_KVM 2031 if constexpr (!bmcwebEnableMultiHost) 2032 { 2033 asyncResp->res.jsonValue["Links"]["ManagerForServers@odata.count"] = 2034 1; 2035 2036 nlohmann::json::array_t managerForServers; 2037 nlohmann::json::object_t manager; 2038 manager["@odata.id"] = "/redfish/v1/Systems/system"; 2039 managerForServers.emplace_back(std::move(manager)); 2040 2041 asyncResp->res.jsonValue["Links"]["ManagerForServers"] = 2042 std::move(managerForServers); 2043 } 2044 if constexpr (bmcwebEnableHealthPopulate) 2045 { 2046 auto health = std::make_shared<HealthPopulate>(asyncResp); 2047 health->isManagersHealth = true; 2048 health->populate(); 2049 } 2050 2051 sw_util::populateSoftwareInformation(asyncResp, sw_util::bmcPurpose, 2052 "FirmwareVersion", true); 2053 2054 managerGetLastResetTime(asyncResp); 2055 2056 // ManagerDiagnosticData is added for all BMCs. 2057 nlohmann::json& managerDiagnosticData = 2058 asyncResp->res.jsonValue["ManagerDiagnosticData"]; 2059 managerDiagnosticData["@odata.id"] = 2060 "/redfish/v1/Managers/bmc/ManagerDiagnosticData"; 2061 2062 #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA 2063 auto pids = std::make_shared<GetPIDValues>(asyncResp); 2064 pids->run(); 2065 #endif 2066 2067 getMainChassisId(asyncResp, 2068 [](const std::string& chassisId, 2069 const std::shared_ptr<bmcweb::AsyncResp>& aRsp) { 2070 aRsp->res.jsonValue["Links"]["ManagerForChassis@odata.count"] = 1; 2071 nlohmann::json::array_t managerForChassis; 2072 nlohmann::json::object_t managerObj; 2073 boost::urls::url chassiUrl = 2074 boost::urls::format("/redfish/v1/Chassis/{}", chassisId); 2075 managerObj["@odata.id"] = chassiUrl; 2076 managerForChassis.emplace_back(std::move(managerObj)); 2077 aRsp->res.jsonValue["Links"]["ManagerForChassis"] = 2078 std::move(managerForChassis); 2079 aRsp->res.jsonValue["Links"]["ManagerInChassis"]["@odata.id"] = 2080 chassiUrl; 2081 }); 2082 2083 sdbusplus::asio::getProperty<double>( 2084 *crow::connections::systemBus, "org.freedesktop.systemd1", 2085 "/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager", 2086 "Progress", 2087 [asyncResp](const boost::system::error_code& ec, double val) { 2088 if (ec) 2089 { 2090 BMCWEB_LOG_ERROR("Error while getting progress"); 2091 messages::internalError(asyncResp->res); 2092 return; 2093 } 2094 if (val < 1.0) 2095 { 2096 asyncResp->res.jsonValue["Status"]["Health"] = "OK"; 2097 asyncResp->res.jsonValue["Status"]["State"] = "Starting"; 2098 return; 2099 } 2100 checkForQuiesced(asyncResp); 2101 }); 2102 2103 constexpr std::array<std::string_view, 1> interfaces = { 2104 "xyz.openbmc_project.Inventory.Item.Bmc"}; 2105 dbus::utility::getSubTree( 2106 "/xyz/openbmc_project/inventory", 0, interfaces, 2107 [asyncResp]( 2108 const boost::system::error_code& ec, 2109 const dbus::utility::MapperGetSubTreeResponse& subtree) { 2110 if (ec) 2111 { 2112 BMCWEB_LOG_DEBUG("D-Bus response error on GetSubTree {}", ec); 2113 return; 2114 } 2115 if (subtree.empty()) 2116 { 2117 BMCWEB_LOG_DEBUG("Can't find bmc D-Bus object!"); 2118 return; 2119 } 2120 // Assume only 1 bmc D-Bus object 2121 // Throw an error if there is more than 1 2122 if (subtree.size() > 1) 2123 { 2124 BMCWEB_LOG_DEBUG("Found more than 1 bmc D-Bus object!"); 2125 messages::internalError(asyncResp->res); 2126 return; 2127 } 2128 2129 if (subtree[0].first.empty() || subtree[0].second.size() != 1) 2130 { 2131 BMCWEB_LOG_DEBUG("Error getting bmc D-Bus object!"); 2132 messages::internalError(asyncResp->res); 2133 return; 2134 } 2135 2136 const std::string& path = subtree[0].first; 2137 const std::string& connectionName = subtree[0].second[0].first; 2138 2139 for (const auto& interfaceName : subtree[0].second[0].second) 2140 { 2141 if (interfaceName == 2142 "xyz.openbmc_project.Inventory.Decorator.Asset") 2143 { 2144 sdbusplus::asio::getAllProperties( 2145 *crow::connections::systemBus, connectionName, path, 2146 "xyz.openbmc_project.Inventory.Decorator.Asset", 2147 [asyncResp](const boost::system::error_code& ec2, 2148 const dbus::utility::DBusPropertiesMap& 2149 propertiesList) { 2150 if (ec2) 2151 { 2152 BMCWEB_LOG_DEBUG("Can't get bmc asset!"); 2153 return; 2154 } 2155 2156 const std::string* partNumber = nullptr; 2157 const std::string* serialNumber = nullptr; 2158 const std::string* manufacturer = nullptr; 2159 const std::string* model = nullptr; 2160 const std::string* sparePartNumber = nullptr; 2161 2162 const bool success = sdbusplus::unpackPropertiesNoThrow( 2163 dbus_utils::UnpackErrorPrinter(), propertiesList, 2164 "PartNumber", partNumber, "SerialNumber", 2165 serialNumber, "Manufacturer", manufacturer, "Model", 2166 model, "SparePartNumber", sparePartNumber); 2167 2168 if (!success) 2169 { 2170 messages::internalError(asyncResp->res); 2171 return; 2172 } 2173 2174 if (partNumber != nullptr) 2175 { 2176 asyncResp->res.jsonValue["PartNumber"] = 2177 *partNumber; 2178 } 2179 2180 if (serialNumber != nullptr) 2181 { 2182 asyncResp->res.jsonValue["SerialNumber"] = 2183 *serialNumber; 2184 } 2185 2186 if (manufacturer != nullptr) 2187 { 2188 asyncResp->res.jsonValue["Manufacturer"] = 2189 *manufacturer; 2190 } 2191 2192 if (model != nullptr) 2193 { 2194 asyncResp->res.jsonValue["Model"] = *model; 2195 } 2196 2197 if (sparePartNumber != nullptr) 2198 { 2199 asyncResp->res.jsonValue["SparePartNumber"] = 2200 *sparePartNumber; 2201 } 2202 }); 2203 } 2204 else if (interfaceName == 2205 "xyz.openbmc_project.Inventory.Decorator.LocationCode") 2206 { 2207 getLocation(asyncResp, connectionName, path); 2208 } 2209 } 2210 }); 2211 }); 2212 2213 BMCWEB_ROUTE(app, "/redfish/v1/Managers/bmc/") 2214 .privileges(redfish::privileges::patchManager) 2215 .methods(boost::beast::http::verb::patch)( 2216 [&app](const crow::Request& req, 2217 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 2218 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 2219 { 2220 return; 2221 } 2222 std::optional<nlohmann::json> oem; 2223 std::optional<nlohmann::json> links; 2224 std::optional<std::string> datetime; 2225 2226 if (!json_util::readJsonPatch(req, asyncResp->res, "Oem", oem, 2227 "DateTime", datetime, "Links", links)) 2228 { 2229 return; 2230 } 2231 2232 if (oem) 2233 { 2234 #ifdef BMCWEB_ENABLE_REDFISH_OEM_MANAGER_FAN_DATA 2235 std::optional<nlohmann::json> openbmc; 2236 if (!redfish::json_util::readJson(*oem, asyncResp->res, "OpenBmc", 2237 openbmc)) 2238 { 2239 return; 2240 } 2241 if (openbmc) 2242 { 2243 std::optional<nlohmann::json> fan; 2244 if (!redfish::json_util::readJson(*openbmc, asyncResp->res, 2245 "Fan", fan)) 2246 { 2247 return; 2248 } 2249 if (fan) 2250 { 2251 auto pid = std::make_shared<SetPIDValues>(asyncResp, *fan); 2252 pid->run(); 2253 } 2254 } 2255 #else 2256 messages::propertyUnknown(asyncResp->res, "Oem"); 2257 return; 2258 #endif 2259 } 2260 if (links) 2261 { 2262 std::optional<nlohmann::json> activeSoftwareImage; 2263 if (!redfish::json_util::readJson(*links, asyncResp->res, 2264 "ActiveSoftwareImage", 2265 activeSoftwareImage)) 2266 { 2267 return; 2268 } 2269 if (activeSoftwareImage) 2270 { 2271 std::optional<std::string> odataId; 2272 if (!json_util::readJson(*activeSoftwareImage, asyncResp->res, 2273 "@odata.id", odataId)) 2274 { 2275 return; 2276 } 2277 2278 if (odataId) 2279 { 2280 setActiveFirmwareImage(asyncResp, *odataId); 2281 } 2282 } 2283 } 2284 if (datetime) 2285 { 2286 setDateTime(asyncResp, *datetime); 2287 } 2288 }); 2289 } 2290 2291 inline void requestRoutesManagerCollection(App& app) 2292 { 2293 BMCWEB_ROUTE(app, "/redfish/v1/Managers/") 2294 .privileges(redfish::privileges::getManagerCollection) 2295 .methods(boost::beast::http::verb::get)( 2296 [&app](const crow::Request& req, 2297 const std::shared_ptr<bmcweb::AsyncResp>& asyncResp) { 2298 if (!redfish::setUpRedfishRoute(app, req, asyncResp)) 2299 { 2300 return; 2301 } 2302 // Collections don't include the static data added by SubRoute 2303 // because it has a duplicate entry for members 2304 asyncResp->res.jsonValue["@odata.id"] = "/redfish/v1/Managers"; 2305 asyncResp->res.jsonValue["@odata.type"] = 2306 "#ManagerCollection.ManagerCollection"; 2307 asyncResp->res.jsonValue["Name"] = "Manager Collection"; 2308 asyncResp->res.jsonValue["Members@odata.count"] = 1; 2309 nlohmann::json::array_t members; 2310 nlohmann::json& bmc = members.emplace_back(); 2311 bmc["@odata.id"] = "/redfish/v1/Managers/bmc"; 2312 asyncResp->res.jsonValue["Members"] = std::move(members); 2313 }); 2314 } 2315 } // namespace redfish 2316