blob: f2c73f8b25878a978cd37b65ac6d587fb2e6a9c3 [file] [log] [blame] [edit]
#include "config.h"
#include "single_fab.hpp"
#include "constants.hpp"
#include "event_logger.hpp"
#include "parser.hpp"
#include "types.hpp"
#include <nlohmann/json.hpp>
#include <utility/common_utility.hpp>
#include <utility/json_utility.hpp>
#include <utility/vpd_specific_utility.hpp>
namespace vpd
{
constexpr auto pimPersistVsbpPath =
"/var/lib/phosphor-inventory-manager/xyz/openbmc_project/inventory/system/chassis/motherboard/com.ibm.ipzvpd.VSBP";
constexpr auto IM_SIZE_IN_BYTES = 0x04;
constexpr auto IM_KW_VALUE_OFFSET = 0x000005fb;
std::string SingleFab::getImFromPersistedLocation() const noexcept
{
try
{
auto l_parsedVsbpJsonObj =
jsonUtility::getParsedJson(pimPersistVsbpPath);
if (!l_parsedVsbpJsonObj.contains("value0") ||
!l_parsedVsbpJsonObj["value0"].contains(constants::kwdIM) ||
!l_parsedVsbpJsonObj["value0"][constants::kwdIM].is_array())
{
throw std::runtime_error(
"Json is empty or mandatory tag(s) missing from JSON");
}
const types::BinaryVector l_imValue =
l_parsedVsbpJsonObj["value0"][constants::kwdIM]
.get<types::BinaryVector>();
std::ostringstream l_imData;
for (const auto& l_byte : l_imValue)
{
l_imData << std::setw(2) << std::setfill('0') << std::hex
<< static_cast<int>(l_byte);
}
return l_imData.str();
}
catch (const std::exception& l_ex)
{}
return std::string();
}
std::string SingleFab::getImFromPlanar() const noexcept
{
try
{
const std::string l_systemPlanarPath(SYSTEM_VPD_FILE_PATH);
Parser l_parserObj(l_systemPlanarPath, nlohmann::json{});
std::shared_ptr<ParserInterface> l_vpdParserInstance =
l_parserObj.getVpdParserInstance();
auto l_readValue = l_vpdParserInstance->readKeywordFromHardware(
std::make_tuple(constants::recVSBP, constants::kwdIM));
if (auto l_keywordValue =
std::get_if<types::BinaryVector>(&l_readValue);
l_keywordValue && !l_keywordValue->empty())
{
std::ostringstream l_imData;
for (const auto& l_byte : *l_keywordValue)
{
l_imData << std::setw(2) << std::setfill('0') << std::hex
<< static_cast<int>(l_byte);
}
return l_imData.str();
}
}
catch (const std::ifstream::failure& l_ex)
{}
return std::string();
}
bool SingleFab::setImOnPlanar(const std::string& i_imValue) const noexcept
{
try
{
types::BinaryVector l_imValue;
const std::string l_systemPlanarEepromPath = SYSTEM_VPD_FILE_PATH;
// Convert string to vector of bytes
for (auto l_value : i_imValue | std::views::chunk(2))
{
std::string l_byteString(l_value.begin(), l_value.end());
l_imValue.push_back(
static_cast<uint8_t>(std::stoi(l_byteString, nullptr, 16)));
}
std::shared_ptr<Parser> l_parserObj = std::make_shared<Parser>(
l_systemPlanarEepromPath, nlohmann::json{});
int l_bytes_updated = l_parserObj->updateVpdKeywordOnHardware(
std::make_tuple(constants::recVSBP, constants::kwdIM, l_imValue));
return l_bytes_updated > 0 ? true : false;
}
catch (const std::exception& l_ex)
{
return false;
}
}
bool SingleFab::isFieldModeEnabled() const noexcept
{
try
{
std::vector<std::string> l_cmdOutput =
commonUtility::executeCmd("/sbin/fw_printenv fieldmode");
if (l_cmdOutput.size() > 0)
{
commonUtility::toLower(l_cmdOutput[0]);
// Remove the new line character from the string.
l_cmdOutput[0].erase(l_cmdOutput[0].length() - 1);
return l_cmdOutput[0] == "fieldmode=true" ? true : false;
}
}
catch (const std::exception& l_ex)
{}
return false;
}
void SingleFab::updateSystemImValueInVpdToP11Series(
std::string i_currentImValuePlanar) const noexcept
{
bool l_retVal{false};
if (!i_currentImValuePlanar.empty())
{
if (i_currentImValuePlanar.compare(
constants::VALUE_4, constants::VALUE_1,
std::to_string(constants::VALUE_3)) ==
constants::STR_CMP_SUCCESS)
{
i_currentImValuePlanar.replace(constants::VALUE_4,
constants::VALUE_1,
std::to_string(constants::VALUE_2));
}
// update the IM value to P11 series(6000x). Replace the first character
// of IM value string with '6'
l_retVal = setImOnPlanar(i_currentImValuePlanar.replace(
constants::VALUE_0, constants::VALUE_1,
std::to_string(constants::VALUE_6)));
}
if (!l_retVal)
{
EventLogger::createSyncPel(
types::ErrorType::InternalFailure,
types::SeverityType::Informational, __FILE__, __FUNCTION__, 0,
std::string("Failed to update IM value to P11 series."),
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
}
}
int SingleFab::singleFabImOverride() const noexcept
{
const std::string& l_planarImValue = getImFromPlanar();
const std::string& l_eBmcImValue = getImFromPersistedLocation();
const bool& l_isFieldModeEnabled = isFieldModeEnabled();
const bool& l_isLabModeEnabled =
!l_isFieldModeEnabled; // used for understanding
const bool& l_isPowerVsImage = vpdSpecificUtility::isPowerVsImage();
const bool& l_isNormalImage = !l_isPowerVsImage; // used for understanding
if (!isValidImSeries(l_planarImValue))
{
// Create Errorlog for invalid IM series encountered
EventLogger::createSyncPel(
types::ErrorType::InvalidSystem, types::SeverityType::Error,
__FILE__, __FUNCTION__, 0,
std::string("Invalid IM found on the system planar, IM value : ") +
l_planarImValue,
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
return constants::SUCCESS;
}
if (!l_eBmcImValue.empty())
{
if (isP10System(l_eBmcImValue))
{
if (isP10System(l_planarImValue))
{
if (l_isFieldModeEnabled && l_isNormalImage)
{
EventLogger::createSyncPel(
types::ErrorType::SystemTypeMismatch,
types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
std::string("Mismatch in IM value found eBMC IM [") +
l_eBmcImValue + std::string("] planar IM [") +
l_planarImValue +
std::string("] Field mode enabled [") +
((l_isFieldModeEnabled) ? "true" : "false") +
std::string("]"),
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
return constants::FAILURE;
}
}
else if (isP11System(l_planarImValue))
{
if (!(l_isLabModeEnabled && l_isNormalImage))
{
EventLogger::createSyncPel(
types::ErrorType::SystemTypeMismatch,
types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
std::string("Mismatch in IM value found eBMC IM [") +
l_eBmcImValue + std::string("] planar IM [") +
l_planarImValue +
std::string("] Field mode enabled [") +
((l_isFieldModeEnabled) ? "true" : "false") +
std::string("]"),
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
return constants::FAILURE;
}
}
}
else if (isP11System(l_eBmcImValue))
{
if (l_isPowerVsImage)
{
EventLogger::createSyncPel(
types::ErrorType::SystemTypeMismatch,
types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
std::string("Mismatch in IM value found eBMC IM [") +
l_eBmcImValue + std::string("] planar IM [") +
l_planarImValue +
std::string("] Field mode enabled [") +
((l_isFieldModeEnabled) ? "true" : "false") +
std::string("]"),
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
return constants::FAILURE;
}
else
{
if (isP10System(l_planarImValue))
{
updateSystemImValueInVpdToP11Series(l_planarImValue);
}
}
}
}
else
{
if (isP11System(l_planarImValue) && l_isPowerVsImage)
{
EventLogger::createSyncPel(
types::ErrorType::SystemTypeMismatch,
types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
std::string("Mismatch in IM value found eBMC IM [") +
l_eBmcImValue + std::string("] planar IM [") +
l_planarImValue + std::string("] Field mode enabled [") +
((l_isFieldModeEnabled) ? "true" : "false") +
std::string("]"),
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
return constants::FAILURE;
}
else if (isP10System(l_planarImValue) && l_isNormalImage)
{
if (l_isLabModeEnabled)
{
EventLogger::createSyncPel(
types::ErrorType::UnknownSystemSettings,
types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
std::string("Mismatch in IM value found eBMC IM [") +
l_eBmcImValue + std::string("] planar IM [") +
l_planarImValue +
std::string("] Field mode enabled [") +
((l_isFieldModeEnabled) ? "true" : "false") +
std::string("]"),
std::nullopt, std::nullopt, std::nullopt, std::nullopt);
}
else
{
updateSystemImValueInVpdToP11Series(l_planarImValue);
}
}
}
return constants::SUCCESS;
}
} // namespace vpd