xref: /openbmc/openpower-vpd-parser/vpd_tool_impl.cpp (revision e008432ac875223b916a8c63cc951caec090364e)
1 #include "vpd_tool_impl.hpp"
2 
3 #include "impl.hpp"
4 #include "parser_factory.hpp"
5 #include "vpd_exceptions.hpp"
6 
7 #include <cstdlib>
8 #include <filesystem>
9 #include <iostream>
10 #include <sdbusplus/bus.hpp>
11 #include <variant>
12 #include <vector>
13 
14 using namespace std;
15 using namespace openpower::vpd;
16 using namespace inventory;
17 using namespace openpower::vpd::manager::editor;
18 namespace fs = std::filesystem;
19 using json = nlohmann::json;
20 using namespace openpower::vpd::exceptions;
21 using namespace openpower::vpd::parser;
22 using namespace openpower::vpd::parser::factory;
23 using namespace openpower::vpd::parser::interface;
24 
25 Binary VpdTool::toBinary(const std::string& value)
26 {
27     Binary val{};
28     if (value.find("0x") == string::npos)
29     {
30         val.assign(value.begin(), value.end());
31     }
32     else if (value.find("0x") != string::npos)
33     {
34         stringstream ss;
35         ss.str(value.substr(2));
36         string byteStr{};
37 
38         if (value.length() % 2 != 0)
39         {
40             throw runtime_error(
41                 "VPD-TOOL write option accepts 2 digit hex numbers. (Eg. 0x1 "
42                 "should be given as 0x01). Aborting the write operation.");
43         }
44 
45         if (value.find_first_not_of("0123456789abcdefABCDEF", 2) !=
46             std::string::npos)
47         {
48             throw runtime_error("Provide a valid hexadecimal input.");
49         }
50 
51         while (ss >> setw(2) >> byteStr)
52         {
53             uint8_t byte = strtoul(byteStr.c_str(), nullptr, 16);
54 
55             val.push_back(byte);
56         }
57     }
58 
59     else
60     {
61         throw runtime_error("The value to be updated should be either in ascii "
62                             "or in hex. Refer --help option");
63     }
64     return val;
65 }
66 
67 void VpdTool::printReturnCode(int returnCode)
68 {
69     if (returnCode)
70     {
71         cout << "\n Command failed with the return code " << returnCode
72              << ". Continuing the execution. " << endl;
73     }
74 }
75 
76 void VpdTool::eraseInventoryPath(string& fru)
77 {
78     // Power supply frupath comes with INVENTORY_PATH appended in prefix.
79     // Stripping it off inorder to avoid INVENTORY_PATH duplication
80     // during getVINIProperties() execution.
81     fru.erase(0, sizeof(INVENTORY_PATH) - 1);
82 }
83 
84 void VpdTool::debugger(json output)
85 {
86     cout << output.dump(4) << '\n';
87 }
88 
89 auto VpdTool::makeDBusCall(const string& objectName, const string& interface,
90                            const string& kw)
91 {
92     auto bus = sdbusplus::bus::new_default();
93     auto properties =
94         bus.new_method_call(INVENTORY_MANAGER_SERVICE, objectName.c_str(),
95                             "org.freedesktop.DBus.Properties", "Get");
96     properties.append(interface);
97     properties.append(kw);
98     auto result = bus.call(properties);
99 
100     if (result.is_method_error())
101     {
102         throw runtime_error("Get api failed");
103     }
104     return result;
105 }
106 
107 json VpdTool::getVINIProperties(string invPath)
108 {
109     variant<Binary> response;
110     json kwVal = json::object({});
111 
112     vector<string> keyword{"CC", "SN", "PN", "FN", "DR"};
113     string interface = "com.ibm.ipzvpd.VINI";
114     string objectName = {};
115 
116     if (invPath.find(INVENTORY_PATH) != string::npos)
117     {
118         objectName = invPath;
119         eraseInventoryPath(invPath);
120     }
121     else
122     {
123         objectName = INVENTORY_PATH + invPath;
124     }
125     for (string kw : keyword)
126     {
127         try
128         {
129             makeDBusCall(objectName, interface, kw).read(response);
130 
131             if (auto vec = get_if<Binary>(&response))
132             {
133                 string printableVal = getPrintableValue(*vec);
134                 kwVal.emplace(kw, printableVal);
135             }
136         }
137         catch (const sdbusplus::exception_t& e)
138         {
139             if (string(e.name()) ==
140                 string("org.freedesktop.DBus.Error.UnknownObject"))
141             {
142                 kwVal.emplace(invPath, json::object({}));
143                 objFound = false;
144                 break;
145             }
146         }
147     }
148 
149     return kwVal;
150 }
151 
152 void VpdTool::getExtraInterfaceProperties(const string& invPath,
153                                           const string& extraInterface,
154                                           const json& prop, json& output)
155 {
156     variant<string> response;
157 
158     string objectName = INVENTORY_PATH + invPath;
159 
160     for (const auto& itProp : prop.items())
161     {
162         string kw = itProp.key();
163         try
164         {
165             makeDBusCall(objectName, extraInterface, kw).read(response);
166 
167             if (auto str = get_if<string>(&response))
168             {
169                 output.emplace(kw, *str);
170             }
171         }
172         catch (const sdbusplus::exception_t& e)
173         {
174             if (std::string(e.name()) ==
175                 std::string("org.freedesktop.DBus.Error.UnknownObject"))
176             {
177                 objFound = false;
178                 break;
179             }
180             else if (std::string(e.name()) ==
181                      std::string("org.freedesktop.DBus.Error.UnknownProperty"))
182             {
183                 output.emplace(kw, "");
184             }
185         }
186     }
187 }
188 
189 json VpdTool::interfaceDecider(json& itemEEPROM)
190 {
191     if (itemEEPROM.find("inventoryPath") == itemEEPROM.end())
192     {
193         throw runtime_error("Inventory Path not found");
194     }
195 
196     if (itemEEPROM.find("extraInterfaces") == itemEEPROM.end())
197     {
198         throw runtime_error("Extra Interfaces not found");
199     }
200 
201     json subOutput = json::object({});
202     fruType = "FRU";
203 
204     json j;
205     objFound = true;
206     string invPath = itemEEPROM.at("inventoryPath");
207 
208     j = getVINIProperties(invPath);
209 
210     if (objFound)
211     {
212         subOutput.insert(j.begin(), j.end());
213         json js;
214         if (itemEEPROM.find("type") != itemEEPROM.end())
215         {
216             fruType = itemEEPROM.at("type");
217         }
218         js.emplace("TYPE", fruType);
219 
220         if (invPath.find("powersupply") != string::npos)
221         {
222             js.emplace("type", POWER_SUPPLY_TYPE_INTERFACE);
223         }
224         else if (invPath.find("fan") != string::npos)
225         {
226             js.emplace("type", FAN_INTERFACE);
227         }
228 
229         for (const auto& ex : itemEEPROM["extraInterfaces"].items())
230         {
231             if (!(ex.value().is_null()))
232             {
233                 // TODO: Remove this if condition check once inventory json is
234                 // updated with xyz location code interface.
235                 if (ex.key() == "com.ibm.ipzvpd.Location")
236                 {
237                     getExtraInterfaceProperties(
238                         invPath,
239                         "xyz.openbmc_project.Inventory.Decorator.LocationCode",
240                         ex.value(), js);
241                 }
242                 else
243                 {
244                     getExtraInterfaceProperties(invPath, ex.key(), ex.value(),
245                                                 js);
246                 }
247             }
248             if ((ex.key().find("Item") != string::npos) &&
249                 (ex.value().is_null()))
250             {
251                 js.emplace("type", ex.key());
252             }
253             subOutput.insert(js.begin(), js.end());
254         }
255     }
256     return subOutput;
257 }
258 
259 json VpdTool::getPresentPropJson(const std::string& invPath,
260                                  std::string& parentPresence)
261 {
262     std::variant<bool> response;
263     makeDBusCall(invPath, "xyz.openbmc_project.Inventory.Item", "Present")
264         .read(response);
265 
266     std::string presence{};
267 
268     if (auto pVal = get_if<bool>(&response))
269     {
270         presence = *pVal ? "true" : "false";
271         if (parentPresence.empty())
272         {
273             parentPresence = presence;
274         }
275     }
276     else
277     {
278         presence = parentPresence;
279     }
280 
281     json js;
282     js.emplace("Present", presence);
283     return js;
284 }
285 
286 json VpdTool::parseInvJson(const json& jsObject, char flag, string fruPath)
287 {
288     json output = json::object({});
289     bool validObject = false;
290 
291     if (jsObject.find("frus") == jsObject.end())
292     {
293         throw runtime_error("Frus missing in Inventory json");
294     }
295     else
296     {
297         for (const auto& itemFRUS : jsObject["frus"].items())
298         {
299             string parentPresence{};
300             for (auto itemEEPROM : itemFRUS.value())
301             {
302                 json subOutput = json::object({});
303                 try
304                 {
305                     if (flag == 'O')
306                     {
307                         if (itemEEPROM.find("inventoryPath") ==
308                             itemEEPROM.end())
309                         {
310                             throw runtime_error("Inventory Path not found");
311                         }
312                         else if (itemEEPROM.at("inventoryPath") == fruPath)
313                         {
314                             validObject = true;
315                             subOutput = interfaceDecider(itemEEPROM);
316                             json presentJs = getPresentPropJson(
317                                 "/xyz/openbmc_project/inventory" + fruPath,
318                                 parentPresence);
319                             subOutput.insert(presentJs.begin(),
320                                              presentJs.end());
321                             output.emplace(fruPath, subOutput);
322                             return output;
323                         }
324                         else // this else is to keep track of parent present
325                              // property.
326                         {
327                             json presentJs = getPresentPropJson(
328                                 "/xyz/openbmc_project/inventory" +
329                                     string(itemEEPROM.at("inventoryPath")),
330                                 parentPresence);
331                         }
332                     }
333                     else
334                     {
335                         subOutput = interfaceDecider(itemEEPROM);
336                         json presentJs = getPresentPropJson(
337                             "/xyz/openbmc_project/inventory" +
338                                 string(itemEEPROM.at("inventoryPath")),
339                             parentPresence);
340                         subOutput.insert(presentJs.begin(), presentJs.end());
341                         output.emplace(string(itemEEPROM.at("inventoryPath")),
342                                        subOutput);
343                     }
344                 }
345                 catch (const sdbusplus::exception::SdBusError& e)
346                 {
347                     // if any of frupath doesn't have Present property of its
348                     // own, emplace its parent's present property value.
349                     if (e.name() == std::string("org.freedesktop.DBus.Error."
350                                                 "UnknownProperty") &&
351                         (((flag == 'O') && validObject) || flag == 'I'))
352                     {
353                         json presentJs;
354                         presentJs.emplace("Present", parentPresence);
355                         subOutput.insert(presentJs.begin(), presentJs.end());
356                         output.emplace(string(itemEEPROM.at("inventoryPath")),
357                                        subOutput);
358                     }
359 
360                     // for the user given child frupath which doesn't have
361                     // Present prop (vpd-tool -o).
362                     if ((flag == 'O') && validObject)
363                     {
364                         return output;
365                     }
366                 }
367                 catch (const exception& e)
368                 {
369                     cerr << e.what();
370                 }
371             }
372         }
373         if ((flag == 'O') && (!validObject))
374         {
375             throw runtime_error(
376                 "Invalid object path. Refer --dumpInventory/-i option.");
377         }
378     }
379     return output;
380 }
381 
382 void VpdTool::dumpInventory(const nlohmann::basic_json<>& jsObject)
383 {
384     char flag = 'I';
385     json output = json::array({});
386     output.emplace_back(parseInvJson(jsObject, flag, ""));
387     debugger(output);
388 }
389 
390 void VpdTool::dumpObject(const nlohmann::basic_json<>& jsObject)
391 {
392     char flag = 'O';
393     json output = json::array({});
394     output.emplace_back(parseInvJson(jsObject, flag, fruPath));
395     debugger(output);
396 }
397 
398 void VpdTool::readKeyword()
399 {
400     string interface = "com.ibm.ipzvpd.";
401     variant<Binary> response;
402 
403     try
404     {
405         json output = json::object({});
406         json kwVal = json::object({});
407         makeDBusCall(INVENTORY_PATH + fruPath, interface + recordName, keyword)
408             .read(response);
409 
410         string printableVal{};
411         if (auto vec = get_if<Binary>(&response))
412         {
413             printableVal = getPrintableValue(*vec);
414         }
415         kwVal.emplace(keyword, printableVal);
416 
417         output.emplace(fruPath, kwVal);
418 
419         debugger(output);
420     }
421     catch (const json::exception& e)
422     {
423         json output = json::object({});
424         json kwVal = json::object({});
425     }
426 }
427 
428 int VpdTool::updateKeyword()
429 {
430     Binary val = toBinary(value);
431     auto bus = sdbusplus::bus::new_default();
432     auto properties =
433         bus.new_method_call(BUSNAME, OBJPATH, IFACE, "WriteKeyword");
434     properties.append(static_cast<sdbusplus::message::object_path>(fruPath));
435     properties.append(recordName);
436     properties.append(keyword);
437     properties.append(val);
438     auto result = bus.call(properties);
439 
440     if (result.is_method_error())
441     {
442         throw runtime_error("Get api failed");
443     }
444     return 0;
445 }
446 
447 void VpdTool::forceReset(const nlohmann::basic_json<>& jsObject)
448 {
449     for (const auto& itemFRUS : jsObject["frus"].items())
450     {
451         for (const auto& itemEEPROM : itemFRUS.value().items())
452         {
453             string fru = itemEEPROM.value().at("inventoryPath");
454 
455             fs::path fruCachePath = INVENTORY_MANAGER_CACHE;
456             fruCachePath += INVENTORY_PATH;
457             fruCachePath += fru;
458 
459             try
460             {
461                 for (const auto& it : fs::directory_iterator(fruCachePath))
462                 {
463                     if (fs::is_regular_file(it.status()))
464                     {
465                         fs::remove(it);
466                     }
467                 }
468             }
469             catch (const fs::filesystem_error& e)
470             {
471             }
472         }
473     }
474 
475     cout.flush();
476     string udevRemove = "udevadm trigger -c remove -s \"*nvmem*\" -v";
477     int returnCode = system(udevRemove.c_str());
478     printReturnCode(returnCode);
479 
480     string invManagerRestart =
481         "systemctl restart xyz.openbmc_project.Inventory.Manager.service";
482     returnCode = system(invManagerRestart.c_str());
483     printReturnCode(returnCode);
484 
485     string sysVpdRestart = "systemctl restart system-vpd.service";
486     returnCode = system(sysVpdRestart.c_str());
487     printReturnCode(returnCode);
488 
489     string udevAdd = "udevadm trigger -c add -s \"*nvmem*\" -v";
490     returnCode = system(udevAdd.c_str());
491     printReturnCode(returnCode);
492 }
493 
494 int VpdTool::updateHardware(const uint32_t offset)
495 {
496     int rc = 0;
497     const Binary& val = static_cast<const Binary&>(toBinary(value));
498     ifstream inventoryJson(INVENTORY_JSON_SYM_LINK);
499     try
500     {
501         auto json = nlohmann::json::parse(inventoryJson);
502         EditorImpl edit(fruPath, json, recordName, keyword);
503 
504         edit.updateKeyword(val, offset, false);
505     }
506     catch (const json::parse_error& ex)
507     {
508         throw(VpdJsonException("Json Parsing failed", INVENTORY_JSON_SYM_LINK));
509     }
510     return rc;
511 }
512 
513 void VpdTool::readKwFromHw(const uint32_t& startOffset)
514 {
515     ifstream inventoryJson(INVENTORY_JSON_SYM_LINK);
516     auto jsonFile = nlohmann::json::parse(inventoryJson);
517 
518     Binary completeVPDFile;
519     completeVPDFile.resize(65504);
520     fstream vpdFileStream;
521     vpdFileStream.open(fruPath,
522                        std::ios::in | std::ios::out | std::ios::binary);
523 
524     vpdFileStream.seekg(startOffset, ios_base::cur);
525     vpdFileStream.read(reinterpret_cast<char*>(&completeVPDFile[0]), 65504);
526     completeVPDFile.resize(vpdFileStream.gcount());
527     vpdFileStream.clear(std::ios_base::eofbit);
528 
529     if (completeVPDFile.empty())
530     {
531         throw std::runtime_error("Invalid File");
532     }
533     Impl obj(completeVPDFile);
534     std::string keywordVal = obj.readKwFromHw(recordName, keyword);
535 
536     if (!keywordVal.empty())
537     {
538         json output = json::object({});
539         json kwVal = json::object({});
540         kwVal.emplace(keyword, keywordVal);
541 
542         output.emplace(fruPath, kwVal);
543 
544         debugger(output);
545     }
546     else
547     {
548         std::cerr << "The given keyword " << keyword << " or record "
549                   << recordName
550                   << " or both are not present in the given FRU path "
551                   << fruPath << std::endl;
552     }
553 }
554 
555 void VpdTool::printFixSystemVPDOption(UserOption option)
556 {
557     switch (option)
558     {
559         case VpdTool::EXIT:
560             cout << "\nEnter 0 => To exit successfully : ";
561             break;
562         case VpdTool::BMC_DATA_FOR_ALL:
563             cout << "\n\nEnter 1 => If you choose the data on BMC for all "
564                     "mismatching record-keyword pairs";
565             break;
566         case VpdTool::SYSTEM_BACKPLANE_DATA_FOR_ALL:
567             cout << "\nEnter 2 => If you choose the data on System "
568                     "Backplane for all mismatching record-keyword pairs";
569             break;
570         case VpdTool::MORE_OPTIONS:
571             cout << "\nEnter 3 => If you wish to explore more options";
572             break;
573         case VpdTool::BMC_DATA_FOR_CURRENT:
574             cout << "\nEnter 4 => If you choose the data on BMC as the "
575                     "right value";
576             break;
577         case VpdTool::SYSTEM_BACKPLANE_DATA_FOR_CURRENT:
578             cout << "\nEnter 5 => If you choose the data on System "
579                     "Backplane as the right value";
580             break;
581         case VpdTool::NEW_VALUE_ON_BOTH:
582             cout << "\nEnter 6 => If you wish to enter a new value to "
583                     "update both on BMC and System Backplane";
584             break;
585         case VpdTool::SKIP_CURRENT:
586             cout << "\nEnter 7 => If you wish to skip the above "
587                     "record-keyword pair";
588             break;
589     }
590 }
591 
592 int VpdTool::fixSystemVPD()
593 {
594     cout << "\nRestorable record-keyword pairs and their data on BMC & "
595             "System Backplane.\n ";
596 
597     cout << "\n|==============================================================="
598             "=======================================|"
599          << endl;
600 
601     cout << left << setw(6) << "S.No" << left << setw(8) << "Record" << left
602          << setw(9) << "Keyword" << left << setw(30) << "Data On BMC" << left
603          << setw(30) << "Data On System Backplane" << left << setw(14)
604          << "Data Mismatch";
605 
606     cout << "\n|==============================================================="
607             "=======================================|"
608          << endl;
609 
610     int num = 0;
611 
612     // Get system VPD data in map
613     Binary vpdVector{};
614     json js{};
615 
616     auto jsonToParse = INVENTORY_JSON_DEFAULT;
617     if (fs::exists(INVENTORY_JSON_SYM_LINK))
618     {
619         jsonToParse = INVENTORY_JSON_SYM_LINK;
620     }
621 
622     ifstream inventoryJson(jsonToParse);
623     if (!inventoryJson)
624     {
625         throw runtime_error("VPD JSON file not found");
626     }
627     js = json::parse(inventoryJson);
628 
629     vpdVector = getVpdDataInVector(js, constants::systemVpdFilePath);
630     ParserInterface* parser = ParserFactory::getParser(vpdVector);
631     auto parseResult = parser->parse();
632     ParserFactory::freeParser(parser);
633 
634     unordered_map<string, DbusPropertyMap> vpdMap;
635 
636     if (auto pVal = get_if<Store>(&parseResult))
637     {
638         vpdMap = pVal->getVpdMap();
639     }
640     else
641     {
642         std::cerr << "\n System backplane VPD is not of type IPZ. Unable to "
643                      "parse the VPD "
644                   << constants::systemVpdFilePath << " . Exit with error."
645                   << std::endl;
646         exit(1);
647     }
648 
649     // Get system VPD D-Bus Data and store it in a map
650     using GetAllResultType =
651         std::vector<std::pair<std::string, std::variant<Binary>>>;
652     using IntfPropMap = std::map<std::string, GetAllResultType>;
653 
654     IntfPropMap svpdBusData;
655 
656     const auto vsys = getAllDBusProperty<GetAllResultType>(
657         constants::pimIntf,
658         "/xyz/openbmc_project/inventory/system/chassis/motherboard",
659         "com.ibm.ipzvpd.VSYS");
660     svpdBusData.emplace("VSYS", vsys);
661 
662     const auto vcen = getAllDBusProperty<GetAllResultType>(
663         constants::pimIntf,
664         "/xyz/openbmc_project/inventory/system/chassis/motherboard",
665         "com.ibm.ipzvpd.VCEN");
666     svpdBusData.emplace("VCEN", vcen);
667 
668     const auto lxr0 = getAllDBusProperty<GetAllResultType>(
669         constants::pimIntf,
670         "/xyz/openbmc_project/inventory/system/chassis/motherboard",
671         "com.ibm.ipzvpd.LXR0");
672     svpdBusData.emplace("LXR0", lxr0);
673 
674     const auto util = getAllDBusProperty<GetAllResultType>(
675         constants::pimIntf,
676         "/xyz/openbmc_project/inventory/system/chassis/motherboard",
677         "com.ibm.ipzvpd.UTIL");
678     svpdBusData.emplace("UTIL", util);
679 
680     for (const auto& recordKw : svpdKwdMap)
681     {
682         string record = recordKw.first;
683 
684         // Extract specific record data from the svpdBusData map.
685         const auto& rec = svpdBusData.find(record);
686 
687         if (rec == svpdBusData.end())
688         {
689             std::cerr << record << " not a part of critical system VPD records."
690                       << std::endl;
691             continue;
692         }
693 
694         const auto& recData = svpdBusData.find(record)->second;
695 
696         string busStr{}, hwValStr{};
697         string mismatch = "NO"; // no mismatch
698 
699         for (const auto& keyword : recordKw.second)
700         {
701             string hardwareValue{};
702             auto recItr = vpdMap.find(record);
703 
704             if (recItr != vpdMap.end())
705             {
706                 DbusPropertyMap& kwValMap = recItr->second;
707                 auto kwItr = kwValMap.find(keyword);
708                 if (kwItr != kwValMap.end())
709                 {
710                     hardwareValue = kwItr->second;
711                 }
712             }
713 
714             std::variant<Binary> kwValue;
715             for (auto& kwData : recData)
716             {
717                 if (kwData.first == keyword)
718                 {
719                     kwValue = kwData.second;
720                     break;
721                 }
722             }
723 
724             if (keyword != "SE")
725             {
726                 ostringstream hwValStream;
727                 hwValStream << "0x";
728                 hwValStr = hwValStream.str();
729 
730                 for (uint16_t byte : hardwareValue)
731                 {
732                     hwValStream << setfill('0') << setw(2) << hex << byte;
733                     hwValStr = hwValStream.str();
734                 }
735 
736                 if (const auto value = get_if<Binary>(&kwValue))
737                 {
738                     busStr = byteArrayToHexString(*value);
739                 }
740                 if (busStr != hwValStr)
741                 {
742                     mismatch = "YES";
743                 }
744             }
745             else
746             {
747                 if (const auto value = get_if<Binary>(&kwValue))
748                 {
749                     busStr = getPrintableValue(*value);
750                 }
751                 if (busStr != hardwareValue)
752                 {
753                     mismatch = "YES";
754                 }
755                 hwValStr = hardwareValue;
756             }
757             recKwData.push_back(
758                 make_tuple(++num, record, keyword, busStr, hwValStr, mismatch));
759             cout << left << setw(6) << num << left << setw(8) << record << left
760                  << setw(9) << keyword << left << setw(30) << setfill(' ')
761                  << busStr << left << setw(30) << setfill(' ') << hwValStr
762                  << left << setw(14) << mismatch;
763 
764             cout << "\n|--------------------------------------------------"
765                     "----------------------------------------------------|"
766                  << endl;
767         }
768     }
769     parseSVPDOptions(js);
770     return 0;
771 }
772 
773 void VpdTool::parseSVPDOptions(const nlohmann::json& json)
774 {
775     do
776     {
777         printFixSystemVPDOption(VpdTool::BMC_DATA_FOR_ALL);
778         printFixSystemVPDOption(VpdTool::SYSTEM_BACKPLANE_DATA_FOR_ALL);
779         printFixSystemVPDOption(VpdTool::MORE_OPTIONS);
780         printFixSystemVPDOption(VpdTool::EXIT);
781 
782         int option = 0;
783         cin >> option;
784         cout << "\n|==========================================================="
785                 "===========================================|"
786              << endl;
787 
788         if (json.find("frus") == json.end())
789         {
790             throw runtime_error("Frus not found in json");
791         }
792 
793         bool mismatchFound = false;
794 
795         if (option == VpdTool::BMC_DATA_FOR_ALL)
796         {
797             for (const auto& data : recKwData)
798             {
799                 if (get<5>(data) == "YES")
800                 {
801                     EditorImpl edit(constants::systemVpdFilePath, json,
802                                     get<1>(data), get<2>(data));
803                     edit.updateKeyword(toBinary(get<3>(data)), 0, true);
804                     mismatchFound = true;
805                 }
806             }
807 
808             if (mismatchFound)
809             {
810                 cout << "\nData updated successfully for all mismatching "
811                         "record-keyword pairs by choosing their corresponding "
812                         "data on BMC. Exit successfully.\n"
813                      << endl;
814             }
815             else
816             {
817                 cout << "\nNo mismatch found for any of the above mentioned "
818                         "record-keyword pair. Exit successfully.\n";
819             }
820 
821             exit(0);
822         }
823         else if (option == VpdTool::SYSTEM_BACKPLANE_DATA_FOR_ALL)
824         {
825             for (const auto& data : recKwData)
826             {
827                 if (get<5>(data) == "YES")
828                 {
829                     EditorImpl edit(constants::systemVpdFilePath, json,
830                                     get<1>(data), get<2>(data));
831                     edit.updateKeyword(toBinary(get<4>(data)), 0, true);
832                     mismatchFound = true;
833                 }
834             }
835 
836             if (mismatchFound)
837             {
838                 cout << "\nData updated successfully for all mismatching "
839                         "record-keyword pairs by choosing their corresponding "
840                         "data on System Backplane.\n"
841                      << endl;
842             }
843             else
844             {
845                 cout << "\nNo mismatch found for any of the above mentioned "
846                         "record-keyword pair. Exit successfully.\n";
847             }
848 
849             exit(0);
850         }
851         else if (option == VpdTool::MORE_OPTIONS)
852         {
853             cout << "\nIterate through all restorable record-keyword pairs\n";
854 
855             for (const auto& data : recKwData)
856             {
857                 do
858                 {
859                     cout << "\n|==============================================="
860                             "=================================================="
861                             "=====|"
862                          << endl;
863 
864                     cout << left << setw(6) << "S.No" << left << setw(8)
865                          << "Record" << left << setw(9) << "Keyword" << left
866                          << setw(30) << setfill(' ') << "Data On BMC" << left
867                          << setw(30) << setfill(' ')
868                          << "Data On System Backplane" << left << setw(14)
869                          << "Data Mismatch" << endl;
870 
871                     cout << left << setw(6) << get<0>(data) << left << setw(8)
872                          << get<1>(data) << left << setw(9) << get<2>(data)
873                          << left << setw(30) << setfill(' ') << get<3>(data)
874                          << left << setw(30) << setfill(' ') << get<4>(data)
875                          << left << setw(14) << get<5>(data);
876 
877                     cout << "\n|==============================================="
878                             "=================================================="
879                             "=====|"
880                          << endl;
881 
882                     if (get<5>(data) == "NO")
883                     {
884                         cout << "\nNo mismatch found.\n";
885                         printFixSystemVPDOption(VpdTool::NEW_VALUE_ON_BOTH);
886                         printFixSystemVPDOption(VpdTool::SKIP_CURRENT);
887                         printFixSystemVPDOption(VpdTool::EXIT);
888                     }
889                     else
890                     {
891                         printFixSystemVPDOption(VpdTool::BMC_DATA_FOR_CURRENT);
892                         printFixSystemVPDOption(
893                             VpdTool::SYSTEM_BACKPLANE_DATA_FOR_CURRENT);
894                         printFixSystemVPDOption(VpdTool::NEW_VALUE_ON_BOTH);
895                         printFixSystemVPDOption(VpdTool::SKIP_CURRENT);
896                         printFixSystemVPDOption(VpdTool::EXIT);
897                     }
898 
899                     cin >> option;
900 
901                     cout << "\n|==============================================="
902                             "=================================================="
903                             "=====|"
904                          << endl;
905 
906                     EditorImpl edit(constants::systemVpdFilePath, json,
907                                     get<1>(data), get<2>(data));
908 
909                     if (option == VpdTool::BMC_DATA_FOR_CURRENT)
910                     {
911                         edit.updateKeyword(toBinary(get<3>(data)), 0, true);
912                         cout << "\nData updated successfully.\n";
913                         break;
914                     }
915                     else if (option ==
916                              VpdTool::SYSTEM_BACKPLANE_DATA_FOR_CURRENT)
917                     {
918                         edit.updateKeyword(toBinary(get<4>(data)), 0, true);
919                         cout << "\nData updated successfully.\n";
920                         break;
921                     }
922                     else if (option == VpdTool::NEW_VALUE_ON_BOTH)
923                     {
924                         string value;
925                         cout << "\nEnter the new value to update both on BMC & "
926                                 "System Backplane (Value should be in ASCII or "
927                                 "in HEX(prefixed with 0x)) : ";
928                         cin >> value;
929                         cout << "\n|==========================================="
930                                 "=============================================="
931                                 "=============|"
932                              << endl;
933 
934                         edit.updateKeyword(toBinary(value), 0, true);
935                         cout << "\nData updated successfully.\n";
936                         break;
937                     }
938                     else if (option == VpdTool::SKIP_CURRENT)
939                     {
940                         cout << "\nSkipped the above record-keyword pair. "
941                                 "Continue to the next available pair.\n";
942                         break;
943                     }
944                     else if (option == VpdTool::EXIT)
945                     {
946                         cout << "\nExit successfully\n";
947                         exit(0);
948                     }
949                     else
950                     {
951                         cout << "\nProvide a valid option. Retrying for the "
952                                 "current record-keyword pair\n";
953                     }
954                 } while (1);
955             }
956             exit(0);
957         }
958         else if (option == VpdTool::EXIT)
959         {
960             cout << "\nExit successfully";
961             exit(0);
962         }
963         else
964         {
965             cout << "\nProvide a valid option. Retry.";
966             continue;
967         }
968 
969     } while (true);
970 }
971