Back up and Restore update

The commit updates the flow of Back up and Restore of system VPD.

Updates:
- Removing call_out_inventory_path tag from the PEL create call
in case there is any issue while backup or restore.
This was done to avoid any calls from phosphor-logging to
VPD-Manager at this stage as Manager service is not up at this
point resulting in a D-Bus call failure.

- Removing dupicate code as and where possible in the flow.

The commit is an extension of following commit:
https://gerrit.openbmc.org/c/openbmc/openpower-vpd-parser/+/62250/

Tests:
PEL was created duringthe flow and confirmed that there was no
call to VPD-Manager from phosphor-logging.

Tested for all te scenarios w.r.t backup and restore of system
VPD.

Signed-off-by: Sunny Srivastava <sunnsr25@in.ibm.com>
Change-Id: I92779247438bf50d907fd7f60ed94dda6fadb2a0
diff --git a/ibm_vpd_app.cpp b/ibm_vpd_app.cpp
index 9799982..8acb03a 100644
--- a/ibm_vpd_app.cpp
+++ b/ibm_vpd_app.cpp
@@ -44,6 +44,21 @@
 using namespace openpower::vpd::manager::editor;
 
 /**
+ * @brief API declaration, Populate Dbus.
+ *
+ * This method invokes all the populateInterface functions
+ * and notifies PIM about dbus object.
+ *
+ * @param[in] vpdMap - Either IPZ vpd map or Keyword vpd map based on the
+ * input.
+ * @param[in] js - Inventory json object
+ * @param[in] filePath - Path of the vpd file
+ * @param[in] preIntrStr - Interface string
+ */
+template <typename T>
+static void populateDbus(T& vpdMap, nlohmann::json& js, const string& filePath);
+
+/**
  * @brief Returns the BMC state
  */
 static auto getBMCState()
@@ -859,14 +874,11 @@
  * @brief Parse the given EEPROM file.
  *
  * @param[in] vpdFilePath - Path of EEPROM file
- * @param[in] invPath - Path of inventory object
  * @param[in] js- Reference to vpd inventory json object
- *
  * @return Parsed VPD map
  */
 std::variant<KeywordVpdMap, openpower::vpd::Store>
-    parseVpdFile(const std::string& vpdFilePath, const std::string& invPath,
-                 const nlohmann::json& js)
+    parseVpdFile(const std::string& vpdFilePath, const nlohmann::json& js)
 {
     uint32_t vpdStartOffset = 0;
     for (const auto& item : js["frus"][vpdFilePath])
@@ -874,13 +886,17 @@
         if (item.find("offset") != item.end())
         {
             vpdStartOffset = item["offset"];
+            break;
         }
     }
 
     Binary vpdVector = getVpdDataInVector(js, vpdFilePath);
 
     ParserInterface* parser = ParserFactory::getParser(
-        vpdVector, (pimPath + invPath), vpdFilePath, vpdStartOffset);
+        vpdVector,
+        (pimPath + js["frus"][vpdFilePath][0]["inventoryPath"]
+                       .get_ref<const nlohmann::json::string_t&>()),
+        vpdFilePath, vpdStartOffset);
 
     auto parseResult = parser->parse();
 
@@ -921,13 +937,17 @@
     }
     else
     {
-        auto backupVpdParsedResult = parseVpdFile(systemVpdBackupPath,
-                                                  backupVpdInvPath, js);
+        auto backupVpdParsedResult = parseVpdFile(systemVpdBackupPath, js);
 
         if (auto pVal = get_if<Store>(&backupVpdParsedResult))
         {
             backupVpdMap = pVal->getVpdMap();
         }
+        else
+        {
+            std::cerr << "Invalid format of VPD in back up. Restore aborted."
+                      << std::endl;
+        }
     }
 }
 
