1 #include "config.h" 2 3 #include "ibm_vpd_utils.hpp" 4 5 #include "common_utility.hpp" 6 #include "const.hpp" 7 #include "defines.hpp" 8 #include "vpd_exceptions.hpp" 9 10 #include <boost/algorithm/string.hpp> 11 #include <gpiod.hpp> 12 #include <nlohmann/json.hpp> 13 #include <phosphor-logging/elog-errors.hpp> 14 #include <phosphor-logging/log.hpp> 15 #include <sdbusplus/server.hpp> 16 #include <xyz/openbmc_project/Common/error.hpp> 17 18 #include <filesystem> 19 #include <fstream> 20 #include <iomanip> 21 #include <regex> 22 #include <sstream> 23 #include <vector> 24 25 using json = nlohmann::json; 26 27 namespace openpower 28 { 29 namespace vpd 30 { 31 using namespace openpower::vpd::constants; 32 using namespace inventory; 33 using namespace phosphor::logging; 34 using namespace sdbusplus::xyz::openbmc_project::Common::Error; 35 using namespace record; 36 using namespace openpower::vpd::exceptions; 37 using namespace common::utility; 38 using Severity = openpower::vpd::constants::PelSeverity; 39 namespace fs = std::filesystem; 40 41 // mapping of severity enum to severity interface 42 static std::unordered_map<Severity, std::string> sevMap = { 43 {Severity::INFORMATIONAL, 44 "xyz.openbmc_project.Logging.Entry.Level.Informational"}, 45 {Severity::DEBUG, "xyz.openbmc_project.Logging.Entry.Level.Debug"}, 46 {Severity::NOTICE, "xyz.openbmc_project.Logging.Entry.Level.Notice"}, 47 {Severity::WARNING, "xyz.openbmc_project.Logging.Entry.Level.Warning"}, 48 {Severity::CRITICAL, "xyz.openbmc_project.Logging.Entry.Level.Critical"}, 49 {Severity::EMERGENCY, "xyz.openbmc_project.Logging.Entry.Level.Emergency"}, 50 {Severity::ERROR, "xyz.openbmc_project.Logging.Entry.Level.Error"}, 51 {Severity::ALERT, "xyz.openbmc_project.Logging.Entry.Level.Alert"}}; 52 53 namespace inventory 54 { 55 56 MapperResponse 57 getObjectSubtreeForInterfaces(const std::string& root, const int32_t depth, 58 const std::vector<std::string>& interfaces) 59 { 60 auto bus = sdbusplus::bus::new_default(); 61 auto mapperCall = bus.new_method_call(mapperDestination, mapperObjectPath, 62 mapperInterface, "GetSubTree"); 63 mapperCall.append(root); 64 mapperCall.append(depth); 65 mapperCall.append(interfaces); 66 67 MapperResponse result = {}; 68 69 try 70 { 71 auto response = bus.call(mapperCall); 72 73 response.read(result); 74 } 75 catch (const sdbusplus::exception_t& e) 76 { 77 log<level::ERR>("Error in mapper GetSubTree", 78 entry("ERROR=%s", e.what())); 79 } 80 81 return result; 82 } 83 84 MapperGetObjectResponse getObject(const std::string& objectPath, 85 const std::vector<std::string>& interfaces) 86 { 87 auto bus = sdbusplus::bus::new_default(); 88 auto mapperCall = bus.new_method_call(mapperDestination, mapperObjectPath, 89 mapperInterface, "GetObject"); 90 mapperCall.append(objectPath); 91 mapperCall.append(interfaces); 92 93 MapperGetObjectResponse result = {}; 94 95 try 96 { 97 auto response = bus.call(mapperCall); 98 99 response.read(result); 100 } 101 catch (const sdbusplus::exception_t& e) 102 { 103 log<level::ERR>("Error in mapper GetObject", 104 entry("ERROR=%s", e.what())); 105 } 106 107 return result; 108 } 109 110 } // namespace inventory 111 112 LE2ByteData readUInt16LE(Binary::const_iterator iterator) 113 { 114 LE2ByteData lowByte = *iterator; 115 LE2ByteData highByte = *(iterator + 1); 116 lowByte |= (highByte << 8); 117 return lowByte; 118 } 119 120 /** @brief Encodes a keyword for D-Bus. 121 */ 122 std::string encodeKeyword(const std::string& kw, const std::string& encoding) 123 { 124 if (encoding == "MAC") 125 { 126 std::string res{}; 127 size_t first = kw[0]; 128 res += toHex(first >> 4); 129 res += toHex(first & 0x0f); 130 for (size_t i = 1; i < kw.size(); ++i) 131 { 132 res += ":"; 133 res += toHex(kw[i] >> 4); 134 res += toHex(kw[i] & 0x0f); 135 } 136 return res; 137 } 138 else if (encoding == "DATE") 139 { 140 // Date, represent as 141 // <year>-<month>-<day> <hour>:<min> 142 std::string res{}; 143 static constexpr uint8_t skipPrefix = 3; 144 145 auto strItr = kw.begin(); 146 advance(strItr, skipPrefix); 147 for_each(strItr, kw.end(), [&res](size_t c) { res += c; }); 148 149 res.insert(BD_YEAR_END, 1, '-'); 150 res.insert(BD_MONTH_END, 1, '-'); 151 res.insert(BD_DAY_END, 1, ' '); 152 res.insert(BD_HOUR_END, 1, ':'); 153 154 return res; 155 } 156 else // default to string encoding 157 { 158 return std::string(kw.begin(), kw.end()); 159 } 160 } 161 162 std::string readBusProperty(const std::string& obj, const std::string& inf, 163 const std::string& prop) 164 { 165 std::string propVal{}; 166 std::string object = INVENTORY_PATH + obj; 167 auto bus = sdbusplus::bus::new_default(); 168 auto properties = bus.new_method_call( 169 "xyz.openbmc_project.Inventory.Manager", object.c_str(), 170 "org.freedesktop.DBus.Properties", "Get"); 171 properties.append(inf); 172 properties.append(prop); 173 auto result = bus.call(properties); 174 if (!result.is_method_error()) 175 { 176 inventory::Value val; 177 result.read(val); 178 if (auto pVal = std::get_if<Binary>(&val)) 179 { 180 propVal.assign(reinterpret_cast<const char*>(pVal->data()), 181 pVal->size()); 182 } 183 else if (auto pVal = std::get_if<std::string>(&val)) 184 { 185 propVal.assign(pVal->data(), pVal->size()); 186 } 187 else if (auto pVal = get_if<bool>(&val)) 188 { 189 if (*pVal == false) 190 { 191 propVal = "false"; 192 } 193 else 194 { 195 propVal = "true"; 196 } 197 } 198 } 199 return propVal; 200 } 201 202 void createPEL(const std::map<std::string, std::string>& additionalData, 203 const Severity& sev, const std::string& errIntf, sd_bus* sdBus) 204 { 205 // This pointer will be NULL in case the call is made from ibm-read-vpd. In 206 // that case a sync call will do. 207 if (sdBus == nullptr) 208 { 209 createSyncPEL(additionalData, sev, errIntf); 210 } 211 else 212 { 213 std::string errDescription{}; 214 auto pos = additionalData.find("DESCRIPTION"); 215 if (pos != additionalData.end()) 216 { 217 errDescription = pos->second; 218 } 219 else 220 { 221 errDescription = "Description field missing in additional data"; 222 } 223 224 std::string pelSeverity = 225 "xyz.openbmc_project.Logging.Entry.Level.Error"; 226 auto itr = sevMap.find(sev); 227 if (itr != sevMap.end()) 228 { 229 pelSeverity = itr->second; 230 } 231 232 // Implies this is a call from Manager. Hence we need to make an async 233 // call to avoid deadlock with Phosphor-logging. 234 auto rc = sd_bus_call_method_async( 235 sdBus, NULL, loggerService, loggerObjectPath, loggerCreateInterface, 236 "Create", NULL, NULL, "ssa{ss}", errIntf.c_str(), 237 pelSeverity.c_str(), 1, "DESCRIPTION", errDescription.c_str()); 238 239 if (rc < 0) 240 { 241 log<level::ERR>("Error calling sd_bus_call_method_async", 242 entry("RC=%d", rc), entry("MSG=%s", strerror(-rc))); 243 } 244 } 245 } 246 247 void createSyncPEL(const std::map<std::string, std::string>& additionalData, 248 const Severity& sev, const std::string& errIntf) 249 { 250 try 251 { 252 std::string pelSeverity = 253 "xyz.openbmc_project.Logging.Entry.Level.Error"; 254 auto bus = sdbusplus::bus::new_default(); 255 auto service = getService(bus, loggerObjectPath, loggerCreateInterface); 256 auto method = bus.new_method_call(service.c_str(), loggerObjectPath, 257 loggerCreateInterface, "Create"); 258 259 auto itr = sevMap.find(sev); 260 if (itr != sevMap.end()) 261 { 262 pelSeverity = itr->second; 263 } 264 265 method.append(errIntf, pelSeverity, additionalData); 266 auto resp = bus.call(method); 267 } 268 catch (const sdbusplus::exception_t& e) 269 { 270 std::cerr << "Dbus call to phosphor-logging Create failed. Reason:" 271 << e.what(); 272 } 273 } 274 275 inventory::VPDfilepath getVpdFilePath(const std::string& jsonFile, 276 const std::string& ObjPath) 277 { 278 std::ifstream inventoryJson(jsonFile); 279 const auto& jsonObject = json::parse(inventoryJson); 280 inventory::VPDfilepath filePath{}; 281 282 if (jsonObject.find("frus") == jsonObject.end()) 283 { 284 throw(VpdJsonException( 285 "Invalid JSON structure - frus{} object not found in ", jsonFile)); 286 } 287 288 const nlohmann::json& groupFRUS = 289 jsonObject["frus"].get_ref<const nlohmann::json::object_t&>(); 290 for (const auto& itemFRUS : groupFRUS.items()) 291 { 292 const std::vector<nlohmann::json>& groupEEPROM = 293 itemFRUS.value().get_ref<const nlohmann::json::array_t&>(); 294 for (const auto& itemEEPROM : groupEEPROM) 295 { 296 if (itemEEPROM["inventoryPath"] 297 .get_ref<const nlohmann::json::string_t&>() == ObjPath) 298 { 299 filePath = itemFRUS.key(); 300 return filePath; 301 } 302 } 303 } 304 305 return filePath; 306 } 307 308 bool isPathInJson(const std::string& eepromPath) 309 { 310 bool present = false; 311 std::ifstream inventoryJson(INVENTORY_JSON_SYM_LINK); 312 313 try 314 { 315 auto js = json::parse(inventoryJson); 316 if (js.find("frus") == js.end()) 317 { 318 throw(VpdJsonException( 319 "Invalid JSON structure - frus{} object not found in ", 320 INVENTORY_JSON_SYM_LINK)); 321 } 322 json fruJson = js["frus"]; 323 324 if (fruJson.find(eepromPath) != fruJson.end()) 325 { 326 present = true; 327 } 328 } 329 catch (const json::parse_error& ex) 330 { 331 throw(VpdJsonException("Json Parsing failed", INVENTORY_JSON_SYM_LINK)); 332 } 333 return present; 334 } 335 336 bool isRecKwInDbusJson(const std::string& recordName, 337 const std::string& keyword) 338 { 339 std::ifstream propertyJson(DBUS_PROP_JSON); 340 json dbusProperty; 341 bool present = false; 342 343 if (propertyJson.is_open()) 344 { 345 try 346 { 347 auto dbusPropertyJson = json::parse(propertyJson); 348 if (dbusPropertyJson.find("dbusProperties") == 349 dbusPropertyJson.end()) 350 { 351 throw(VpdJsonException("dbusProperties{} object not found in " 352 "DbusProperties json : ", 353 DBUS_PROP_JSON)); 354 } 355 356 dbusProperty = dbusPropertyJson["dbusProperties"]; 357 if (dbusProperty.contains(recordName)) 358 { 359 const std::vector<std::string>& kwdsToPublish = 360 dbusProperty[recordName]; 361 if (find(kwdsToPublish.begin(), kwdsToPublish.end(), keyword) != 362 kwdsToPublish.end()) // present 363 { 364 present = true; 365 } 366 } 367 } 368 catch (const json::parse_error& ex) 369 { 370 throw(VpdJsonException("Json Parsing failed", DBUS_PROP_JSON)); 371 } 372 } 373 else 374 { 375 // If dbus properties json is not available, we assume the given 376 // record-keyword is part of dbus-properties json. So setting the bool 377 // variable to true. 378 present = true; 379 } 380 return present; 381 } 382 383 vpdType vpdTypeCheck(const Binary& vpdVector) 384 { 385 // Read first 3 Bytes to check the 11S bar code format 386 std::string is11SFormat = ""; 387 for (uint8_t i = 0; i < FORMAT_11S_LEN; i++) 388 { 389 is11SFormat += vpdVector[MEMORY_VPD_DATA_START + i]; 390 } 391 392 if (vpdVector[IPZ_DATA_START] == KW_VAL_PAIR_START_TAG) 393 { 394 // IPZ VPD FORMAT 395 return vpdType::IPZ_VPD; 396 } 397 else if (vpdVector[KW_VPD_DATA_START] == KW_VPD_START_TAG) 398 { 399 // KEYWORD VPD FORMAT 400 return vpdType::KEYWORD_VPD; 401 } 402 else if (((vpdVector[SPD_BYTE_3] & SPD_BYTE_BIT_0_3_MASK) == 403 SPD_MODULE_TYPE_DDIMM) && 404 (is11SFormat.compare(MEMORY_VPD_START_TAG) == 0)) 405 { 406 // DDIMM Memory VPD format 407 if ((vpdVector[SPD_BYTE_2] & SPD_BYTE_MASK) == SPD_DRAM_TYPE_DDR5) 408 { 409 return vpdType::DDR5_DDIMM_MEMORY_VPD; 410 } 411 else if ((vpdVector[SPD_BYTE_2] & SPD_BYTE_MASK) == SPD_DRAM_TYPE_DDR4) 412 { 413 return vpdType::DDR4_DDIMM_MEMORY_VPD; 414 } 415 } 416 else if ((vpdVector[SPD_BYTE_2] & SPD_BYTE_MASK) == SPD_DRAM_TYPE_DDR5) 417 { 418 // ISDIMM Memory VPD format 419 return vpdType::DDR5_ISDIMM_MEMORY_VPD; 420 } 421 else if ((vpdVector[SPD_BYTE_2] & SPD_BYTE_MASK) == SPD_DRAM_TYPE_DDR4) 422 { 423 // ISDIMM Memory VPD format 424 return vpdType::DDR4_ISDIMM_MEMORY_VPD; 425 } 426 427 // INVALID VPD FORMAT 428 return vpdType::INVALID_VPD_FORMAT; 429 } 430 431 const std::string getIM(const Parsed& vpdMap) 432 { 433 Binary imVal; 434 auto property = vpdMap.find("VSBP"); 435 if (property != vpdMap.end()) 436 { 437 auto kw = (property->second).find("IM"); 438 if (kw != (property->second).end()) 439 { 440 copy(kw->second.begin(), kw->second.end(), back_inserter(imVal)); 441 } 442 } 443 444 std::ostringstream oss; 445 for (auto& i : imVal) 446 { 447 oss << std::setw(2) << std::setfill('0') << std::hex 448 << static_cast<int>(i); 449 } 450 451 return oss.str(); 452 } 453 454 const std::string getHW(const Parsed& vpdMap) 455 { 456 Binary hwVal; 457 auto prop = vpdMap.find("VINI"); 458 if (prop != vpdMap.end()) 459 { 460 auto kw = (prop->second).find("HW"); 461 if (kw != (prop->second).end()) 462 { 463 copy(kw->second.begin(), kw->second.end(), back_inserter(hwVal)); 464 } 465 } 466 467 // The planar pass only comes from the LSB of the HW keyword, 468 // where as the MSB is used for other purposes such as signifying clock 469 // termination. 470 hwVal[0] = 0x00; 471 472 std::ostringstream hwString; 473 for (auto& i : hwVal) 474 { 475 hwString << std::setw(2) << std::setfill('0') << std::hex 476 << static_cast<int>(i); 477 } 478 479 return hwString.str(); 480 } 481 482 std::string getSystemsJson(const Parsed& vpdMap) 483 { 484 std::string jsonPath = "/usr/share/vpd/"; 485 std::string jsonName{}; 486 487 std::ifstream systemJson(SYSTEM_JSON); 488 if (!systemJson) 489 { 490 throw((VpdJsonException("Failed to access Json path", SYSTEM_JSON))); 491 } 492 493 try 494 { 495 auto js = json::parse(systemJson); 496 497 std::string hwKeyword = getHW(vpdMap); 498 const std::string imKeyword = getIM(vpdMap); 499 500 transform(hwKeyword.begin(), hwKeyword.end(), hwKeyword.begin(), 501 ::toupper); 502 503 if (js.find("system") == js.end()) 504 { 505 throw std::runtime_error("Invalid systems Json"); 506 } 507 508 if (js["system"].find(imKeyword) == js["system"].end()) 509 { 510 throw std::runtime_error( 511 "Invalid system. This system type is not present " 512 "in the systemsJson. IM: " + 513 imKeyword); 514 } 515 516 if ((js["system"][imKeyword].find("constraint") != 517 js["system"][imKeyword].end()) && 518 js["system"][imKeyword]["constraint"].find("HW") != 519 js["system"][imKeyword]["constraint"].end()) 520 { 521 // collect hw versions from json, and check hwKeyword is part of it 522 // if hwKeyword is found there then load respective json 523 // otherwise load default one. 524 for (const auto& hwVersion : 525 js["system"][imKeyword]["constraint"]["HW"]) 526 { 527 std::string hw = hwVersion; 528 transform(hw.begin(), hw.end(), hw.begin(), ::toupper); 529 530 if (hw == hwKeyword) 531 { 532 jsonName = js["system"][imKeyword]["constraint"]["json"]; 533 break; 534 } 535 } 536 537 if (jsonName.empty() && js["system"][imKeyword].find("default") != 538 js["system"][imKeyword].end()) 539 { 540 jsonName = js["system"][imKeyword]["default"]; 541 } 542 } 543 else if (js["system"][imKeyword].find("default") != 544 js["system"][imKeyword].end()) 545 { 546 jsonName = js["system"][imKeyword]["default"]; 547 } 548 else 549 { 550 throw std::runtime_error( 551 "Bad System json. Neither constraint nor default found"); 552 } 553 554 jsonPath += jsonName; 555 } 556 557 catch (const json::parse_error& ex) 558 { 559 throw(VpdJsonException("Json Parsing failed", SYSTEM_JSON)); 560 } 561 return jsonPath; 562 } 563 564 void udevToGenericPath(std::string& file, const std::string& driver) 565 { 566 // Sample udevEvent i2c path : 567 // "/sys/devices/platform/ahb/ahb:apb/ahb:apb:bus@1e78a000/1e78a480.i2c-bus/i2c-8/8-0051/8-00510/nvmem" 568 // find if the path contains the word i2c in it. 569 if (file.find("i2c") != std::string::npos) 570 { 571 std::string i2cBusAddr{}; 572 573 // Every udev i2c path should have the common pattern 574 // "i2c-bus_number/bus_number-vpd_address". Search for 575 // "bus_number-vpd_address". 576 std::regex i2cPattern("((i2c)-[0-9]+\\/)([0-9]+-[0-9]{4})"); 577 std::smatch match; 578 if (std::regex_search(file, match, i2cPattern)) 579 { 580 i2cBusAddr = match.str(3); 581 } 582 else 583 { 584 std::cerr << "The given udev path < " << file 585 << " > doesn't match the required pattern. Skipping VPD " 586 "collection." 587 << std::endl; 588 exit(EXIT_SUCCESS); 589 } 590 // Forming the generic file path 591 file = i2cPathPrefix + driver + "/" + i2cBusAddr + "/eeprom"; 592 } 593 // Sample udevEvent spi path : 594 // "/sys/devices/platform/ahb/ahb:apb/1e79b000.fsi/fsi-master/fsi0/slave@00:00/00:00:00:04/spi_master/spi2/spi2.0/spi2.00/nvmem" 595 // find if the path contains the word spi in it. 596 else if (file.find("spi") != std::string::npos) 597 { 598 // Every udev spi path will have common pattern "spi<Digit>/", which 599 // describes the spi bus number at which the fru is connected; Followed 600 // by a slash following the vpd address of the fru. Taking the above 601 // input as a common key, we try to search for the pattern "spi<Digit>/" 602 // using regular expression. 603 std::regex spiPattern("((spi)[0-9]+)(\\/)"); 604 std::string spiBus{}; 605 std::smatch match; 606 if (std::regex_search(file, match, spiPattern)) 607 { 608 spiBus = match.str(1); 609 } 610 else 611 { 612 std::cerr << "The given udev path < " << file 613 << " > doesn't match the required pattern. Skipping VPD " 614 "collection." 615 << std::endl; 616 exit(EXIT_SUCCESS); 617 } 618 // Forming the generic path 619 file = spiPathPrefix + driver + "/" + spiBus + ".0/eeprom"; 620 } 621 else 622 { 623 std::cerr << "\n The given EEPROM path < " << file 624 << " > is not valid. It's neither I2C nor " 625 "SPI path. Skipping VPD collection.." 626 << std::endl; 627 exit(EXIT_SUCCESS); 628 } 629 } 630 std::string getBadVpdName(const std::string& file) 631 { 632 std::string badVpd = BAD_VPD_DIR; 633 if (file.find("i2c") != std::string::npos) 634 { 635 badVpd += "i2c-"; 636 std::regex i2cPattern("(at24/)([0-9]+-[0-9]+)\\/"); 637 std::smatch match; 638 if (std::regex_search(file, match, i2cPattern)) 639 { 640 badVpd += match.str(2); 641 } 642 } 643 else if (file.find("spi") != std::string::npos) 644 { 645 std::regex spiPattern("((spi)[0-9]+)(.0)"); 646 std::smatch match; 647 if (std::regex_search(file, match, spiPattern)) 648 { 649 badVpd += match.str(1); 650 } 651 } 652 return badVpd; 653 } 654 655 void dumpBadVpd(const std::string& file, const Binary& vpdVector) 656 { 657 fs::path badVpdDir = BAD_VPD_DIR; 658 fs::create_directory(badVpdDir); 659 std::string badVpdPath = getBadVpdName(file); 660 if (fs::exists(badVpdPath)) 661 { 662 std::error_code ec; 663 fs::remove(badVpdPath, ec); 664 if (ec) // error code 665 { 666 std::string error = "Error removing the existing broken vpd in "; 667 error += badVpdPath; 668 error += ". Error code : "; 669 error += ec.value(); 670 error += ". Error message : "; 671 error += ec.message(); 672 throw std::runtime_error(error); 673 } 674 } 675 std::ofstream badVpdFileStream(badVpdPath, std::ofstream::binary); 676 if (!badVpdFileStream) 677 { 678 throw std::runtime_error( 679 "Failed to open bad vpd file path in /tmp/bad-vpd. " 680 "Unable to dump the broken/bad vpd file."); 681 } 682 badVpdFileStream.write(reinterpret_cast<const char*>(vpdVector.data()), 683 vpdVector.size()); 684 } 685 686 const std::string getKwVal(const Parsed& vpdMap, const std::string& rec, 687 const std::string& kwd) 688 { 689 std::string kwVal{}; 690 691 auto findRec = vpdMap.find(rec); 692 693 // check if record is found in map we got by parser 694 if (findRec != vpdMap.end()) 695 { 696 auto findKwd = findRec->second.find(kwd); 697 698 if (findKwd != findRec->second.end()) 699 { 700 kwVal = findKwd->second; 701 } 702 else 703 { 704 std::cout << "Keyword not found" << std::endl; 705 } 706 } 707 else 708 { 709 std::cerr << "Record not found" << std::endl; 710 } 711 712 return kwVal; 713 } 714 715 std::string hexString(const std::variant<Binary, std::string>& kw) 716 { 717 std::string hexString; 718 std::visit( 719 [&hexString](auto&& kw) { 720 std::stringstream ss; 721 std::string hexRep = "0x"; 722 ss << hexRep; 723 for (auto& kwVal : kw) 724 { 725 ss << std::setfill('0') << std::setw(2) << std::hex 726 << static_cast<int>(kwVal); 727 } 728 hexString = ss.str(); 729 }, 730 kw); 731 return hexString; 732 } 733 734 std::string getPrintableValue(const std::variant<Binary, std::string>& kwVal) 735 { 736 std::string kwString{}; 737 std::visit( 738 [&kwString](auto&& kwVal) { 739 const auto it = 740 std::find_if(kwVal.begin(), kwVal.end(), 741 [](const auto& kw) { return !isprint(kw); }); 742 if (it != kwVal.end()) 743 { 744 kwString = hexString(kwVal); 745 } 746 else 747 { 748 kwString = std::string(kwVal.begin(), kwVal.end()); 749 } 750 }, 751 kwVal); 752 return kwString; 753 } 754 755 void executePostFailAction(const nlohmann::json& json, const std::string& file) 756 { 757 if ((json["frus"][file].at(0)).find("postActionFail") == 758 json["frus"][file].at(0).end()) 759 { 760 return; 761 } 762 763 uint8_t pinValue = 0; 764 std::string pinName; 765 766 for (const auto& postAction : 767 (json["frus"][file].at(0))["postActionFail"].items()) 768 { 769 if (postAction.key() == "pin") 770 { 771 pinName = postAction.value(); 772 } 773 else if (postAction.key() == "value") 774 { 775 // Get the value to set 776 pinValue = postAction.value(); 777 } 778 } 779 780 std::cout << "Setting GPIO: " << pinName << " to " << (int)pinValue 781 << std::endl; 782 783 try 784 { 785 gpiod::line outputLine = gpiod::find_line(pinName); 786 787 if (!outputLine) 788 { 789 throw GpioException( 790 "Couldn't find output line for the GPIO. Skipping " 791 "this GPIO action."); 792 } 793 outputLine.request( 794 {"Disable line", ::gpiod::line_request::DIRECTION_OUTPUT, 0}, 795 pinValue); 796 } 797 catch (const std::exception& e) 798 { 799 std::string i2cBusAddr; 800 std::string errMsg = e.what(); 801 errMsg += "\nGPIO: " + pinName; 802 803 if ((json["frus"][file].at(0)["postActionFail"].find( 804 "gpioI2CAddress")) != 805 json["frus"][file].at(0)["postActionFail"].end()) 806 { 807 i2cBusAddr = 808 json["frus"][file].at(0)["postActionFail"]["gpioI2CAddress"]; 809 errMsg += " i2cBusAddress: " + i2cBusAddr; 810 } 811 812 throw GpioException(e.what()); 813 } 814 815 return; 816 } 817 818 std::optional<bool> isPresent(const nlohmann::json& json, 819 const std::string& file) 820 821 { 822 if ((json["frus"][file].at(0)).find("presence") != 823 json["frus"][file].at(0).end()) 824 { 825 if (((json["frus"][file].at(0)["presence"]).find("pin") != 826 json["frus"][file].at(0)["presence"].end()) && 827 ((json["frus"][file].at(0)["presence"]).find("value") != 828 json["frus"][file].at(0)["presence"].end())) 829 { 830 std::string presPinName = 831 json["frus"][file].at(0)["presence"]["pin"]; 832 Byte presPinValue = json["frus"][file].at(0)["presence"]["value"]; 833 834 try 835 { 836 gpiod::line presenceLine = gpiod::find_line(presPinName); 837 838 if (!presenceLine) 839 { 840 std::cerr << "Couldn't find the presence line for - " 841 << presPinName << std::endl; 842 843 throw GpioException( 844 "Couldn't find the presence line for the " 845 "GPIO. Skipping this GPIO action."); 846 } 847 848 presenceLine.request({"Read the presence line", 849 gpiod::line_request::DIRECTION_INPUT, 0}); 850 851 Byte gpioData = presenceLine.get_value(); 852 853 return (gpioData == presPinValue); 854 } 855 catch (const std::exception& e) 856 { 857 std::string i2cBusAddr; 858 std::string errMsg = e.what(); 859 errMsg += " GPIO : " + presPinName; 860 861 if ((json["frus"][file].at(0)["presence"]) 862 .find("gpioI2CAddress") != 863 json["frus"][file].at(0)["presence"].end()) 864 { 865 i2cBusAddr = 866 json["frus"][file].at(0)["presence"]["gpioI2CAddress"]; 867 errMsg += " i2cBusAddress: " + i2cBusAddr; 868 } 869 870 // Take failure postAction 871 executePostFailAction(json, file); 872 throw GpioException(errMsg); 873 } 874 } 875 else 876 { 877 // missing required information 878 std::cerr 879 << "VPD inventory JSON missing basic information of presence " 880 "for this FRU : [" 881 << file << "]. Executing executePostFailAction." << std::endl; 882 883 // Take failure postAction 884 executePostFailAction(json, file); 885 886 return false; 887 } 888 } 889 return std::optional<bool>{}; 890 } 891 892 bool executePreAction(const nlohmann::json& json, const std::string& file) 893 { 894 auto present = isPresent(json, file); 895 if (present && !present.value()) 896 { 897 executePostFailAction(json, file); 898 return false; 899 } 900 901 if ((json["frus"][file].at(0)).find("preAction") != 902 json["frus"][file].at(0).end()) 903 { 904 if (((json["frus"][file].at(0)["preAction"]).find("pin") != 905 json["frus"][file].at(0)["preAction"].end()) && 906 ((json["frus"][file].at(0)["preAction"]).find("value") != 907 json["frus"][file].at(0)["preAction"].end())) 908 { 909 std::string pinName = json["frus"][file].at(0)["preAction"]["pin"]; 910 // Get the value to set 911 Byte pinValue = json["frus"][file].at(0)["preAction"]["value"]; 912 913 std::cout << "Setting GPIO: " << pinName << " to " << (int)pinValue 914 << std::endl; 915 try 916 { 917 gpiod::line outputLine = gpiod::find_line(pinName); 918 919 if (!outputLine) 920 { 921 std::cerr << "Couldn't find the line for output pin - " 922 << pinName << std::endl; 923 throw GpioException( 924 "Couldn't find output line for the GPIO. " 925 "Skipping this GPIO action."); 926 } 927 outputLine.request({"FRU pre-action", 928 ::gpiod::line_request::DIRECTION_OUTPUT, 0}, 929 pinValue); 930 } 931 catch (const std::exception& e) 932 { 933 std::string i2cBusAddr; 934 std::string errMsg = e.what(); 935 errMsg += " GPIO : " + pinName; 936 937 if ((json["frus"][file].at(0)["preAction"]) 938 .find("gpioI2CAddress") != 939 json["frus"][file].at(0)["preAction"].end()) 940 { 941 i2cBusAddr = 942 json["frus"][file].at(0)["preAction"]["gpioI2CAddress"]; 943 errMsg += " i2cBusAddress: " + i2cBusAddr; 944 } 945 946 // Take failure postAction 947 executePostFailAction(json, file); 948 throw GpioException(errMsg); 949 } 950 } 951 else 952 { 953 // missing required information 954 std::cerr 955 << "VPD inventory JSON missing basic information of preAction " 956 "for this FRU : [" 957 << file << "]. Executing executePostFailAction." << std::endl; 958 959 // Take failure postAction 960 executePostFailAction(json, file); 961 return false; 962 } 963 } 964 return true; 965 } 966 967 void insertOrMerge(inventory::InterfaceMap& map, 968 const inventory::Interface& interface, 969 inventory::PropertyMap&& property) 970 { 971 if (map.find(interface) != map.end()) 972 { 973 auto& prop = map.at(interface); 974 prop.insert(property.begin(), property.end()); 975 } 976 else 977 { 978 map.emplace(interface, property); 979 } 980 } 981 982 BIOSAttrValueType readBIOSAttribute(const std::string& attrName) 983 { 984 std::tuple<std::string, BIOSAttrValueType, BIOSAttrValueType> attrVal; 985 auto bus = sdbusplus::bus::new_default(); 986 auto method = bus.new_method_call( 987 "xyz.openbmc_project.BIOSConfigManager", 988 "/xyz/openbmc_project/bios_config/manager", 989 "xyz.openbmc_project.BIOSConfig.Manager", "GetAttribute"); 990 method.append(attrName); 991 try 992 { 993 auto result = bus.call(method); 994 result.read(std::get<0>(attrVal), std::get<1>(attrVal), 995 std::get<2>(attrVal)); 996 } 997 catch (const sdbusplus::exception::SdBusError& e) 998 { 999 std::cerr << "Failed to read BIOS Attribute: " << attrName << std::endl; 1000 std::cerr << e.what() << std::endl; 1001 } 1002 return std::get<1>(attrVal); 1003 } 1004 1005 std::string getPowerState() 1006 { 1007 // TODO: How do we handle multiple chassis? 1008 std::string powerState{}; 1009 auto bus = sdbusplus::bus::new_default(); 1010 auto properties = 1011 bus.new_method_call("xyz.openbmc_project.State.Chassis0", 1012 "/xyz/openbmc_project/state/chassis0", 1013 "org.freedesktop.DBus.Properties", "Get"); 1014 properties.append("xyz.openbmc_project.State.Chassis"); 1015 properties.append("CurrentPowerState"); 1016 auto result = bus.call(properties); 1017 if (!result.is_method_error()) 1018 { 1019 std::variant<std::string> val; 1020 result.read(val); 1021 if (auto pVal = std::get_if<std::string>(&val)) 1022 { 1023 powerState = *pVal; 1024 } 1025 } 1026 std::cout << "Power state is: " << powerState << std::endl; 1027 return powerState; 1028 } 1029 1030 Binary getVpdDataInVector(const nlohmann::json& js, const std::string& file) 1031 { 1032 uint32_t offset = 0; 1033 // check if offset present? 1034 for (const auto& item : js["frus"][file]) 1035 { 1036 if (item.find("offset") != item.end()) 1037 { 1038 offset = item["offset"]; 1039 } 1040 } 1041 1042 // TODO: Figure out a better way to get max possible VPD size. 1043 auto maxVPDSize = std::min(std::filesystem::file_size(file), 1044 static_cast<uintmax_t>(65504)); 1045 1046 Binary vpdVector; 1047 vpdVector.resize(maxVPDSize); 1048 std::ifstream vpdFile; 1049 vpdFile.exceptions(std::ifstream::badbit | std::ifstream::failbit); 1050 try 1051 { 1052 vpdFile.open(file, std::ios::binary | std::ios::in); 1053 vpdFile.seekg(offset, std::ios_base::cur); 1054 vpdFile.read(reinterpret_cast<char*>(&vpdVector[0]), maxVPDSize); 1055 vpdVector.resize(vpdFile.gcount()); 1056 } 1057 catch (const std::ifstream::failure& fail) 1058 { 1059 std::cerr << "Exception in file handling [" << file 1060 << "] error : " << fail.what(); 1061 std::cerr << "EEPROM file size =" << std::filesystem::file_size(file) 1062 << std::endl; 1063 std::cerr << "Stream file size = " << vpdFile.gcount() << std::endl; 1064 std::cerr << " Vector size" << vpdVector.size() << std::endl; 1065 throw; 1066 } 1067 1068 // Make sure we reset the EEPROM pointer to a "safe" location if it was 1069 // a DDIMM SPD that we just read. 1070 for (const auto& item : js["frus"][file]) 1071 { 1072 if (item.find("extraInterfaces") != item.end()) 1073 { 1074 if (item["extraInterfaces"].find( 1075 "xyz.openbmc_project.Inventory.Item.Dimm") != 1076 item["extraInterfaces"].end()) 1077 { 1078 // check added here for DDIMM only workaround 1079 vpdType dimmType = vpdTypeCheck(vpdVector); 1080 if (dimmType == constants::DDR4_DDIMM_MEMORY_VPD || 1081 dimmType == constants::DDR5_DDIMM_MEMORY_VPD) 1082 { 1083 try 1084 { 1085 // moves the EEPROM pointer to 2048 'th byte. 1086 vpdFile.seekg(2047, std::ios::beg); 1087 // Read that byte and discard - to affirm the move 1088 // operation. 1089 char ch; 1090 vpdFile.read(&ch, sizeof(ch)); 1091 } 1092 catch (const std::ifstream::failure& fail) 1093 { 1094 std::cerr << "Exception in file handling [" << file 1095 << "] error : " << fail.what(); 1096 std::cerr << "Stream file size = " << vpdFile.gcount() 1097 << std::endl; 1098 throw; 1099 } 1100 } 1101 break; 1102 } 1103 } 1104 } 1105 1106 return vpdVector; 1107 } 1108 1109 std::string getDbusNameForThisKw(const std::string& keyword) 1110 { 1111 if (keyword[0] == constants::POUND_KW) 1112 { 1113 return (std::string(constants::POUND_KW_PREFIX) + keyword[1]); 1114 } 1115 else if (isdigit(keyword[0])) 1116 { 1117 return (std::string(constants::NUMERIC_KW_PREFIX) + keyword); 1118 } 1119 return keyword; 1120 } 1121 1122 void clearVpdOnRemoval(const std::string& objPath, 1123 inventory::InterfaceMap& interfacesPropMap) 1124 { 1125 std::vector<std::string> vpdRelatedInterfaces{ 1126 constants::invOperationalStatusIntf, constants::invItemIntf, 1127 constants::invAssetIntf}; 1128 1129 std::vector<std::string> interfaces{}; 1130 auto mapperResponse = inventory::getObject(objPath, interfaces); 1131 1132 for (const auto& [service, interfaceList] : mapperResponse) 1133 { 1134 // Handle FRUs under PIM 1135 if (service.compare(pimService) != 0) 1136 { 1137 continue; 1138 } 1139 1140 for (const auto& interface : interfaceList) 1141 { 1142 // Only process for VPD related interfaces. 1143 if ((interface.find("com.ibm.ipzvpd") != std::string::npos) || 1144 ((std::find(vpdRelatedInterfaces.begin(), 1145 vpdRelatedInterfaces.end(), interface)) != 1146 vpdRelatedInterfaces.end())) 1147 { 1148 const auto propertyList = getAllDBusProperty<GetAllResultType>( 1149 service, objPath, interface); 1150 1151 inventory::PropertyMap propertyValueMap; 1152 for (auto aProperty : propertyList) 1153 { 1154 const auto& propertyName = std::get<0>(aProperty); 1155 const auto& propertyValue = std::get<1>(aProperty); 1156 1157 if (std::holds_alternative<Binary>(propertyValue)) 1158 { 1159 propertyValueMap.emplace(propertyName, Binary{}); 1160 } 1161 else if (std::holds_alternative<std::string>(propertyValue)) 1162 { 1163 propertyValueMap.emplace(propertyName, std::string{}); 1164 } 1165 else if (std::holds_alternative<bool>(propertyValue)) 1166 { 1167 if (propertyName.compare("Present") == 0) 1168 { 1169 propertyValueMap.emplace(propertyName, false); 1170 } 1171 else if (propertyName.compare("Functional") == 0) 1172 { 1173 propertyValueMap.emplace(propertyName, true); 1174 } 1175 } 1176 } 1177 interfacesPropMap.emplace(interface, 1178 std::move(propertyValueMap)); 1179 } 1180 } 1181 } 1182 } 1183 1184 void findBackupVPDPaths(std::string& backupEepromPath, 1185 std::string& backupInvPath, const nlohmann::json& js) 1186 { 1187 for (const auto& item : js["frus"][constants::systemVpdFilePath]) 1188 { 1189 if (item.find("systemVpdBackupPath") != item.end()) 1190 { 1191 backupEepromPath = item["systemVpdBackupPath"]; 1192 for (const auto& item : js["frus"][backupEepromPath]) 1193 { 1194 if (item.find("inventoryPath") != item.end()) 1195 { 1196 backupInvPath = item["inventoryPath"]; 1197 break; 1198 } 1199 } 1200 break; 1201 } 1202 } 1203 } 1204 1205 void getBackupRecordKeyword(std::string& record, std::string& keyword) 1206 { 1207 for (const auto& recordKw : svpdKwdMap) 1208 { 1209 if (record == recordKw.first) 1210 { 1211 for (const auto& keywordInfo : recordKw.second) 1212 { 1213 if (keyword == get<0>(keywordInfo)) 1214 { 1215 record = get<4>(keywordInfo); 1216 keyword = get<5>(keywordInfo); 1217 break; 1218 } 1219 } 1220 break; 1221 } 1222 } 1223 } 1224 1225 bool isReadOnlyEEPROM(const std::string& vpdPath, 1226 const nlohmann::json& jsObject) 1227 { 1228 // check if given path is FRU path 1229 if (jsObject["frus"].contains(vpdPath)) 1230 { 1231 return jsObject["frus"][vpdPath].at(0).value("readOnly", false); 1232 } 1233 1234 const nlohmann::json& fruList = 1235 jsObject["frus"].get_ref<const nlohmann::json::object_t&>(); 1236 1237 for (const auto& fru : fruList.items()) 1238 { 1239 const auto fruPath = fru.key(); 1240 1241 // If given VPD path is either the inventory path or redundant EEPROM 1242 // path. 1243 if ((vpdPath == 1244 jsObject["frus"][fruPath].at(0).value("inventoryPath", "")) || 1245 (vpdPath == 1246 jsObject["frus"][fruPath].at(0).value("redundantEeprom", ""))) 1247 { 1248 return jsObject["frus"][fruPath].at(0).value("readOnly", false); 1249 } 1250 } 1251 1252 // Given path not found in JSON 1253 return false; 1254 } 1255 } // namespace vpd 1256 } // namespace openpower 1257