1 /** 2 * Copyright © 2019 IBM Corporation 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #include "data_interface.hpp" 18 19 #include "util.hpp" 20 21 #include <phosphor-logging/lg2.hpp> 22 #include <xyz/openbmc_project/State/BMC/server.hpp> 23 #include <xyz/openbmc_project/State/Boot/Progress/server.hpp> 24 25 #ifdef PEL_ENABLE_PHAL 26 #include <libphal.H> 27 #endif 28 29 // Use a timeout of 10s for D-Bus calls so if there are 30 // timeouts the callers of the PEL creation method won't 31 // also timeout. 32 constexpr auto dbusTimeout = 10000000; 33 34 namespace openpower 35 { 36 namespace pels 37 { 38 39 namespace service_name 40 { 41 constexpr auto objectMapper = "xyz.openbmc_project.ObjectMapper"; 42 constexpr auto vpdManager = "com.ibm.VPD.Manager"; 43 constexpr auto ledGroupManager = "xyz.openbmc_project.LED.GroupManager"; 44 constexpr auto hwIsolation = "org.open_power.HardwareIsolation"; 45 constexpr auto biosConfigMgr = "xyz.openbmc_project.BIOSConfigManager"; 46 constexpr auto bootRawProgress = "xyz.openbmc_project.State.Boot.Raw"; 47 constexpr auto pldm = "xyz.openbmc_project.PLDM"; 48 constexpr auto inventoryManager = "xyz.openbmc_project.Inventory.Manager"; 49 constexpr auto entityManager = "xyz.openbmc_project.EntityManager"; 50 } // namespace service_name 51 52 namespace object_path 53 { 54 constexpr auto objectMapper = "/xyz/openbmc_project/object_mapper"; 55 constexpr auto systemInv = "/xyz/openbmc_project/inventory/system"; 56 constexpr auto motherBoardInv = 57 "/xyz/openbmc_project/inventory/system/chassis/motherboard"; 58 constexpr auto baseInv = "/xyz/openbmc_project/inventory"; 59 constexpr auto bmcState = "/xyz/openbmc_project/state/bmc0"; 60 constexpr auto chassisState = "/xyz/openbmc_project/state/chassis0"; 61 constexpr auto hostState = "/xyz/openbmc_project/state/host0"; 62 constexpr auto enableHostPELs = 63 "/xyz/openbmc_project/logging/send_event_logs_to_host"; 64 constexpr auto vpdManager = "/com/ibm/VPD/Manager"; 65 constexpr auto logSetting = "/xyz/openbmc_project/logging/settings"; 66 constexpr auto hwIsolation = "/xyz/openbmc_project/hardware_isolation"; 67 constexpr auto biosConfigMgr = "/xyz/openbmc_project/bios_config/manager"; 68 constexpr auto bootRawProgress = "/xyz/openbmc_project/state/boot/raw0"; 69 } // namespace object_path 70 71 namespace interface 72 { 73 constexpr auto dbusProperty = "org.freedesktop.DBus.Properties"; 74 constexpr auto objectMapper = "xyz.openbmc_project.ObjectMapper"; 75 constexpr auto invAsset = "xyz.openbmc_project.Inventory.Decorator.Asset"; 76 constexpr auto bootProgress = "xyz.openbmc_project.State.Boot.Progress"; 77 constexpr auto enable = "xyz.openbmc_project.Object.Enable"; 78 constexpr auto bmcState = "xyz.openbmc_project.State.BMC"; 79 constexpr auto chassisState = "xyz.openbmc_project.State.Chassis"; 80 constexpr auto hostState = "xyz.openbmc_project.State.Host"; 81 constexpr auto viniRecordVPD = "com.ibm.ipzvpd.VINI"; 82 constexpr auto vsbpRecordVPD = "com.ibm.ipzvpd.VSBP"; 83 constexpr auto locCode = "xyz.openbmc_project.Inventory.Decorator.LocationCode"; 84 constexpr auto compatible = 85 "xyz.openbmc_project.Inventory.Decorator.Compatible"; 86 constexpr auto vpdManager = "com.ibm.VPD.Manager"; 87 constexpr auto ledGroup = "xyz.openbmc_project.Led.Group"; 88 constexpr auto operationalStatus = 89 "xyz.openbmc_project.State.Decorator.OperationalStatus"; 90 constexpr auto logSetting = "xyz.openbmc_project.Logging.Settings"; 91 constexpr auto associationDef = "xyz.openbmc_project.Association.Definitions"; 92 constexpr auto dumpEntry = "xyz.openbmc_project.Dump.Entry"; 93 constexpr auto dumpProgress = "xyz.openbmc_project.Common.Progress"; 94 constexpr auto hwIsolationCreate = "org.open_power.HardwareIsolation.Create"; 95 constexpr auto hwIsolationEntry = "xyz.openbmc_project.HardwareIsolation.Entry"; 96 constexpr auto association = "xyz.openbmc_project.Association"; 97 constexpr auto biosConfigMgr = "xyz.openbmc_project.BIOSConfig.Manager"; 98 constexpr auto bootRawProgress = "xyz.openbmc_project.State.Boot.Raw"; 99 constexpr auto invItem = "xyz.openbmc_project.Inventory.Item"; 100 constexpr auto invFan = "xyz.openbmc_project.Inventory.Item.Fan"; 101 constexpr auto invPowerSupply = 102 "xyz.openbmc_project.Inventory.Item.PowerSupply"; 103 constexpr auto inventoryManager = "xyz.openbmc_project.Inventory.Manager"; 104 } // namespace interface 105 106 using namespace sdbusplus::server::xyz::openbmc_project::state::boot; 107 using namespace sdbusplus::server::xyz::openbmc_project::state; 108 namespace match_rules = sdbusplus::bus::match::rules; 109 110 const DBusInterfaceList hotplugInterfaces{interface::invFan, 111 interface::invPowerSupply}; 112 113 std::pair<std::string, std::string> 114 DataInterfaceBase::extractConnectorFromLocCode( 115 const std::string& locationCode) 116 { 117 auto base = locationCode; 118 std::string connector{}; 119 120 auto pos = base.find("-T"); 121 if (pos != std::string::npos) 122 { 123 connector = base.substr(pos); 124 base = base.substr(0, pos); 125 } 126 127 return {base, connector}; 128 } 129 130 DataInterface::DataInterface(sdbusplus::bus_t& bus) : _bus(bus) 131 { 132 readBMCFWVersion(); 133 readServerFWVersion(); 134 readBMCFWVersionID(); 135 136 // Watch the BootProgress property 137 _properties.emplace_back(std::make_unique<PropertyWatcher<DataInterface>>( 138 bus, object_path::hostState, interface::bootProgress, "BootProgress", 139 *this, [this](const auto& value) { 140 this->_bootState = std::get<std::string>(value); 141 auto status = Progress::convertProgressStagesFromString( 142 std::get<std::string>(value)); 143 144 if ((status == Progress::ProgressStages::SystemInitComplete) || 145 (status == Progress::ProgressStages::OSRunning)) 146 { 147 setHostUp(true); 148 } 149 else 150 { 151 setHostUp(false); 152 } 153 })); 154 155 // Watch the host PEL enable property 156 _properties.emplace_back(std::make_unique<PropertyWatcher<DataInterface>>( 157 bus, object_path::enableHostPELs, interface::enable, "Enabled", *this, 158 [this](const auto& value) { 159 if (std::get<bool>(value) != this->_sendPELsToHost) 160 { 161 lg2::info("The send PELs to host setting changed to {VAL}", 162 "VAL", std::get<bool>(value)); 163 } 164 this->_sendPELsToHost = std::get<bool>(value); 165 })); 166 167 // Watch the BMCState property 168 _properties.emplace_back(std::make_unique<PropertyWatcher<DataInterface>>( 169 bus, object_path::bmcState, interface::bmcState, "CurrentBMCState", 170 *this, [this](const auto& value) { 171 const auto& state = std::get<std::string>(value); 172 this->_bmcState = state; 173 174 // Wait for BMC ready to start watching for 175 // plugs so things calm down first. 176 if (BMC::convertBMCStateFromString(state) == BMC::BMCState::Ready) 177 { 178 startFruPlugWatch(); 179 } 180 })); 181 182 // Watch the chassis current and requested power state properties 183 _properties.emplace_back(std::make_unique<InterfaceWatcher<DataInterface>>( 184 bus, object_path::chassisState, interface::chassisState, *this, 185 [this](const auto& properties) { 186 auto state = properties.find("CurrentPowerState"); 187 if (state != properties.end()) 188 { 189 this->_chassisState = std::get<std::string>(state->second); 190 } 191 192 auto trans = properties.find("RequestedPowerTransition"); 193 if (trans != properties.end()) 194 { 195 this->_chassisTransition = std::get<std::string>(trans->second); 196 } 197 })); 198 199 // Watch the CurrentHostState property 200 _properties.emplace_back(std::make_unique<PropertyWatcher<DataInterface>>( 201 bus, object_path::hostState, interface::hostState, "CurrentHostState", 202 *this, [this](const auto& value) { 203 this->_hostState = std::get<std::string>(value); 204 })); 205 206 // Watch the BaseBIOSTable property for the hmc managed attribute 207 _properties.emplace_back(std::make_unique<PropertyWatcher<DataInterface>>( 208 bus, object_path::biosConfigMgr, interface::biosConfigMgr, 209 "BaseBIOSTable", service_name::biosConfigMgr, *this, 210 [this](const auto& value) { 211 const auto& attributes = std::get<BiosAttributes>(value); 212 213 auto it = attributes.find("pvm_hmc_managed"); 214 if (it != attributes.end()) 215 { 216 const auto& currentValVariant = std::get<5>(it->second); 217 auto currentVal = std::get_if<std::string>(¤tValVariant); 218 if (currentVal) 219 { 220 this->_hmcManaged = 221 (*currentVal == "Enabled") ? true : false; 222 } 223 } 224 })); 225 } 226 227 DBusPropertyMap DataInterface::getAllProperties( 228 const std::string& service, const std::string& objectPath, 229 const std::string& interface) const 230 { 231 DBusPropertyMap properties; 232 233 auto method = _bus.new_method_call(service.c_str(), objectPath.c_str(), 234 interface::dbusProperty, "GetAll"); 235 method.append(interface); 236 auto reply = _bus.call(method, dbusTimeout); 237 238 reply.read(properties); 239 240 return properties; 241 } 242 243 void DataInterface::getProperty( 244 const std::string& service, const std::string& objectPath, 245 const std::string& interface, const std::string& property, 246 DBusValue& value) const 247 { 248 auto method = _bus.new_method_call(service.c_str(), objectPath.c_str(), 249 interface::dbusProperty, "Get"); 250 method.append(interface, property); 251 auto reply = _bus.call(method, dbusTimeout); 252 253 reply.read(value); 254 } 255 256 DBusPathList DataInterface::getPaths(const DBusInterfaceList& interfaces) const 257 { 258 auto method = _bus.new_method_call( 259 service_name::objectMapper, object_path::objectMapper, 260 interface::objectMapper, "GetSubTreePaths"); 261 262 method.append(std::string{"/"}, 0, interfaces); 263 264 auto reply = _bus.call(method, dbusTimeout); 265 266 DBusPathList paths; 267 reply.read(paths); 268 269 return paths; 270 } 271 272 DBusService DataInterface::getService(const std::string& objectPath, 273 const std::string& interface) const 274 { 275 auto method = _bus.new_method_call(service_name::objectMapper, 276 object_path::objectMapper, 277 interface::objectMapper, "GetObject"); 278 279 method.append(objectPath, std::vector<std::string>({interface})); 280 281 auto reply = _bus.call(method, dbusTimeout); 282 283 std::map<DBusService, DBusInterfaceList> response; 284 reply.read(response); 285 286 if (!response.empty()) 287 { 288 return response.begin()->first; 289 } 290 291 return std::string{}; 292 } 293 294 void DataInterface::readBMCFWVersion() 295 { 296 _bmcFWVersion = 297 phosphor::logging::util::getOSReleaseValue("VERSION").value_or(""); 298 } 299 300 void DataInterface::readServerFWVersion() 301 { 302 auto value = 303 phosphor::logging::util::getOSReleaseValue("VERSION_ID").value_or(""); 304 if ((value != "") && (value.find_last_of(')') != std::string::npos)) 305 { 306 std::size_t pos = value.find_first_of('(') + 1; 307 _serverFWVersion = value.substr(pos, value.find_last_of(')') - pos); 308 } 309 } 310 311 void DataInterface::readBMCFWVersionID() 312 { 313 _bmcFWVersionID = 314 phosphor::logging::util::getOSReleaseValue("VERSION_ID").value_or(""); 315 } 316 317 std::string DataInterface::getMachineTypeModel() const 318 { 319 std::string model; 320 try 321 { 322 auto service = getService(object_path::systemInv, interface::invAsset); 323 if (!service.empty()) 324 { 325 DBusValue value; 326 getProperty(service, object_path::systemInv, interface::invAsset, 327 "Model", value); 328 329 model = std::get<std::string>(value); 330 } 331 } 332 catch (const std::exception& e) 333 { 334 lg2::warning("Failed reading Model property from " 335 "interface: {IFACE} exception: {ERROR}", 336 "IFACE", interface::invAsset, "ERROR", e); 337 } 338 339 return model; 340 } 341 342 std::string DataInterface::getMachineSerialNumber() const 343 { 344 std::string sn; 345 try 346 { 347 auto service = getService(object_path::systemInv, interface::invAsset); 348 if (!service.empty()) 349 { 350 DBusValue value; 351 getProperty(service, object_path::systemInv, interface::invAsset, 352 "SerialNumber", value); 353 354 sn = std::get<std::string>(value); 355 } 356 } 357 catch (const std::exception& e) 358 { 359 lg2::warning("Failed reading SerialNumber property from " 360 "interface: {IFACE} exception: {ERROR}", 361 "IFACE", interface::invAsset, "ERROR", e); 362 } 363 364 return sn; 365 } 366 367 std::string DataInterface::getMotherboardCCIN() const 368 { 369 std::string ccin; 370 371 try 372 { 373 auto service = 374 getService(object_path::motherBoardInv, interface::viniRecordVPD); 375 if (!service.empty()) 376 { 377 DBusValue value; 378 getProperty(service, object_path::motherBoardInv, 379 interface::viniRecordVPD, "CC", value); 380 381 auto cc = std::get<std::vector<uint8_t>>(value); 382 ccin = std::string{cc.begin(), cc.end()}; 383 } 384 } 385 catch (const std::exception& e) 386 { 387 lg2::warning("Failed reading Motherboard CCIN property from " 388 "interface: {IFACE} exception: {ERROR}", 389 "IFACE", interface::viniRecordVPD, "ERROR", e); 390 } 391 392 return ccin; 393 } 394 395 std::vector<uint8_t> DataInterface::getSystemIMKeyword() const 396 { 397 std::vector<uint8_t> systemIM; 398 399 try 400 { 401 auto service = 402 getService(object_path::motherBoardInv, interface::vsbpRecordVPD); 403 if (!service.empty()) 404 { 405 DBusValue value; 406 getProperty(service, object_path::motherBoardInv, 407 interface::vsbpRecordVPD, "IM", value); 408 409 systemIM = std::get<std::vector<uint8_t>>(value); 410 } 411 } 412 catch (const std::exception& e) 413 { 414 lg2::warning("Failed reading System IM property from " 415 "interface: {IFACE} exception: {ERROR}", 416 "IFACE", interface::vsbpRecordVPD, "ERROR", e); 417 } 418 419 return systemIM; 420 } 421 422 void DataInterface::getHWCalloutFields( 423 const std::string& inventoryPath, std::string& fruPartNumber, 424 std::string& ccin, std::string& serialNumber) const 425 { 426 // For now, attempt to get all of the properties directly on the path 427 // passed in. In the future, may need to make use of an algorithm 428 // to figure out which inventory objects actually hold these 429 // interfaces in the case of non FRUs, or possibly another service 430 // will provide this info. Any missing interfaces will result 431 // in exceptions being thrown. 432 433 auto service = getService(inventoryPath, interface::viniRecordVPD); 434 435 auto properties = 436 getAllProperties(service, inventoryPath, interface::viniRecordVPD); 437 438 auto value = std::get<std::vector<uint8_t>>(properties["FN"]); 439 fruPartNumber = std::string{value.begin(), value.end()}; 440 441 value = std::get<std::vector<uint8_t>>(properties["CC"]); 442 ccin = std::string{value.begin(), value.end()}; 443 444 value = std::get<std::vector<uint8_t>>(properties["SN"]); 445 serialNumber = std::string{value.begin(), value.end()}; 446 } 447 448 std::string 449 DataInterface::getLocationCode(const std::string& inventoryPath) const 450 { 451 auto service = getService(inventoryPath, interface::locCode); 452 453 DBusValue locCode; 454 getProperty(service, inventoryPath, interface::locCode, "LocationCode", 455 locCode); 456 457 return std::get<std::string>(locCode); 458 } 459 460 std::string 461 DataInterface::addLocationCodePrefix(const std::string& locationCode) 462 { 463 static const std::string locationCodePrefix{"Ufcs-"}; 464 465 // Technically there are 2 location code prefixes, Ufcs and Umts, so 466 // if it already starts with a U then don't need to do anything. 467 if (locationCode.front() != 'U') 468 { 469 return locationCodePrefix + locationCode; 470 } 471 472 return locationCode; 473 } 474 475 std::string DataInterface::expandLocationCode(const std::string& locationCode, 476 uint16_t /*node*/) const 477 { 478 // Location codes for connectors are the location code of the FRU they are 479 // on, plus a '-Tx' segment. Remove this last segment before expanding it 480 // and then add it back in afterwards. This way, the connector doesn't have 481 // to be in the model just so that it can be expanded. 482 auto [baseLoc, connectorLoc] = extractConnectorFromLocCode(locationCode); 483 484 auto method = 485 _bus.new_method_call(service_name::vpdManager, object_path::vpdManager, 486 interface::vpdManager, "GetExpandedLocationCode"); 487 488 method.append(addLocationCodePrefix(baseLoc), static_cast<uint16_t>(0)); 489 490 auto reply = _bus.call(method, dbusTimeout); 491 492 std::string expandedLocationCode; 493 reply.read(expandedLocationCode); 494 495 if (!connectorLoc.empty()) 496 { 497 expandedLocationCode += connectorLoc; 498 } 499 500 return expandedLocationCode; 501 } 502 503 std::vector<std::string> DataInterface::getInventoryFromLocCode( 504 const std::string& locationCode, uint16_t node, bool expanded) const 505 { 506 std::string methodName = expanded ? "GetFRUsByExpandedLocationCode" 507 : "GetFRUsByUnexpandedLocationCode"; 508 509 // Remove the connector segment, if present, so that this method call 510 // returns an inventory path that getHWCalloutFields() can be used with. 511 // (The serial number, etc, aren't stored on the connector in the 512 // inventory, and may not even be modeled.) 513 auto [baseLoc, connectorLoc] = extractConnectorFromLocCode(locationCode); 514 515 auto method = 516 _bus.new_method_call(service_name::vpdManager, object_path::vpdManager, 517 interface::vpdManager, methodName.c_str()); 518 519 if (expanded) 520 { 521 method.append(baseLoc); 522 } 523 else 524 { 525 method.append(addLocationCodePrefix(baseLoc), node); 526 } 527 528 auto reply = _bus.call(method, dbusTimeout); 529 530 std::vector<sdbusplus::message::object_path> entries; 531 reply.read(entries); 532 533 std::vector<std::string> paths; 534 535 // Note: The D-Bus method will fail if nothing found. 536 std::for_each(entries.begin(), entries.end(), 537 [&paths](const auto& path) { paths.push_back(path); }); 538 539 return paths; 540 } 541 542 void DataInterface::assertLEDGroup(const std::string& ledGroup, 543 bool value) const 544 { 545 DBusValue variant = value; 546 auto method = 547 _bus.new_method_call(service_name::ledGroupManager, ledGroup.c_str(), 548 interface::dbusProperty, "Set"); 549 method.append(interface::ledGroup, "Asserted", variant); 550 _bus.call(method, dbusTimeout); 551 } 552 553 void DataInterface::setFunctional(const std::string& objectPath, 554 bool value) const 555 { 556 DBusPropertyMap prop{{"Functional", value}}; 557 DBusInterfaceMap iface{{interface::operationalStatus, prop}}; 558 559 // PIM takes a relative path like /system/chassis so remove 560 // /xyz/openbmc_project/inventory if present. 561 std::string path{objectPath}; 562 if (path.starts_with(object_path::baseInv)) 563 { 564 path = objectPath.substr(strlen(object_path::baseInv)); 565 } 566 DBusObjectMap object{{path, iface}}; 567 568 auto method = _bus.new_method_call(service_name::inventoryManager, 569 object_path::baseInv, 570 interface::inventoryManager, "Notify"); 571 method.append(std::move(object)); 572 _bus.call(method, dbusTimeout); 573 } 574 575 using AssociationTuple = std::tuple<std::string, std::string, std::string>; 576 using AssociationsProperty = std::vector<AssociationTuple>; 577 578 void DataInterface::setCriticalAssociation(const std::string& objectPath) const 579 { 580 DBusValue getAssociationValue; 581 582 auto service = getService(objectPath, interface::associationDef); 583 584 getProperty(service, objectPath, interface::associationDef, "Associations", 585 getAssociationValue); 586 587 auto association = std::get<AssociationsProperty>(getAssociationValue); 588 589 AssociationTuple critAssociation{ 590 "health_rollup", "critical", 591 "/xyz/openbmc_project/inventory/system/chassis"}; 592 593 if (std::find(association.begin(), association.end(), critAssociation) == 594 association.end()) 595 { 596 association.push_back(critAssociation); 597 DBusValue setAssociationValue = association; 598 599 auto method = _bus.new_method_call(service.c_str(), objectPath.c_str(), 600 interface::dbusProperty, "Set"); 601 602 method.append(interface::associationDef, "Associations", 603 setAssociationValue); 604 _bus.call(method, dbusTimeout); 605 } 606 } 607 608 std::vector<std::string> DataInterface::getSystemNames() const 609 { 610 DBusSubTree subtree; 611 DBusValue names; 612 613 auto method = _bus.new_method_call(service_name::objectMapper, 614 object_path::objectMapper, 615 interface::objectMapper, "GetSubTree"); 616 method.append(std::string{"/"}, 0, 617 std::vector<std::string>{interface::compatible}); 618 auto reply = _bus.call(method, dbusTimeout); 619 620 reply.read(subtree); 621 if (subtree.empty()) 622 { 623 throw std::runtime_error("Compatible interface not on D-Bus"); 624 } 625 626 for (const auto& [path, interfaceMap] : subtree) 627 { 628 auto iface = interfaceMap.find(service_name::entityManager); 629 if (iface == interfaceMap.end()) 630 { 631 continue; 632 } 633 634 getProperty(iface->first, path, interface::compatible, "Names", names); 635 636 return std::get<std::vector<std::string>>(names); 637 } 638 639 throw std::runtime_error("EM Compatible interface not on D-Bus"); 640 } 641 642 bool DataInterface::getQuiesceOnError() const 643 { 644 bool ret = false; 645 646 try 647 { 648 auto service = 649 getService(object_path::logSetting, interface::logSetting); 650 if (!service.empty()) 651 { 652 DBusValue value; 653 getProperty(service, object_path::logSetting, interface::logSetting, 654 "QuiesceOnHwError", value); 655 656 ret = std::get<bool>(value); 657 } 658 } 659 catch (const std::exception& e) 660 { 661 lg2::warning("Failed reading QuiesceOnHwError property from " 662 "interface: {IFACE} exception: {ERROR}", 663 "IFACE", interface::logSetting, "ERROR", e); 664 } 665 666 return ret; 667 } 668 669 std::vector<bool> 670 DataInterface::checkDumpStatus(const std::vector<std::string>& type) const 671 { 672 DBusSubTree subtree; 673 std::vector<bool> result(type.size(), false); 674 675 // Query GetSubTree for the availability of dump interface 676 auto method = _bus.new_method_call(service_name::objectMapper, 677 object_path::objectMapper, 678 interface::objectMapper, "GetSubTree"); 679 method.append(std::string{"/"}, 0, 680 std::vector<std::string>{interface::dumpEntry}); 681 auto reply = _bus.call(method, dbusTimeout); 682 683 reply.read(subtree); 684 685 if (subtree.empty()) 686 { 687 return result; 688 } 689 690 std::vector<bool>::iterator itDumpStatus = result.begin(); 691 uint8_t count = 0; 692 for (const auto& [path, serviceInfo] : subtree) 693 { 694 const auto& service = serviceInfo.begin()->first; 695 // Check for dump type on the object path 696 for (const auto& it : type) 697 { 698 if (path.find(it) != std::string::npos) 699 { 700 DBusValue value, progress; 701 702 // If dump type status is already available go for next path 703 if (*itDumpStatus) 704 { 705 break; 706 } 707 708 // Check for valid dump to be available if following 709 // conditions are met for the dump entry path - 710 // Offloaded == false and Status == Completed 711 getProperty(service, path, interface::dumpEntry, "Offloaded", 712 value); 713 getProperty(service, path, interface::dumpProgress, "Status", 714 progress); 715 auto offload = std::get<bool>(value); 716 auto status = std::get<std::string>(progress); 717 if (!offload && (status.find("Completed") != std::string::npos)) 718 { 719 *itDumpStatus = true; 720 count++; 721 if (count >= type.size()) 722 { 723 return result; 724 } 725 break; 726 } 727 } 728 ++itDumpStatus; 729 } 730 itDumpStatus = result.begin(); 731 } 732 733 return result; 734 } 735 736 void DataInterface::createGuardRecord(const std::vector<uint8_t>& binPath, 737 const std::string& type, 738 const std::string& logPath) const 739 { 740 try 741 { 742 auto method = _bus.new_method_call( 743 service_name::hwIsolation, object_path::hwIsolation, 744 interface::hwIsolationCreate, "CreateWithEntityPath"); 745 method.append(binPath, type, sdbusplus::message::object_path(logPath)); 746 // Note: hw isolation "CreateWithEntityPath" got dependency on logging 747 // api's. Making d-bus call no reply type to avoid cyclic dependency. 748 // Added minimal timeout to catch initial failures. 749 // Need to revisit this design later to avoid cyclic dependency. 750 constexpr auto hwIsolationTimeout = 100000; // in micro seconds 751 _bus.call_noreply(method, hwIsolationTimeout); 752 } 753 754 catch (const sdbusplus::exception_t& e) 755 { 756 std::string errName = e.name(); 757 // SD_BUS_ERROR_TIMEOUT error is expected, due to PEL api dependency 758 // mentioned above. Ignoring the error. 759 if (errName != SD_BUS_ERROR_TIMEOUT) 760 { 761 lg2::error("GUARD D-Bus call exception. Path={PATH}, " 762 "interface = {IFACE}, exception = {ERROR}", 763 "PATH", object_path::hwIsolation, "IFACE", 764 interface::hwIsolationCreate, "ERROR", e); 765 } 766 } 767 } 768 769 void DataInterface::createProgressSRC( 770 const uint64_t& priSRC, const std::vector<uint8_t>& srcStruct) const 771 { 772 DBusValue variant = std::make_tuple(priSRC, srcStruct); 773 774 auto method = _bus.new_method_call(service_name::bootRawProgress, 775 object_path::bootRawProgress, 776 interface::dbusProperty, "Set"); 777 778 method.append(interface::bootRawProgress, "Value", variant); 779 780 _bus.call(method, dbusTimeout); 781 } 782 783 std::vector<uint32_t> DataInterface::getLogIDWithHwIsolation() const 784 { 785 std::vector<std::string> association = {"xyz.openbmc_project.Association"}; 786 std::string hwErrorLog = "/isolated_hw_errorlog"; 787 std::string errorLog = "/error_log"; 788 DBusPathList paths; 789 std::vector<uint32_t> ids; 790 791 // Get all latest mapper associations 792 paths = getPaths(association); 793 for (auto& path : paths) 794 { 795 // Look for object path with hardware isolation entry if any 796 size_t pos = path.find(hwErrorLog); 797 if (pos != std::string::npos) 798 { 799 // Get the object path 800 std::string ph = path; 801 ph.erase(pos, hwErrorLog.length()); 802 auto service = getService(ph, interface::hwIsolationEntry); 803 if (!service.empty()) 804 { 805 bool status; 806 DBusValue value; 807 808 // Read the Resolved property from object path 809 getProperty(service, ph, interface::hwIsolationEntry, 810 "Resolved", value); 811 812 status = std::get<bool>(value); 813 814 // If the entry isn't resolved 815 if (!status) 816 { 817 auto assocService = 818 getService(path, interface::association); 819 if (!assocService.empty()) 820 { 821 DBusValue endpoints; 822 823 // Read Endpoints property 824 getProperty(assocService, path, interface::association, 825 "endpoints", endpoints); 826 827 auto logPath = 828 std::get<std::vector<std::string>>(endpoints); 829 if (!logPath.empty()) 830 { 831 // Get OpenBMC event log Id 832 uint32_t id = stoi(logPath[0].substr( 833 logPath[0].find_last_of('/') + 1)); 834 ids.push_back(id); 835 } 836 } 837 } 838 } 839 } 840 841 // Look for object path with error_log entry if any 842 pos = path.find(errorLog); 843 if (pos != std::string::npos) 844 { 845 auto service = getService(path, interface::association); 846 if (!service.empty()) 847 { 848 DBusValue value; 849 850 // Read Endpoints property 851 getProperty(service, path, interface::association, "endpoints", 852 value); 853 854 auto logPath = std::get<std::vector<std::string>>(value); 855 if (!logPath.empty()) 856 { 857 // Get OpenBMC event log Id 858 uint32_t id = stoi( 859 logPath[0].substr(logPath[0].find_last_of('/') + 1)); 860 ids.push_back(id); 861 } 862 } 863 } 864 } 865 866 if (ids.size() > 1) 867 { 868 // remove duplicates to have only unique ids 869 std::sort(ids.begin(), ids.end()); 870 ids.erase(std::unique(ids.begin(), ids.end()), ids.end()); 871 } 872 return ids; 873 } 874 875 std::vector<uint8_t> DataInterface::getRawProgressSRC(void) const 876 { 877 using RawProgressProperty = std::tuple<uint64_t, std::vector<uint8_t>>; 878 879 DBusValue value; 880 getProperty(service_name::bootRawProgress, object_path::bootRawProgress, 881 interface::bootRawProgress, "Value", value); 882 883 const auto& rawProgress = std::get<RawProgressProperty>(value); 884 return std::get<1>(rawProgress); 885 } 886 887 std::optional<std::vector<uint8_t>> 888 DataInterface::getDIProperty(const std::string& locationCode) const 889 { 890 std::vector<uint8_t> viniDI; 891 892 try 893 { 894 // Note : The hardcoded value 0 should be changed when comes to 895 // multinode system. 896 auto objectPath = getInventoryFromLocCode(locationCode, 0, true); 897 898 DBusValue value; 899 getProperty(service_name::inventoryManager, objectPath[0], 900 interface::viniRecordVPD, "DI", value); 901 902 viniDI = std::get<std::vector<uint8_t>>(value); 903 } 904 catch (const std::exception& e) 905 { 906 lg2::warning( 907 "Failed reading DI property for the location code : {LOC_CODE} from " 908 "interface: {IFACE} exception: {ERROR}", 909 "LOC_CODE", locationCode, "IFACE", interface::viniRecordVPD, 910 "ERROR", e); 911 return std::nullopt; 912 } 913 914 return viniDI; 915 } 916 917 std::optional<bool> 918 DataInterfaceBase::isDIMMLocCode(const std::string& locCode) const 919 { 920 if (_locationCache.contains(locCode)) 921 { 922 return _locationCache.at(locCode); 923 } 924 else 925 { 926 return std::nullopt; 927 } 928 } 929 930 void DataInterfaceBase::addDIMMLocCode(const std::string& locCode, 931 bool isFRUDIMM) 932 { 933 _locationCache.insert({locCode, isFRUDIMM}); 934 } 935 936 std::expected<bool, std::string> 937 DataInterfaceBase::isDIMM(const std::string& locCode) 938 { 939 auto isDIMMType = isDIMMLocCode(locCode); 940 if (isDIMMType.has_value()) 941 { 942 return isDIMMType.value(); 943 } 944 #ifndef PEL_ENABLE_PHAL 945 return std::unexpected<std::string>( 946 std::format("PHAL feature is not enabled, so the LocationCode:[{}] " 947 "cannot be determined as DIMM", 948 locCode)); 949 #else 950 else 951 { 952 // Invoke pHAL API inorder to fetch the FRU Type 953 auto fruType = openpower::phal::pdbg::getFRUType(locCode); 954 if (fruType.has_value()) 955 { 956 bool isDIMMFRU{false}; 957 if (fruType.value() == ENUM_ATTR_TYPE_DIMM) 958 { 959 isDIMMFRU = true; 960 } 961 addDIMMLocCode(locCode, isDIMMFRU); 962 return isDIMMFRU; 963 } 964 else 965 { 966 std::string msg{std::format("Failed to determine the HW Type, " 967 "LocationCode:[{}]", 968 locCode)}; 969 if (openpower::phal::exception::errMsgMap.contains(fruType.error())) 970 { 971 msg = std::format( 972 "{} PHALErrorMsg:[{}]", msg, 973 openpower::phal::exception::errMsgMap.at(fruType.error())); 974 } 975 else 976 { 977 msg = std::format( 978 "{} PHALErrorMsg:[Unknown PHALErrorCode:{}]", msg, 979 std::to_underlying<openpower::phal::exception::ERR_TYPE>( 980 fruType.error())); 981 } 982 return std::unexpected<std::string>(msg); 983 } 984 } 985 #endif 986 } 987 988 void DataInterface::startFruPlugWatch() 989 { 990 // Add a watch on inventory InterfacesAdded and then find all 991 // existing hotpluggable interfaces and add propertiesChanged 992 // watches on them. 993 994 _invIaMatch = std::make_unique<sdbusplus::bus::match_t>( 995 _bus, match_rules::interfacesAdded(object_path::baseInv), 996 std::bind(&DataInterface::inventoryIfaceAdded, this, 997 std::placeholders::_1)); 998 try 999 { 1000 auto paths = getPaths(hotplugInterfaces); 1001 1002 _invPresentMatches.clear(); 1003 1004 std::for_each(paths.begin(), paths.end(), 1005 [this](const auto& path) { addHotplugWatch(path); }); 1006 } 1007 catch (const sdbusplus::exception_t& e) 1008 { 1009 lg2::warning("Failed getting FRU paths to watch: {ERROR}", "ERROR", e); 1010 } 1011 } 1012 1013 void DataInterface::addHotplugWatch(const std::string& path) 1014 { 1015 if (!_invPresentMatches.contains(path)) 1016 { 1017 _invPresentMatches.emplace( 1018 path, 1019 std::make_unique<sdbusplus::bus::match_t>( 1020 _bus, match_rules::propertiesChanged(path, interface::invItem), 1021 std::bind(&DataInterface::presenceChanged, this, 1022 std::placeholders::_1))); 1023 } 1024 } 1025 1026 void DataInterface::inventoryIfaceAdded(sdbusplus::message_t& msg) 1027 { 1028 sdbusplus::message::object_path path; 1029 DBusInterfaceMap interfaces; 1030 1031 msg.read(path, interfaces); 1032 1033 // Check if any of the new interfaces are for hot pluggable FRUs. 1034 if (std::find_if(interfaces.begin(), interfaces.end(), 1035 [](const auto& interfacePair) { 1036 return std::find(hotplugInterfaces.begin(), 1037 hotplugInterfaces.end(), 1038 interfacePair.first) != 1039 hotplugInterfaces.end(); 1040 }) == interfaces.end()) 1041 { 1042 return; 1043 } 1044 1045 addHotplugWatch(path.str); 1046 1047 // If an Inventory.Item interface was also added, check presence now. 1048 1049 // Notes: 1050 // * This assumes the Inv.Item and Inv.Fan/PS are added together which 1051 // is currently the case. 1052 // * If the code ever switches to something without a Present 1053 // property, then the IA signal itself would probably indicate presence. 1054 1055 auto itemIt = interfaces.find(interface::invItem); 1056 if (itemIt != interfaces.end()) 1057 { 1058 notifyPresenceSubsribers(path.str, itemIt->second); 1059 } 1060 } 1061 1062 void DataInterface::presenceChanged(sdbusplus::message_t& msg) 1063 { 1064 DBusInterface interface; 1065 DBusPropertyMap properties; 1066 1067 msg.read(interface, properties); 1068 if (interface != interface::invItem) 1069 { 1070 return; 1071 } 1072 1073 std::string path = msg.get_path(); 1074 notifyPresenceSubsribers(path, properties); 1075 } 1076 1077 void DataInterface::notifyPresenceSubsribers(const std::string& path, 1078 const DBusPropertyMap& properties) 1079 { 1080 auto prop = properties.find("Present"); 1081 if ((prop == properties.end()) || (!std::get<bool>(prop->second))) 1082 { 1083 return; 1084 } 1085 1086 std::string locCode; 1087 1088 try 1089 { 1090 auto service = getService(path, interface::locCode); 1091 1092 // If the hotplugged FRU is hosted by PLDM, then it is 1093 // in an IO expansion drawer and we don't care about it. 1094 if (service == service_name::pldm) 1095 { 1096 return; 1097 } 1098 1099 locCode = getLocationCode(path); 1100 } 1101 catch (const sdbusplus::exception_t& e) 1102 { 1103 lg2::debug("Could not get location code for {PATH}: {ERROR}", "PATH", 1104 path, "ERROR", e); 1105 return; 1106 } 1107 1108 lg2::debug("Detected FRU {PATH} ({LOC}) present ", "PATH", path, "LOC", 1109 locCode); 1110 1111 // Tell the subscribers. 1112 setFruPresent(locCode); 1113 } 1114 } // namespace pels 1115 } // namespace openpower 1116