xref: /openbmc/openpower-vpd-parser/vpd-manager/editor_impl.cpp (revision c78d887ccac761a70f2682fe3d5b948383cf56bd)
1 #include "config.h"
2 
3 #include "editor_impl.hpp"
4 
5 #include "vpdecc/vpdecc.h"
6 
7 #include "common_utility.hpp"
8 #include "ibm_vpd_utils.hpp"
9 #include "ipz_parser.hpp"
10 #include "parser_factory.hpp"
11 #include "vpd_exceptions.hpp"
12 
13 #include <phosphor-logging/elog-errors.hpp>
14 #include <xyz/openbmc_project/Common/error.hpp>
15 
16 using namespace openpower::vpd::parser::interface;
17 using namespace openpower::vpd::constants;
18 using namespace openpower::vpd::parser::factory;
19 using namespace openpower::vpd::ipz::parser;
20 
21 namespace openpower
22 {
23 namespace vpd
24 {
25 namespace manager
26 {
27 namespace editor
28 {
29 
30 void EditorImpl::checkPTForRecord(Binary::const_iterator& iterator,
31                                   Byte ptLength)
32 {
33     // auto iterator = ptRecord.cbegin();
34     auto end = std::next(iterator, ptLength + 1);
35 
36     // Look at each entry in the PT keyword for the record name
37     while (iterator < end)
38     {
39         auto stop = std::next(iterator, lengths::RECORD_NAME);
40         std::string record(iterator, stop);
41 
42         if (record == thisRecord.recName)
43         {
44             // Skip record name and record type
45             std::advance(iterator, lengths::RECORD_NAME + sizeof(RecordType));
46 
47             // Get record offset
48             thisRecord.recOffset = readUInt16LE(iterator);
49 
50             // pass the record offset length to read record length
51             std::advance(iterator, lengths::RECORD_OFFSET);
52             thisRecord.recSize = readUInt16LE(iterator);
53 
54             std::advance(iterator, lengths::RECORD_LENGTH);
55             thisRecord.recECCoffset = readUInt16LE(iterator);
56 
57             ECCLength len;
58             std::advance(iterator, lengths::RECORD_ECC_OFFSET);
59             len = readUInt16LE(iterator);
60             thisRecord.recECCLength = len;
61 
62             // once we find the record we don't need to look further
63             return;
64         }
65         else
66         {
67             // Jump the record
68             std::advance(iterator, lengths::RECORD_NAME + sizeof(RecordType) +
69                                        sizeof(RecordOffset) +
70                                        sizeof(RecordLength) +
71                                        sizeof(ECCOffset) + sizeof(ECCLength));
72         }
73     }
74     // imples the record was not found
75     throw std::runtime_error("Record not found");
76 }
77 
78 void EditorImpl::updateData(const Binary& kwdData)
79 {
80     std::size_t lengthToUpdate = kwdData.size() <= thisRecord.kwdDataLength
81                                      ? kwdData.size()
82                                      : thisRecord.kwdDataLength;
83 
84     auto iteratorToNewdata = kwdData.cbegin();
85     auto end = iteratorToNewdata;
86     std::advance(end, lengthToUpdate);
87 
88     // update data in file buffer as it will be needed to update ECC
89     // avoiding extra stream operation here
90     auto iteratorToKWdData = vpdFile.begin();
91     std::advance(iteratorToKWdData, thisRecord.kwDataOffset);
92     std::copy(iteratorToNewdata, end, iteratorToKWdData);
93 
94 #ifdef ManagerTest
95     auto startItr = vpdFile.begin();
96     std::advance(iteratorToKWdData, thisRecord.kwDataOffset);
97     auto endItr = startItr;
98     std::advance(endItr, thisRecord.kwdDataLength);
99 
100     Binary updatedData(startItr, endItr);
101     if (updatedData == kwdData)
102     {
103         throw std::runtime_error("Data updated successfully");
104     }
105 #else
106 
107     // update data in EEPROM as well. As we will not write complete file back
108     vpdFileStream.seekp(startOffset + thisRecord.kwDataOffset, std::ios::beg);
109 
110     iteratorToNewdata = kwdData.cbegin();
111     std::copy(iteratorToNewdata, end,
112               std::ostreambuf_iterator<char>(vpdFileStream));
113 
114     // get a hold to new data in case encoding is needed
115     thisRecord.kwdUpdatedData.resize(thisRecord.kwdDataLength);
116     auto itrToKWdData = vpdFile.cbegin();
117     std::advance(itrToKWdData, thisRecord.kwDataOffset);
118     auto kwdDataEnd = itrToKWdData;
119     std::advance(kwdDataEnd, thisRecord.kwdDataLength);
120     std::copy(itrToKWdData, kwdDataEnd, thisRecord.kwdUpdatedData.begin());
121 #endif
122 }
123 
124 void EditorImpl::checkRecordForKwd()
125 {
126     RecordOffset recOffset = thisRecord.recOffset;
127 
128     // Amount to skip for record ID, size, and the RT keyword
129     constexpr auto skipBeg = sizeof(RecordId) + sizeof(RecordSize) +
130                              lengths::KW_NAME + sizeof(KwSize);
131 
132     auto iterator = vpdFile.cbegin();
133     std::advance(iterator, recOffset + skipBeg + lengths::RECORD_NAME);
134 
135     auto end = iterator;
136     std::advance(end, thisRecord.recSize);
137     std::size_t dataLength = 0;
138 
139     while (iterator < end)
140     {
141         // Note keyword name
142         std::string kw(iterator, iterator + lengths::KW_NAME);
143 
144         // Check if the Keyword starts with '#'
145         char kwNameStart = *iterator;
146         std::advance(iterator, lengths::KW_NAME);
147 
148         // if keyword starts with #
149         if (POUND_KW == kwNameStart)
150         {
151             // Note existing keyword data length
152             dataLength = readUInt16LE(iterator);
153 
154             // Jump past 2Byte keyword length + data
155             std::advance(iterator, sizeof(PoundKwSize));
156         }
157         else
158         {
159             // Note existing keyword data length
160             dataLength = *iterator;
161 
162             // Jump past keyword length and data
163             std::advance(iterator, sizeof(KwSize));
164         }
165 
166         if (thisRecord.recKWd == kw)
167         {
168             thisRecord.kwDataOffset = std::distance(vpdFile.cbegin(), iterator);
169             thisRecord.kwdDataLength = dataLength;
170             return;
171         }
172 
173         // jump the data of current kwd to point to next kwd name
174         std::advance(iterator, dataLength);
175     }
176 
177     throw std::runtime_error("Keyword not found");
178 }
179 
180 void EditorImpl::updateRecordECC()
181 {
182     auto itrToRecordData = vpdFile.cbegin();
183     std::advance(itrToRecordData, thisRecord.recOffset);
184 
185     auto itrToRecordECC = vpdFile.cbegin();
186     std::advance(itrToRecordECC, thisRecord.recECCoffset);
187 
188     auto l_status = vpdecc_create_ecc(
189         const_cast<uint8_t*>(&itrToRecordData[0]), thisRecord.recSize,
190         const_cast<uint8_t*>(&itrToRecordECC[0]), &thisRecord.recECCLength);
191     if (l_status != VPD_ECC_OK)
192     {
193         throw std::runtime_error("Ecc update failed");
194     }
195 
196     auto end = itrToRecordECC;
197     std::advance(end, thisRecord.recECCLength);
198 
199 #ifndef ManagerTest
200     vpdFileStream.seekp(startOffset + thisRecord.recECCoffset, std::ios::beg);
201     std::copy(itrToRecordECC, end,
202               std::ostreambuf_iterator<char>(vpdFileStream));
203 #endif
204 }
205 
206 auto EditorImpl::getValue(offsets::Offsets offset)
207 {
208     auto itr = vpdFile.cbegin();
209     std::advance(itr, offset);
210     LE2ByteData lowByte = *itr;
211     LE2ByteData highByte = *(itr + 1);
212     lowByte |= (highByte << 8);
213 
214     return lowByte;
215 }
216 
217 void EditorImpl::checkRecordData()
218 {
219     auto itrToRecordData = vpdFile.cbegin();
220     std::advance(itrToRecordData, thisRecord.recOffset);
221 
222     auto itrToRecordECC = vpdFile.cbegin();
223     std::advance(itrToRecordECC, thisRecord.recECCoffset);
224 
225     checkECC(itrToRecordData, itrToRecordECC, thisRecord.recSize,
226              thisRecord.recECCLength);
227 }
228 
229 void EditorImpl::checkECC(Binary::const_iterator& itrToRecData,
230                           Binary::const_iterator& itrToECCData,
231                           RecordLength recLength, ECCLength eccLength)
232 {
233     auto l_status =
234         vpdecc_check_data(const_cast<uint8_t*>(&itrToRecData[0]), recLength,
235                           const_cast<uint8_t*>(&itrToECCData[0]), eccLength);
236 
237     if (l_status == VPD_ECC_CORRECTABLE_DATA)
238     {
239         try
240         {
241             if (vpdFileStream.is_open())
242             {
243                 vpdFileStream.seekp(startOffset + thisRecord.recOffset,
244                                     std::ios::beg);
245                 auto end = itrToRecData;
246                 std::advance(end, recLength);
247                 std::copy(itrToRecData, end,
248                           std::ostreambuf_iterator<char>(vpdFileStream));
249             }
250             else
251             {
252                 throw std::runtime_error("Ecc correction failed");
253             }
254         }
255         catch (const std::fstream::failure& e)
256         {
257             std::cout << "Error while operating on file with exception";
258             throw std::runtime_error("Ecc correction failed");
259         }
260     }
261     else if (l_status != VPD_ECC_OK)
262     {
263         throw std::runtime_error("Ecc check failed");
264     }
265 }
266 
267 void EditorImpl::readVTOC()
268 {
269     // read VTOC offset
270     RecordOffset tocOffset = getValue(offsets::VTOC_PTR);
271 
272     // read VTOC record length
273     RecordLength tocLength = getValue(offsets::VTOC_REC_LEN);
274 
275     // read TOC ecc offset
276     ECCOffset tocECCOffset = getValue(offsets::VTOC_ECC_OFF);
277 
278     // read TOC ecc length
279     ECCLength tocECCLength = getValue(offsets::VTOC_ECC_LEN);
280 
281     auto itrToRecord = vpdFile.cbegin();
282     std::advance(itrToRecord, tocOffset);
283 
284     auto iteratorToECC = vpdFile.cbegin();
285     std::advance(iteratorToECC, tocECCOffset);
286 
287     // validate ecc for the record
288     checkECC(itrToRecord, iteratorToECC, tocLength, tocECCLength);
289 
290     // to get to the record name.
291     std::advance(itrToRecord, sizeof(RecordId) + sizeof(RecordSize) +
292                                   // Skip past the RT keyword, which contains
293                                   // the record name.
294                                   lengths::KW_NAME + sizeof(KwSize));
295 
296     std::string recordName(itrToRecord, itrToRecord + lengths::RECORD_NAME);
297 
298     if ("VTOC" != recordName)
299     {
300         throw std::runtime_error("VTOC record not found");
301     }
302 
303     // jump to length of PT kwd
304     std::advance(itrToRecord, lengths::RECORD_NAME + lengths::KW_NAME);
305 
306     // Note size of PT
307     Byte ptLen = *itrToRecord;
308     std::advance(itrToRecord, 1);
309 
310     checkPTForRecord(itrToRecord, ptLen);
311 }
312 
313 template <typename T>
314 void EditorImpl::makeDbusCall(const std::string& object,
315                               const std::string& interface,
316                               const std::string& property,
317                               const std::variant<T>& data)
318 {
319     auto bus = sdbusplus::bus::new_default();
320     auto properties =
321         bus.new_method_call(INVENTORY_MANAGER_SERVICE, object.c_str(),
322                             "org.freedesktop.DBus.Properties", "Set");
323     properties.append(interface);
324     properties.append(property);
325     properties.append(data);
326 
327     auto result = bus.call(properties);
328 
329     if (result.is_method_error())
330     {
331         throw std::runtime_error("bus call failed");
332     }
333 }
334 
335 void EditorImpl::processAndUpdateCI(const std::string& objectPath)
336 {
337     inventory::ObjectMap objects;
338     for (auto& commonInterface : jsonFile["commonInterfaces"].items())
339     {
340         for (auto& ciPropertyList : commonInterface.value().items())
341         {
342             if (ciPropertyList.value().type() ==
343                 nlohmann::json::value_t::object)
344             {
345                 if ((ciPropertyList.value().value("recordName", "") ==
346                      thisRecord.recName) &&
347                     (ciPropertyList.value().value("keywordName", "") ==
348                      thisRecord.recKWd))
349                 {
350                     inventory::PropertyMap prop;
351                     inventory::InterfaceMap interfaces;
352                     std::string kwdData(thisRecord.kwdUpdatedData.begin(),
353                                         thisRecord.kwdUpdatedData.end());
354 
355                     prop.emplace(ciPropertyList.key(), std::move(kwdData));
356                     interfaces.emplace(commonInterface.key(), std::move(prop));
357                     objects.emplace(objectPath, std::move(interfaces));
358                 }
359             }
360         }
361     }
362     // Notify PIM
363     common::utility::callPIM(std::move(objects));
364 }
365 
366 void EditorImpl::processAndUpdateEI(const nlohmann::json& Inventory,
367                                     const inventory::Path& objPath)
368 {
369     inventory::ObjectMap objects;
370     for (const auto& extraInterface : Inventory["extraInterfaces"].items())
371     {
372         if (extraInterface.value() != NULL)
373         {
374             for (const auto& eiPropertyList : extraInterface.value().items())
375             {
376                 if (eiPropertyList.value().type() ==
377                     nlohmann::json::value_t::object)
378                 {
379                     if ((eiPropertyList.value().value("recordName", "") ==
380                          thisRecord.recName) &&
381                         ((eiPropertyList.value().value("keywordName", "") ==
382                           thisRecord.recKWd)))
383                     {
384                         inventory::PropertyMap prop;
385                         inventory::InterfaceMap interfaces;
386                         std::string kwdData(thisRecord.kwdUpdatedData.begin(),
387                                             thisRecord.kwdUpdatedData.end());
388                         encodeKeyword(kwdData, eiPropertyList.value().value(
389                                                    "encoding", ""));
390 
391                         prop.emplace(eiPropertyList.key(), std::move(kwdData));
392                         interfaces.emplace(extraInterface.key(),
393                                            std::move(prop));
394                         objects.emplace(objPath, std::move(interfaces));
395                     }
396                 }
397             }
398         }
399     }
400     // Notify PIM
401     common::utility::callPIM(std::move(objects));
402 }
403 
404 void EditorImpl::updateCache()
405 {
406     const std::vector<nlohmann::json>& groupEEPROM =
407         jsonFile["frus"][vpdFilePath].get_ref<const nlohmann::json::array_t&>();
408 
409     inventory::ObjectMap objects;
410     // iterate through all the inventories for this file path
411     for (const auto& singleInventory : groupEEPROM)
412     {
413         inventory::PropertyMap prop;
414         inventory::InterfaceMap interfaces;
415         // by default inherit property is true
416         bool isInherit = true;
417 
418         if (singleInventory.find("inherit") != singleInventory.end())
419         {
420             isInherit = singleInventory["inherit"].get<bool>();
421         }
422 
423         if (isInherit)
424         {
425             prop.emplace(getDbusNameForThisKw(thisRecord.recKWd),
426                          thisRecord.kwdUpdatedData);
427             interfaces.emplace(
428                 (IPZ_INTERFACE + (std::string) "." + thisRecord.recName),
429                 std::move(prop));
430             objects.emplace(
431                 (singleInventory["inventoryPath"].get<std::string>()),
432                 std::move(interfaces));
433 
434             // process Common interface
435             processAndUpdateCI(singleInventory["inventoryPath"]
436                                    .get_ref<const nlohmann::json::string_t&>());
437         }
438 
439         // process extra interfaces
440         processAndUpdateEI(singleInventory,
441                            singleInventory["inventoryPath"]
442                                .get_ref<const nlohmann::json::string_t&>());
443 
444         // check if we need to copy some specific records in this case.
445         if (singleInventory.find("copyRecords") != singleInventory.end())
446         {
447             if (find(singleInventory["copyRecords"].begin(),
448                      singleInventory["copyRecords"].end(),
449                      thisRecord.recName) !=
450                 singleInventory["copyRecords"].end())
451             {
452                 prop.emplace(thisRecord.recKWd, thisRecord.kwdUpdatedData);
453                 interfaces.emplace(
454                     (IPZ_INTERFACE + std::string{"."} + thisRecord.recName),
455                     std::move(prop));
456                 objects.emplace(
457                     (singleInventory["inventoryPath"].get<std::string>()),
458                     std::move(interfaces));
459             }
460         }
461     }
462     // Notify PIM
463     common::utility::callPIM(std::move(objects));
464 }
465 
466 void EditorImpl::expandLocationCode(const std::string& locationCodeType)
467 {
468     std::string propertyFCorTM{};
469     std::string propertySE{};
470 
471     if (locationCodeType == "fcs")
472     {
473         propertyFCorTM = readBusProperty(SYSTEM_OBJECT, "com.ibm.ipzvpd.VCEN",
474                                          "FC");
475         propertySE = readBusProperty(SYSTEM_OBJECT, "com.ibm.ipzvpd.VCEN",
476                                      "SE");
477     }
478     else if (locationCodeType == "mts")
479     {
480         propertyFCorTM = readBusProperty(SYSTEM_OBJECT, "com.ibm.ipzvpd.VSYS",
481                                          "TM");
482         propertySE = readBusProperty(SYSTEM_OBJECT, "com.ibm.ipzvpd.VSYS",
483                                      "SE");
484     }
485 
486     const nlohmann::json& groupFRUS =
487         jsonFile["frus"].get_ref<const nlohmann::json::object_t&>();
488     inventory::ObjectMap objects;
489 
490     for (const auto& itemFRUS : groupFRUS.items())
491     {
492         const std::vector<nlohmann::json>& groupEEPROM =
493             itemFRUS.value().get_ref<const nlohmann::json::array_t&>();
494         for (const auto& itemEEPROM : groupEEPROM)
495         {
496             inventory::PropertyMap prop;
497             inventory::InterfaceMap interfaces;
498             const auto& objectPath = itemEEPROM["inventoryPath"];
499             sdbusplus::message::object_path object(objectPath);
500 
501             // check if the given item implements location code interface
502             if (itemEEPROM["extraInterfaces"].find(IBM_LOCATION_CODE_INF) !=
503                 itemEEPROM["extraInterfaces"].end())
504             {
505                 const std::string& unexpandedLocationCode =
506                     itemEEPROM["extraInterfaces"][IBM_LOCATION_CODE_INF]
507                               ["LocationCode"]
508                                   .get_ref<const nlohmann::json::string_t&>();
509                 std::size_t idx = unexpandedLocationCode.find(locationCodeType);
510                 if (idx != std::string::npos)
511                 {
512                     std::string expandedLocationCode(unexpandedLocationCode);
513 
514                     if (locationCodeType == "fcs")
515                     {
516                         expandedLocationCode.replace(
517                             idx, 3,
518                             propertyFCorTM.substr(0, 4) + ".ND0." + propertySE);
519                     }
520                     else if (locationCodeType == "mts")
521                     {
522                         std::replace(propertyFCorTM.begin(),
523                                      propertyFCorTM.end(), '-', '.');
524                         expandedLocationCode.replace(
525                             idx, 3, propertyFCorTM + "." + propertySE);
526                     }
527 
528                     // update the DBUS interface COM as well as XYZ path
529                     prop.emplace("LocationCode", expandedLocationCode);
530                     // TODO depricate this com.ibm interface later
531                     interfaces.emplace(IBM_LOCATION_CODE_INF, prop);
532                     interfaces.emplace(XYZ_LOCATION_CODE_INF, std::move(prop));
533                 }
534             }
535             objects.emplace(std::move(object), std::move(interfaces));
536         }
537     }
538     // Notify PIM
539     common::utility::callPIM(std::move(objects));
540 }
541 
542 #ifndef ManagerTest
543 static void enableRebootGuard()
544 {
545     try
546     {
547         auto bus = sdbusplus::bus::new_default();
548         auto method = bus.new_method_call(
549             "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
550             "org.freedesktop.systemd1.Manager", "StartUnit");
551         method.append("reboot-guard-enable.service", "replace");
552         bus.call_noreply(method);
553     }
554     catch (const sdbusplus::exception_t& e)
555     {
556         std::string errMsg =
557             "Bus call to enable BMC reboot failed for reason: ";
558         errMsg += e.what();
559 
560         throw std::runtime_error(errMsg);
561     }
562 }
563 
564 static void disableRebootGuard()
565 {
566     try
567     {
568         auto bus = sdbusplus::bus::new_default();
569         auto method = bus.new_method_call(
570             "org.freedesktop.systemd1", "/org/freedesktop/systemd1",
571             "org.freedesktop.systemd1.Manager", "StartUnit");
572         method.append("reboot-guard-disable.service", "replace");
573         bus.call_noreply(method);
574     }
575     catch (const sdbusplus::exception_t& e)
576     {
577         using namespace phosphor::logging;
578         using InternalFailure =
579             sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure;
580 
581         std::string errMsg =
582             "Bus call to disable BMC reboot failed for reason: ";
583         errMsg += e.what();
584 
585         log<level::ERR>("Disable boot guard failed");
586         elog<InternalFailure>();
587 
588         throw std::runtime_error(errMsg);
589     }
590 }
591 #endif
592 
593 void EditorImpl::updateKeyword(const Binary& kwdData, uint32_t offset,
594                                const bool& updCache)
595 {
596     try
597     {
598         startOffset = offset;
599 #ifndef ManagerTest
600         // Restrict BMC from rebooting when VPD is being written. This will
601         // prevent any data/ECC corruption in case BMC reboots while VPD update.
602         enableRebootGuard();
603 
604         // TODO: Figure out a better way to get max possible VPD size.
605         Binary completeVPDFile;
606         completeVPDFile.resize(65504);
607         vpdFileStream.open(vpdFilePath,
608                            std::ios::in | std::ios::out | std::ios::binary);
609 
610         vpdFileStream.seekg(startOffset, std::ios_base::cur);
611         vpdFileStream.read(reinterpret_cast<char*>(&completeVPDFile[0]), 65504);
612         completeVPDFile.resize(vpdFileStream.gcount());
613         vpdFileStream.clear(std::ios_base::eofbit);
614 
615         vpdFile = completeVPDFile;
616 
617         if (objPath.empty() &&
618             jsonFile["frus"].find(vpdFilePath) != jsonFile["frus"].end())
619         {
620             objPath = jsonFile["frus"][vpdFilePath][0]["inventoryPath"]
621                           .get_ref<const nlohmann::json::string_t&>();
622         }
623 
624 #else
625 
626         Binary completeVPDFile = vpdFile;
627 
628 #endif
629         if (vpdFile.empty())
630         {
631             throw std::runtime_error("Invalid File");
632         }
633         auto iterator = vpdFile.cbegin();
634         std::advance(iterator, IPZ_DATA_START);
635 
636         Byte vpdType = *iterator;
637         if (vpdType == KW_VAL_PAIR_START_TAG)
638         {
639             // objPath should be empty only in case of test run.
640             ParserInterface* Iparser = ParserFactory::getParser(
641                 completeVPDFile, objPath, vpdFilePath, startOffset);
642             IpzVpdParser* ipzParser = dynamic_cast<IpzVpdParser*>(Iparser);
643 
644             try
645             {
646                 if (ipzParser == nullptr)
647                 {
648                     throw std::runtime_error("Invalid cast");
649                 }
650 
651                 ipzParser->processHeader();
652                 delete ipzParser;
653                 ipzParser = nullptr;
654                 // ParserFactory::freeParser(Iparser);
655 
656                 // process VTOC for PTT rkwd
657                 readVTOC();
658 
659                 // check record for keywrod
660                 checkRecordForKwd();
661 
662                 // Check Data before updating
663                 checkRecordData();
664 
665                 // update the data to the file
666                 updateData(kwdData);
667 
668                 // update the ECC data for the record once data has been updated
669                 updateRecordECC();
670 
671                 if (updCache)
672                 {
673 #ifndef ManagerTest
674                     // update the cache once data has been updated
675                     updateCache();
676 #endif
677                 }
678             }
679             catch (const std::exception& e)
680             {
681                 if (ipzParser != nullptr)
682                 {
683                     delete ipzParser;
684                 }
685                 throw std::runtime_error(e.what());
686             }
687 
688 #ifndef ManagerTest
689             // Once VPD data and Ecc update is done, disable BMC boot guard.
690             disableRebootGuard();
691 #endif
692 
693             return;
694         }
695         else
696         {
697             throw openpower::vpd::exceptions::VpdDataException(
698                 "Could not find start tag in VPD " + vpdFilePath);
699         }
700     }
701     catch (const std::exception& e)
702     {
703 #ifndef ManagerTest
704         // Disable reboot guard.
705         disableRebootGuard();
706 #endif
707 
708         throw std::runtime_error(e.what());
709     }
710 }
711 } // namespace editor
712 } // namespace manager
713 } // namespace vpd
714 } // namespace openpower
715