@@ -960,135 +980,6 @@
 }
 
 /**
- * @brief Populate base panel object on dbus.
-
- * This method invokes all the populateInterface functions
- * and notifies PIM about base panel dbus object.
-
- * @param[in] vpdMap - Base panel vpd map in IPZ format
- * @param[in] js - Inventory json object
- * @param[in] filePath - Path of the base panel vpd file
- */
-void populateBasePanelObjectOnDBus(const Parsed& vpdMap, nlohmann::json& js,
-                                   const string& filePath)
-{
-    inventory::InterfaceMap interfaces;
-    inventory::ObjectMap objects;
-    inventory::PropertyMap prop;
-
-    string ccinFromVpd = getKwVal(vpdMap, "VINI", "CC");
-    transform(ccinFromVpd.begin(), ccinFromVpd.end(), ccinFromVpd.begin(),
-              ::toupper);
-
-    for (const auto& item : js["frus"][filePath])
-    {
-        const auto& objectPath = item["inventoryPath"];
-        sdbusplus::message::object_path object(objectPath);
-
-        vector<string> ccinList;
-        if (item.find("ccin") != item.end())
-        {
-            for (const auto& cc : item["ccin"])
-            {
-                string ccin = cc;
-                transform(ccin.begin(), ccin.end(), ccin.begin(), ::toupper);
-                ccinList.push_back(ccin);
-            }
-        }
-
-        if (!ccinFromVpd.empty() && !ccinList.empty() &&
-            (find(ccinList.begin(), ccinList.end(), ccinFromVpd) ==
-             ccinList.end()))
-        {
-            continue;
-        }
-
-        if ((item.value("noprime", false)))
-        {
-            // Populate one time properties for non-primeable frus.
-            // For the remaining FRUs, this will get handled as a part of
-            // priming the inventory.
-            setOneTimeProperties(objectPath, interfaces);
-        }
-
-        // Populate the VPD keywords and the common interfaces only if we
-        // are asked to inherit that data from the VPD, else only add the
-        // extraInterfaces.
-        if (item.value("inherit", true))
-        {
-            // Each record in the VPD becomes an interface and all
-            // keyword within the record are properties under that
-            // interface.
-            for (const auto& record : vpdMap)
-            {
-                populateFruSpecificInterfaces(
-                    record.second, ipzVpdInf + record.first, interfaces);
-            }
-
-            if (js.find("commonInterfaces") != js.end())
-            {
-                populateInterfaces(js["commonInterfaces"], interfaces, vpdMap,
-                                   false);
-            }
-        }
-        else
-        {
-            // Check if we have been asked to inherit specific record(s)
-            if (item.find("copyRecords") != item.end())
-            {
-                for (const auto& record : item["copyRecords"])
-                {
-                    const string& recordName = record;
-                    if (vpdMap.find(recordName) != vpdMap.end())
-                    {
-                        populateFruSpecificInterfaces(vpdMap.at(recordName),
-                                                      ipzVpdInf + recordName,
-                                                      interfaces);
-                    }
-                }
-            }
-        }
-
-        // Populate interfaces and properties that are common to every FRU
-        // and additional interface that might be defined on a per-FRU
-        // basis.
-        if (item.find("extraInterfaces") != item.end())
-        {
-            populateInterfaces(item["extraInterfaces"], interfaces, vpdMap,
-                               false);
-        }
-
-        // embedded property(true or false) says whether the subfru is embedded
-        // into the parent fru (or) not. VPD sets Present property only for
-        // embedded frus. If the subfru is not an embedded FRU, the subfru may
-        // or may not be physically present. Those non embedded frus will always
-        // have Present=false irrespective of its physical presence or absence.
-        // Eg: nvme drive in nvme slot is not an embedded FRU. So don't set
-        // Present to true for such sub frus.
-        // Eg: ethernet port is embedded into bmc card. So set Present to true
-        // for such sub frus. Also donot populate present property for embedded
-        // subfru which is synthesized. Currently there is no subfru which are
-        // both embedded and synthesized. But still the case is handled here.
-        if ((item.value("embedded", true)) &&
-            (!item.value("synthesized", false)))
-        {
-            // Check if its required to handle presence for this FRU.
-            if (item.value("handlePresence", true))
-            {
-                inventory::PropertyMap presProp;
-                presProp.emplace("Present", true);
-                insertOrMerge(interfaces, invItemIntf, move(presProp));
-            }
-        }
-
-        objects.emplace(move(object), move(interfaces));
-    }
-
-    // Notify PIM
-    common::utility::callPIM(move(objects));
-}
-
-/**
  * @brief API to check if we need to restore system VPD
  * This functionality is only applicable for IPZ VPD data.
 
@@ -1118,6 +1009,7 @@
 
         if (backupVpdMap.empty())
         {
+            std::cerr << "Backup VPD map is empty" << std::endl;
             return;
         }
     }
@@ -1158,52 +1050,35 @@
                     }
                     else
                     {
-                        auto recItr = backupVpdMap.find(backupVpdRecName);
-                        if (recItr != backupVpdMap.end())
+                        backupValue = getKwVal(backupVpdMap, backupVpdRecName,
+                                               backupVpdKwName);
+
+                        if (backupValue.empty())
                         {
-                            string errorMsg = backupVpdRecName +
-                                              " Record does not exist in "
-                                              "the EEPROM file ";
+                            string errorMsg{};
+                            if (backupVpdMap.find(backupVpdRecName) ==
+                                backupVpdMap.end())
+                            {
+                                errorMsg = backupVpdRecName +
+                                           " Record does not exist in "
+                                           "the EEPROM file ";
+                            }
+                            else
+                            {
+                                errorMsg = backupVpdKwName +
+                                           " Keyword not found or empty.";
+                            }
 
                             errorMsg += systemVpdBackupPath;
 
                             PelAdditionalData additionalData;
                             additionalData.emplace("DESCRIPTION", errorMsg);
 
-                            additionalData.emplace("CALLOUT_INVENTORY_PATH",
-                                                   INVENTORY_PATH +
-                                                       backupVpdInvPath);
-
                             createPEL(additionalData, PelSeverity::ERROR,
                                       errIntfForInvalidVPD, nullptr);
 
                             continue;
                         }
-                        else
-                        {
-                            backupValue = getKwVal(backupVpdMap,
-                                                   backupVpdRecName,
-                                                   backupVpdKwName);
-                            if (backupValue.empty())
-                            {
-                                string errorMsg = backupVpdKwName +
-                                                  " Keyword not found in "
-                                                  "the backup VPD file ";
-
-                                errorMsg += systemVpdBackupPath;
-
-                                PelAdditionalData additionalData;
-                                additionalData.emplace("DESCRIPTION", errorMsg);
-
-                                additionalData.emplace("CALLOUT_INVENTORY_PATH",
-                                                       INVENTORY_PATH +
-                                                           backupVpdInvPath);
-
-                                createPEL(additionalData, PelSeverity::ERROR,
-                                          errIntfForInvalidVPD, nullptr);
-                                continue;
-                            }
-                        }
                     }
 
                     Binary backupDataInBinary(backupValue.begin(),
@@ -1242,9 +1117,6 @@
 
                                 // data mismatch
                                 PelAdditionalData additionalData;
-                                additionalData.emplace("CALLOUT_INVENTORY_PATH",
-                                                       INVENTORY_PATH +
-                                                           objectPath);
 
                                 additionalData.emplace("DESCRIPTION", errMsg);
                                 additionalData.emplace(
@@ -1267,6 +1139,13 @@
                                     continue;
                                 }
                             }
+                            else
+                            {
+                                // both the backup and primary data is
+                                // non-default and same. Nothing needs to be
+                                // done.
+                                continue;
+                            }
                         }
 
                         // If the backup is on the cache we need to copy the
@@ -1299,16 +1178,12 @@
                         errMsg += (*it).first;
                         errMsg += " and keyword: ";
                         errMsg += keywordName;
-                        errMsg += ". SSR need to update primary VPD.";
+                        errMsg += ". Update primary VPD.";
 
                         // mfg default on both backup and primary, log PEL
                         PelAdditionalData additionalData;
-                        additionalData.emplace("CALLOUT_INVENTORY_PATH",
-                                               INVENTORY_PATH + objectPath);
-
                         additionalData.emplace("DESCRIPTION", errMsg);
 
-                        // log PEL TODO: Block IPL
                         createPEL(additionalData, PelSeverity::ERROR,
                                   errIntfForVPDDefault, nullptr);
 
@@ -1332,11 +1207,6 @@
             }
         }
     }
-
-    if (!isBackupOnCache)
-    {
-        populateBasePanelObjectOnDBus(backupVpdMap, js, systemVpdBackupPath);
-    }
 }
 
 /**
@@ -1895,8 +1765,7 @@
         try
         {
             variant<KeywordVpdMap, Store> parseResult;
-
-            parseResult = parseVpdFile(file, baseFruInventoryPath, js);
+            parseResult = parseVpdFile(file, js);
 
             if (isSystemVpd)
             {
@@ -1923,6 +1792,12 @@
         }
         catch (const exception& e)
         {
+            if (!systemVpdBackupPath.empty())
+            {
+                file = systemVpdBackupPath;
+                baseFruInventoryPath = backupVpdInvPath;
+            }
+
             executePostFailAction(js, file);
             throw;
         }
@@ -1938,11 +1813,6 @@
     }
     catch (const VpdEccException& ex)
     {
-        if (!systemVpdBackupPath.empty())
-        {
-            baseFruInventoryPath = backupVpdInvPath;
-        }
-
         additionalData.emplace("DESCRIPTION", "ECC check failed");
         additionalData.emplace("CALLOUT_INVENTORY_PATH",
                                INVENTORY_PATH + baseFruInventoryPath);
@@ -1958,12 +1828,6 @@
     }
     catch (const VpdDataException& ex)
     {
-        if (!systemVpdBackupPath.empty())
-        {
-            file = systemVpdBackupPath;
-            baseFruInventoryPath = backupVpdInvPath;
-        }
-
         if (isThisPcieOnPass1planar(js, file))
         {
             std::cout << "Pcie_device  [" << file