blob: d9b0ff184e8510040cf3e3865f70254cc9980b8c [file] [log] [blame]
#pragma once
#include "const.hpp"
#include "types.hpp"
#include <cstddef>
#include <fstream>
#include <nlohmann/json.hpp>
#include <tuple>
namespace openpower
{
namespace vpd
{
namespace manager
{
namespace editor
{
/** @class EditorImpl */
class EditorImpl
{
public:
EditorImpl() = delete;
EditorImpl(const EditorImpl&) = delete;
EditorImpl& operator=(const EditorImpl&) = delete;
EditorImpl(EditorImpl&&) = delete;
EditorImpl& operator=(EditorImpl&&) = delete;
~EditorImpl()
{
}
/** @brief Construct EditorImpl class
*
* @param[in] record - Record Name
* @param[in] kwd - Keyword
* @param[in] vpd - Vpd Vector
*/
EditorImpl(const std::string& record, const std::string& kwd,
Binary&& vpd) :
startOffset(0),
thisRecord(record, kwd), vpdFile(std::move(vpd))
{
}
/** @brief Construct EditorImpl class
*
* @param[in] path - Path to the vpd file
* @param[in] json - Parsed inventory json
* @param[in] record - Record name
* @param[in] kwd - Keyword
* @param[in] inventoryPath - Inventory path of the vpd
*/
EditorImpl(const inventory::Path& path, const nlohmann::json& json,
const std::string& record, const std::string& kwd,
const sdbusplus::message::object_path& inventoryPath) :
vpdFilePath(path),
objPath(inventoryPath), startOffset(0), jsonFile(json),
thisRecord(record, kwd)
{
}
/** @brief Construct EditorImpl class
*
* @param[in] path - EEPROM path
* @param[in] json - Parsed inventory json object
* @param[in] record - Record name
* @param[in] kwd - Keyword name
*/
EditorImpl(const inventory::Path& path, const nlohmann::json& json,
const std::string& record, const std::string& kwd) :
vpdFilePath(path),
jsonFile(json), thisRecord(record, kwd)
{
}
/**
* @brief Update data for keyword
* The method looks for the record name to update in VTOC and then
* looks for the keyword name in that record. when found it updates the data
* of keyword with the given data. It does not block keyword data update in
* case the length of new data is greater than or less than the current data
* length. If the new data length is more than the length alotted to that
* keyword the new data will be truncated to update only the allotted
* length. Similarly if the new data length is less then only that much data
* will be updated for the keyword and remaining bits will be left
* unchanged.
*
* Following is the algorithm used to update keyword:
* 1) Look for the record name in the given VPD file
* 2) Look for the keyword name for which data needs to be updated
* which is the table of contents record.
* 3) update the data for that keyword with the new data
*
* @param[in] kwdData - data to update
* @param[in] updCache - Flag which tells whether to update Cache or not.
*/
void updateKeyword(const Binary& kwdData, const bool& updCache);
/** @brief Expands location code on DBUS
* @param[in] locationCodeType - "fcs" or "mts"
*/
void expandLocationCode(const std::string& locationCodeType);
private:
/** @brief read VTOC record from the vpd file
*/
void readVTOC();
/** @brief validate ecc data for the VTOC record
* @param[in] itrToRecData -iterator to the record data
* @param[in] itrToECCData - iterator to the ECC data
* @param[in] recLength - Length of the record
* @param[in] eccLength - Length of the record's ECC
*/
void checkECC(Binary::const_iterator& itrToRecData,
Binary::const_iterator& itrToECCData,
openpower::vpd::constants::RecordLength recLength,
openpower::vpd::constants::ECCLength eccLength);
/** @brief reads value at the given offset
* @param[in] offset - offset value
* @return value at that offset in bigendian
*/
auto getValue(openpower::vpd::constants::offsets::Offsets offset);
/** @brief Checks if required record name exist in the VPD file
* @param[in] iterator - pointing to start of PT kwd
* @param[in] ptLength - length of the PT kwd
*/
void checkPTForRecord(Binary::const_iterator& iterator, Byte ptLength);
/** @brief Checks for required keyword in the record */
void checkRecordForKwd();
/** @brief update data for given keyword
* @param[in] kwdData- data to be updated
*/
void updateData(const Binary& kwdData);
/** @brief update record ECC */
void updateRecordECC();
/** @brief method to update cache once the data for keyword has been updated
*/
void updateCache();
/** @brief method to process and update CI in case required
* @param[in] - objectPath - path of the object to introspect
*/
void processAndUpdateCI(const std::string& objectPath);
/** @brief method to process and update extra interface
* @param[in] Inventory - single inventory json subpart
* @param[in] objPath - path of the object to introspect
*/
void processAndUpdateEI(const nlohmann::json& Inventory,
const inventory::Path& objPath);
/** @brief method to make busctl call
*
* @param[in] object - bus object path
* @param[in] interface - bus interface
* @param[in] property - property to update on BUS
* @param[in] data - data to be updated on Bus
*
*/
template <typename T>
void makeDbusCall(const std::string& object, const std::string& interface,
const std::string& property, const std::variant<T>& data);
// path to the VPD file to edit
inventory::Path vpdFilePath;
// inventory path of the vpd fru to update keyword
const inventory::Path objPath;
// stream to perform operation on file
std::fstream vpdFileStream;
// offset to get vpd data from EEPROM
uint32_t startOffset;
// file to store parsed json
const nlohmann::json jsonFile;
// structure to hold info about record to edit
struct RecInfo
{
Binary kwdUpdatedData; // need access to it in case encoding is needed
const std::string recName;
const std::string recKWd;
openpower::vpd::constants::RecordOffset recOffset;
openpower::vpd::constants::ECCOffset recECCoffset;
std::size_t recECCLength;
std::size_t kwdDataLength;
openpower::vpd::constants::RecordSize recSize;
openpower::vpd::constants::DataOffset kwDataOffset;
// constructor
RecInfo(const std::string& rec, const std::string& kwd) :
recName(rec), recKWd(kwd), recOffset(0), recECCoffset(0),
recECCLength(0), kwdDataLength(0), recSize(0), kwDataOffset(0)
{
}
} thisRecord;
Binary vpdFile;
// If requested Interface is common Interface
bool isCI;
/** @brief This API will be used to find out Parent FRU of Module/CPU
*
* @param[in] - moduleObjPath, object path of that FRU
* @param[in] - fruType, Type of Parent FRU
* for Module/CPU Parent Type- FruAndModule
*
* @return returns vpd file path of Parent Fru of that Module
*/
std::string getSysPathForThisFruType(const std::string& moduleObjPath,
const std::string& fruType);
/** @brief This API will search for correct EEPROM path for asked CPU
* and will init vpdFilePath
*/
void getVpdPathForCpu();
}; // class EditorImpl
} // namespace editor
} // namespace manager
} // namespace vpd
} // namespace openpower