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