1 #include "guid.hpp" 2 3 #include <host-ipmid/ipmid-api.h> 4 #include <mapper.h> 5 6 #include <phosphor-logging/log.hpp> 7 #include <sstream> 8 #include <string> 9 10 using namespace phosphor::logging; 11 12 namespace cache 13 { 14 15 command::Guid guid; 16 17 } // namespace cache 18 19 namespace command 20 { 21 22 std::unique_ptr<sdbusplus::bus::match_t> matchPtr(nullptr); 23 24 static constexpr auto guidObjPath = "/org/openbmc/control/chassis0"; 25 static constexpr auto propInterface = "org.freedesktop.DBus.Properties"; 26 27 Guid getSystemGUID() 28 { 29 // Canned System GUID for QEMU where the Chassis DBUS object is not 30 // populated 31 Guid guid = {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 32 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10}; 33 34 constexpr auto chassisIntf = "org.openbmc.control.Chassis"; 35 36 sd_bus_message* reply = nullptr; 37 sd_bus_error error = SD_BUS_ERROR_NULL; 38 sd_bus* bus = ipmid_get_sd_bus_connection(); 39 char* uuid = nullptr; 40 char* busname = nullptr; 41 42 do 43 { 44 int rc = mapper_get_service(bus, guidObjPath, &busname); 45 if (rc < 0) 46 { 47 log<level::ERR>("Failed to get bus name", 48 entry("PATH=%s", guidObjPath), 49 entry("ERROR=%s", strerror(-rc))); 50 break; 51 } 52 53 rc = sd_bus_call_method(bus, busname, guidObjPath, propInterface, "Get", 54 &error, &reply, "ss", chassisIntf, "uuid"); 55 if (rc < 0) 56 { 57 log<level::ERR>("Failed to call Get Method", 58 entry("ERROR=%s", strerror(-rc))); 59 break; 60 } 61 62 rc = sd_bus_message_read(reply, "v", "s", &uuid); 63 if (rc < 0 || uuid == NULL) 64 { 65 log<level::ERR>("Failed to get a response", 66 entry("ERROR=%s", strerror(-rc))); 67 break; 68 } 69 70 std::string readUUID(uuid); 71 auto len = readUUID.length(); 72 73 for (size_t iter = 0, inc = 0; iter < len && inc < BMC_GUID_LEN; 74 iter += 2, inc++) 75 { 76 uint8_t hexVal = 77 std::strtoul(readUUID.substr(iter, 2).c_str(), NULL, 16); 78 guid[inc] = hexVal; 79 } 80 } while (0); 81 82 sd_bus_error_free(&error); 83 reply = sd_bus_message_unref(reply); 84 free(busname); 85 86 return guid; 87 } 88 89 void registerGUIDChangeCallback() 90 { 91 if (matchPtr == nullptr) 92 { 93 using namespace sdbusplus::bus::match::rules; 94 sdbusplus::bus::bus bus{ipmid_get_sd_bus_connection()}; 95 96 matchPtr = std::make_unique<sdbusplus::bus::match_t>( 97 bus, 98 path_namespace(guidObjPath) + type::signal() + 99 member("PropertiesChanged") + interface(propInterface), 100 [](sdbusplus::message::message&) { 101 cache::guid = getSystemGUID(); 102 }); 103 } 104 } 105 106 } // namespace command 107