blob: fc32f0e63df558b20ad202468ebc9a618d368d29 [file] [log] [blame]
#include "util.hpp"
#include "estoraged_conf.hpp"
#include "getConfig.hpp"
#include <linux/fs.h>
#include <phosphor-logging/lg2.hpp>
#include <stdplus/fd/create.hpp>
#include <stdplus/fd/managed.hpp>
#include <stdplus/handle/managed.hpp>
#include <xyz/openbmc_project/Common/error.hpp>
#include <filesystem>
#include <fstream>
#include <iostream>
#include <optional>
#include <string>
namespace estoraged
{
namespace util
{
using ::sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure;
using ::stdplus::fd::ManagedFd;
uint64_t findSizeOfBlockDevice(const std::string& devPath)
{
ManagedFd fd;
uint64_t bytes = 0;
try
{
// open block dev
fd = stdplus::fd::open(devPath, stdplus::fd::OpenAccess::ReadOnly);
// get block size
fd.ioctl(BLKGETSIZE64, &bytes);
}
catch (...)
{
lg2::error("erase unable to open blockdev", "REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.DriveEraseFailure"),
"REDFISH_MESSAGE_ARGS", std::to_string(fd.get()));
throw InternalFailure();
}
return bytes;
}
uint8_t findPredictedMediaLifeLeftPercent(const std::string& sysfsPath)
{
// The eMMC spec defines two estimates for the life span of the device
// in the extended CSD field 269 and 268, named estimate A and estimate B.
// Linux exposes the values in the /life_time node.
// estimate A is for A type memory
// estimate B is for B type memory
//
// the estimate are encoded as such
// 0x01 <=> 0% - 10% device life time used
// 0x02 <=> 10% -20% device life time used
// ...
// 0x0A <=> 90% - 100% device life time used
// 0x0B <=> Exceeded its maximum estimated device life time
uint16_t estA = 0, estB = 0;
std::ifstream lifeTimeFile;
try
{
lifeTimeFile.open(sysfsPath + "/life_time", std::ios_base::in);
lifeTimeFile >> std::hex >> estA;
lifeTimeFile >> std::hex >> estB;
if ((estA == 0) || (estA > 11) || (estB == 0) || (estB > 11))
{
throw InternalFailure();
}
}
catch (...)
{
lg2::error("Unable to read sysfs", "REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.DriveEraseFailure"));
lifeTimeFile.close();
return 255;
}
lifeTimeFile.close();
// we are returning lowest LifeLeftPercent
uint8_t maxLifeUsed = 0;
if (estA > estB)
{
maxLifeUsed = estA;
}
else
{
maxLifeUsed = estB;
}
return static_cast<uint8_t>(11 - maxLifeUsed) * 10;
}
std::string getPartNumber(const std::filesystem::path& sysfsPath)
{
std::ifstream partNameFile;
std::string partName;
try
{
std::filesystem::path namePath(sysfsPath);
namePath /= "name";
partNameFile.open(namePath, std::ios_base::in);
partNameFile >> partName;
}
catch (...)
{
lg2::error("Unable to read sysfs", "REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.PartNumberFailure"));
}
partNameFile.close();
if (partName.empty())
{
partName = "unknown";
}
return partName;
}
std::string getSerialNumber(const std::filesystem::path& sysfsPath)
{
std::ifstream serialNumberFile;
std::string serialNumber;
try
{
std::filesystem::path serialPath(sysfsPath);
serialPath /= "serial";
serialNumberFile.open(serialPath, std::ios_base::in);
serialNumberFile >> serialNumber;
}
catch (...)
{
lg2::error("Unable to read sysfs", "REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.SerialNumberFailure"));
}
serialNumberFile.close();
if (serialNumber.empty())
{
serialNumber = "unknown";
}
return serialNumber;
}
std::optional<DeviceInfo> findDevice(const StorageData& data,
const std::filesystem::path& searchDir)
{
/* Check what type of storage device this is. */
estoraged::BasicVariantType typeVariant;
try
{
/* The EntityManager config should have this property set. */
typeVariant = data.at("Type");
}
catch (const boost::container::out_of_range& e)
{
lg2::error("Could not read device type", "REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.FindDeviceFail"));
return std::nullopt;
}
/* Check if location code/ silkscreen name is provided for the drive. */
std::string locationCode;
auto findLocationCode = data.find("LocationCode");
if (findLocationCode != data.end())
{
const std::string* locationCodePtr =
std::get_if<std::string>(&findLocationCode->second);
if (locationCodePtr != nullptr)
{
locationCode = *locationCodePtr;
}
}
/* Check if EraseMaxGeometry is provided. */
uint64_t eraseMaxGeometry = ERASE_MAX_GEOMETRY;
auto findEraseMaxGeometry = data.find("EraseMaxGeometry");
if (findEraseMaxGeometry != data.end())
{
const auto* eraseMaxGeometryPtr =
std::get_if<uint64_t>(&findEraseMaxGeometry->second);
if (eraseMaxGeometryPtr != nullptr)
{
lg2::info("eStorageD new eraseMaxGeometry found on system");
eraseMaxGeometry = *eraseMaxGeometryPtr;
}
}
/* Check if EraseMinGeometry is provided. */
uint64_t eraseMinGeometry = ERASE_MIN_GEOMETRY;
auto findEraseMinGeometry = data.find("EraseMinGeometry");
if (findEraseMinGeometry != data.end())
{
const auto* eraseMinGeometryPtr =
std::get_if<uint64_t>(&findEraseMinGeometry->second);
if (eraseMinGeometryPtr != nullptr)
{
lg2::info("eStorageD new eraseMinGeometry found on system");
eraseMinGeometry = *eraseMinGeometryPtr;
}
}
/*
* Determine the drive type and protocol to report for this device. Note
* that we only support eMMC currently, so report an error for any other
* device types.
*/
std::string deviceType = std::get<std::string>(typeVariant);
/* drive type and protocol to report in the Item.Drive dbus interface */
std::string driveType;
std::string driveProtocol;
if (deviceType.compare("EmmcDevice") == 0)
{
driveType = "SSD";
driveProtocol = "eMMC";
}
else
{
lg2::error("Unsupported device type {TYPE}", "TYPE", deviceType,
"REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.FindDeviceFail"));
return std::nullopt;
}
/* Look for the eMMC in the specified searchDir directory. */
for (const auto& dirEntry : std::filesystem::directory_iterator{searchDir})
{
/*
* We will look at the 'type' file to determine if this is an MMC
* device.
*/
std::filesystem::path curPath(dirEntry.path());
curPath /= "device/type";
if (!std::filesystem::exists(curPath))
{
/* The 'type' file doesn't exist. This must not be an eMMC. */
continue;
}
try
{
std::ifstream typeFile(curPath, std::ios_base::in);
std::string devType;
typeFile >> devType;
if (devType.compare("MMC") == 0 || devType.compare("SD") == 0)
{
/* Found it. Get the sysfs directory and device file. */
std::filesystem::path deviceName(dirEntry.path().filename());
std::filesystem::path sysfsDir = dirEntry.path();
sysfsDir /= "device";
std::filesystem::path deviceFile = "/dev";
deviceFile /= deviceName;
std::string luksName = "luks-" + deviceName.string();
return DeviceInfo{deviceFile, sysfsDir,
luksName, locationCode,
eraseMaxGeometry, eraseMinGeometry,
driveType, driveProtocol};
}
}
catch (...)
{
lg2::error("Failed to read device type for {PATH}", "PATH", curPath,
"REDFISH_MESSAGE_ID",
std::string("OpenBMC.0.1.FindDeviceFail"));
/*
* We will still continue searching, though. Maybe this wasn't the
* device we were looking for, anyway.
*/
}
}
/* Device wasn't found. */
return std::nullopt;
}
} // namespace util
} // namespace estoraged