blob: 9dfcdcfa46783540d2e00cca4a9f3ca64cbd4e7c [file] [log] [blame]
#include "config.h"
#include "manager.hpp"
#include "bios_handler.hpp"
#include "common_utility.hpp"
#include "editor_impl.hpp"
#include "gpioMonitor.hpp"
#include "ibm_vpd_utils.hpp"
#include "ipz_parser.hpp"
#include "parser_factory.hpp"
#include "reader_impl.hpp"
#include "vpd_exceptions.hpp"
#include <filesystem>
#include <phosphor-logging/elog-errors.hpp>
using namespace openpower::vpd::manager;
using namespace openpower::vpd::constants;
using namespace openpower::vpd::inventory;
using namespace openpower::vpd::manager::editor;
using namespace openpower::vpd::manager::reader;
using namespace std;
using namespace openpower::vpd::parser;
using namespace openpower::vpd::parser::factory;
using namespace openpower::vpd::ipz::parser;
using namespace openpower::vpd::exceptions;
using namespace phosphor::logging;
namespace openpower
{
namespace vpd
{
namespace manager
{
Manager::Manager(sdbusplus::bus_t&& bus, const char* busName,
const char* objPath, const char* /*iFace*/) :
ServerObject<ManagerIface>(bus, objPath),
_bus(std::move(bus)), _manager(_bus, objPath)
{
_bus.request_name(busName);
}
void Manager::run()
{
try
{
processJSON();
restoreSystemVpd();
listenHostState();
listenAssetTag();
// Create an instance of the BIOS handler
BiosHandler biosHandler{_bus, *this};
auto event = sdeventplus::Event::get_default();
GpioMonitor gpioMon1(jsonFile, event);
_bus.attach_event(event.get(), SD_EVENT_PRIORITY_IMPORTANT);
cout << "VPD manager event loop started\n";
event.loop();
}
catch (const std::exception& e)
{
std::cerr << e.what() << "\n";
}
}
/**
* @brief An api to get list of blank system VPD properties.
* @param[in] vpdMap - IPZ vpd map.
* @param[in] objectPath - Object path for the FRU.
* @param[out] blankPropertyList - Properties which are blank in System VPD and
* needs to be updated as standby.
*/
static void
getListOfBlankSystemVpd(Parsed& vpdMap, const string& objectPath,
std::vector<RestoredEeproms>& blankPropertyList)
{
for (const auto& systemRecKwdPair : svpdKwdMap)
{
auto it = vpdMap.find(systemRecKwdPair.first);
// check if record is found in map we got by parser
if (it != vpdMap.end())
{
const auto& kwdListForRecord = systemRecKwdPair.second;
for (const auto& keyword : kwdListForRecord)
{
DbusPropertyMap& kwdValMap = it->second;
auto iterator = kwdValMap.find(keyword);
if (iterator != kwdValMap.end())
{
string& kwdValue = iterator->second;
// check bus data
const string& recordName = systemRecKwdPair.first;
const string& busValue = readBusProperty(
objectPath, ipzVpdInf + recordName, keyword);
if (busValue.find_first_not_of(' ') != string::npos)
{
if (kwdValue.find_first_not_of(' ') == string::npos)
{
// implies data is blank on EEPROM but not on cache.
// So EEPROM vpd update is required.
Binary busData(busValue.begin(), busValue.end());
blankPropertyList.push_back(std::make_tuple(
objectPath, recordName, keyword, busData));
}
}
}
}
}
}
}
void Manager::restoreSystemVpd()
{
std::cout << "Attempting system VPD restore" << std::endl;
ParserInterface* parser = nullptr;
try
{
auto vpdVector = getVpdDataInVector(jsonFile, systemVpdFilePath);
parser = ParserFactory::getParser(vpdVector);
auto parseResult = parser->parse();
if (auto pVal = std::get_if<Store>(&parseResult))
{
// map to hold all the keywords whose value is blank and
// needs to be updated at standby.
std::vector<RestoredEeproms> blankSystemVpdProperties{};
getListOfBlankSystemVpd(pVal->getVpdMap(), SYSTEM_OBJECT,
blankSystemVpdProperties);
// if system VPD restore is required, update the
// EEPROM
for (const auto& item : blankSystemVpdProperties)
{
std::cout << "Restoring keyword: " << std::get<2>(item)
<< std::endl;
writeKeyword(std::get<0>(item), std::get<1>(item),
std::get<2>(item), std::get<3>(item));
}
}
else
{
std::cerr << "Not a valid format to restore system VPD"
<< std::endl;
}
}
catch (const std::exception& e)
{
std::cerr << "Failed to restore system VPD due to exception: "
<< e.what() << std::endl;
}
// release the parser object
ParserFactory::freeParser(parser);
}
void Manager::listenHostState()
{
static std::shared_ptr<sdbusplus::bus::match_t> hostState =
std::make_shared<sdbusplus::bus::match_t>(
_bus,
sdbusplus::bus::match::rules::propertiesChanged(
"/xyz/openbmc_project/state/host0",
"xyz.openbmc_project.State.Host"),
[this](sdbusplus::message_t& msg) { hostStateCallBack(msg); });
}
void Manager::hostStateCallBack(sdbusplus::message_t& msg)
{
if (msg.is_method_error())
{
std::cerr << "Error in reading signal " << std::endl;
}
Path object;
PropertyMap propMap;
msg.read(object, propMap);
const auto itr = propMap.find("CurrentHostState");
if (itr != propMap.end())
{
if (auto hostState = std::get_if<std::string>(&(itr->second)))
{
// implies system is moving from standby to power on state
if (*hostState == "xyz.openbmc_project.State.Host.HostState."
"TransitioningToRunning")
{
// check and perfrom recollection for FRUs replaceable at
// standby.
performVPDRecollection();
return;
}
}
std::cerr << "Failed to read Host state" << std::endl;
}
}
void Manager::listenAssetTag()
{
static std::shared_ptr<sdbusplus::bus::match_t> assetMatcher =
std::make_shared<sdbusplus::bus::match_t>(
_bus,
sdbusplus::bus::match::rules::propertiesChanged(
"/xyz/openbmc_project/inventory/system",
"xyz.openbmc_project.Inventory.Decorator.AssetTag"),
[this](sdbusplus::message_t& msg) { assetTagCallback(msg); });
}
void Manager::assetTagCallback(sdbusplus::message_t& msg)
{
if (msg.is_method_error())
{
std::cerr << "Error in reading signal " << std::endl;
}
Path object;
PropertyMap propMap;
msg.read(object, propMap);
const auto itr = propMap.find("AssetTag");
if (itr != propMap.end())
{
if (auto assetTag = std::get_if<std::string>(&(itr->second)))
{
// Call Notify to persist the AssetTag
inventory::ObjectMap objectMap = {
{std::string{"/system"},
{{"xyz.openbmc_project.Inventory.Decorator.AssetTag",
{{"AssetTag", *assetTag}}}}}};
common::utility::callPIM(std::move(objectMap));
}
else
{
std::cerr << "Failed to read asset tag" << std::endl;
}
}
}
void Manager::processJSON()
{
std::ifstream json(INVENTORY_JSON_SYM_LINK, std::ios::binary);
if (!json)
{
throw std::runtime_error("json file not found");
}
jsonFile = nlohmann::json::parse(json);
if (jsonFile.find("frus") == jsonFile.end())
{
throw std::runtime_error("frus group not found in json");
}
const nlohmann::json& groupFRUS =
jsonFile["frus"].get_ref<const nlohmann::json::object_t&>();
for (const auto& itemFRUS : groupFRUS.items())
{
const std::vector<nlohmann::json>& groupEEPROM =
itemFRUS.value().get_ref<const nlohmann::json::array_t&>();
for (const auto& itemEEPROM : groupEEPROM)
{
bool isMotherboard = false;
std::string redundantPath;
if (itemEEPROM["extraInterfaces"].find(
"xyz.openbmc_project.Inventory.Item.Board.Motherboard") !=
itemEEPROM["extraInterfaces"].end())
{
isMotherboard = true;
}
if (itemEEPROM.find("redundantEeprom") != itemEEPROM.end())
{
redundantPath = itemEEPROM["redundantEeprom"]
.get_ref<const nlohmann::json::string_t&>();
}
frus.emplace(
itemEEPROM["inventoryPath"]
.get_ref<const nlohmann::json::string_t&>(),
std::make_tuple(itemFRUS.key(), redundantPath, isMotherboard));
if (itemEEPROM["extraInterfaces"].find(IBM_LOCATION_CODE_INF) !=
itemEEPROM["extraInterfaces"].end())
{
fruLocationCode.emplace(
itemEEPROM["extraInterfaces"][IBM_LOCATION_CODE_INF]
["LocationCode"]
.get_ref<const nlohmann::json::string_t&>(),
itemEEPROM["inventoryPath"]
.get_ref<const nlohmann::json::string_t&>());
}
if (itemEEPROM.value("replaceableAtStandby", false))
{
replaceableFrus.emplace_back(itemFRUS.key());
}
}
}
}
void Manager::writeKeyword(const sdbusplus::message::object_path path,
const std::string recordName,
const std::string keyword, const Binary value)
{
try
{
std::string objPath{path};
// Strip any inventory prefix in path
if (objPath.find(INVENTORY_PATH) == 0)
{
objPath = objPath.substr(sizeof(INVENTORY_PATH) - 1);
}
if (frus.find(objPath) == frus.end())
{
throw std::runtime_error("Inventory path not found");
}
inventory::Path vpdFilePath = std::get<0>(frus.find(objPath)->second);
// instantiate editor class to update the data
EditorImpl edit(vpdFilePath, jsonFile, recordName, keyword, objPath);
uint32_t offset = 0;
// Setup offset, if any
for (const auto& item : jsonFile["frus"][vpdFilePath])
{
if (item.find("offset") != item.end())
{
offset = item["offset"];
break;
}
}
edit.updateKeyword(value, offset, true);
// If we have a redundant EEPROM to update, then update just the EEPROM,
// not the cache since that is already done when we updated the primary
if (!std::get<1>(frus.find(objPath)->second).empty())
{
EditorImpl edit(std::get<1>(frus.find(objPath)->second), jsonFile,
recordName, keyword);
edit.updateKeyword(value, offset, false);
}
// if it is a motehrboard FRU need to check for location expansion
if (std::get<2>(frus.find(objPath)->second))
{
if (recordName == "VCEN" && (keyword == "FC" || keyword == "SE"))
{
edit.expandLocationCode("fcs");
}
else if (recordName == "VSYS" &&
(keyword == "TM" || keyword == "SE"))
{
edit.expandLocationCode("mts");
}
}
return;
}
catch (const std::exception& e)
{
std::cerr << e.what() << std::endl;
}
}
ListOfPaths
Manager::getFRUsByUnexpandedLocationCode(const LocationCode locationCode,
const NodeNumber nodeNumber)
{
ReaderImpl read;
return read.getFrusAtLocation(locationCode, nodeNumber, fruLocationCode);
}
ListOfPaths
Manager::getFRUsByExpandedLocationCode(const LocationCode locationCode)
{
ReaderImpl read;
return read.getFRUsByExpandedLocationCode(locationCode, fruLocationCode);
}
LocationCode Manager::getExpandedLocationCode(const LocationCode locationCode,
const NodeNumber nodeNumber)
{
ReaderImpl read;
return read.getExpandedLocationCode(locationCode, nodeNumber,
fruLocationCode);
}
void Manager::performVPDRecollection()
{
// get list of FRUs replaceable at standby
for (const auto& item : replaceableFrus)
{
const vector<nlohmann::json>& groupEEPROM = jsonFile["frus"][item];
const nlohmann::json& singleFru = groupEEPROM[0];
const string& inventoryPath =
singleFru["inventoryPath"]
.get_ref<const nlohmann::json::string_t&>();
bool prePostActionRequired = false;
if ((jsonFile["frus"][item].at(0)).find("preAction") !=
jsonFile["frus"][item].at(0).end())
{
if (!executePreAction(jsonFile, item))
{
// if the FRU has preAction defined then its execution should
// pass to ensure bind/unbind of data.
// preAction execution failed. should not call bind/unbind.
log<level::ERR>(
"Pre-Action execution failed for the FRU",
entry("ERROR=%s",
("Inventory path: " + inventoryPath).c_str()));
continue;
}
prePostActionRequired = true;
}
if ((singleFru.find("devAddress") == singleFru.end()) ||
(singleFru.find("driverType") == singleFru.end()) ||
(singleFru.find("busType") == singleFru.end()))
{
// The FRUs is marked for replacement but missing mandatory
// fields for recollection. Skip to another replaceable fru.
log<level::ERR>(
"Recollection Failed as mandatory field missing in Json",
entry("ERROR=%s",
("Recollection failed for " + inventoryPath).c_str()));
continue;
}
string str = "echo ";
string deviceAddress = singleFru["devAddress"];
const string& driverType = singleFru["driverType"];
const string& busType = singleFru["busType"];
// devTreeStatus flag is present in json as false to mention
// that the EEPROM is not mentioned in device tree. If this flag
// is absent consider the value to be true, i.e EEPROM is
// mentioned in device tree
if (!singleFru.value("devTreeStatus", true))
{
auto pos = deviceAddress.find('-');
if (pos != string::npos)
{
string busNum = deviceAddress.substr(0, pos);
deviceAddress =
"0x" + deviceAddress.substr(pos + 1, string::npos);
string deleteDevice = str + deviceAddress + " > /sys/bus/" +
busType + "/devices/" + busType + "-" +
busNum + "/delete_device";
executeCmd(deleteDevice);
string addDevice = str + driverType + " " + deviceAddress +
" > /sys/bus/" + busType + "/devices/" +
busType + "-" + busNum + "/new_device";
executeCmd(addDevice);
}
else
{
log<level::ERR>(
"Wrong format of device address in Json",
entry(
"ERROR=%s",
("Recollection failed for " + inventoryPath).c_str()));
continue;
}
}
else
{
executeCmd(createBindUnbindDriverCmnd(deviceAddress, busType,
driverType, "/unbind"));
executeCmd(createBindUnbindDriverCmnd(deviceAddress, busType,
driverType, "/bind"));
}
// this check is added to avoid file system expensive call in case not
// required.
if (prePostActionRequired)
{
// Check if device showed up (test for file)
if (!filesystem::exists(item))
{
// If not, then take failure postAction
executePostFailAction(jsonFile, item);
}
}
}
}
} // namespace manager
} // namespace vpd
} // namespace openpower