blob: 90374d230fc9b087aaea62e6cf1ea59f96740a09 [file] [log] [blame]
#include "gpioMonitor.hpp"
#include "common_utility.hpp"
#include "ibm_vpd_utils.hpp"
#include <boost/asio.hpp>
#include <boost/bind/bind.hpp>
#include <gpiod.hpp>
using namespace std;
using namespace openpower::vpd::constants;
namespace openpower
{
namespace vpd
{
namespace manager
{
bool GpioEventHandler::getPresencePinValue()
{
Byte gpioData = 1;
gpiod::line presenceLine = gpiod::find_line(presencePin);
if (!presenceLine)
{
cerr << "Error getPresencePinValue: couldn't find presence line:"
<< presencePin << " on GPIO \n";
// return previous state as we couldn't read current state
return prevPresPinValue;
}
presenceLine.request(
{"Op-panel presence line", gpiod::line_request::DIRECTION_INPUT, 0});
gpioData = presenceLine.get_value();
return gpioData;
}
void GpioMonitor::initGpioInfos(
std::shared_ptr<boost::asio::io_context>& ioContext)
{
Byte outputValue = 0;
Byte presenceValue = 0;
string presencePinName{}, outputPinName{};
string devNameAddr{}, driverType{}, busType{}, objectPath{};
for (const auto& eachFRU : jsonFile["frus"].items())
{
for (const auto& eachInventory : eachFRU.value())
{
objectPath = eachInventory["inventoryPath"];
if ((eachInventory.find("presence") != eachInventory.end()) &&
(eachInventory.find("preAction") != eachInventory.end()))
{
if (!eachInventory["presence"].value("pollingRequired", false))
{
// Polling not required for this FRU , skip.
continue;
}
for (const auto& presStatus : eachInventory["presence"].items())
{
if (presStatus.key() == "pin")
{
presencePinName = presStatus.value();
}
else if (presStatus.key() == "value")
{
presenceValue = presStatus.value();
}
}
// Based on presence pin value, preAction pin will be set/reset
// This action will be taken before vpd collection.
for (const auto& preAction : eachInventory["preAction"].items())
{
if (preAction.key() == "pin")
{
outputPinName = preAction.value();
}
else if (preAction.key() == "value")
{
outputValue = preAction.value();
}
}
if ((eachInventory.find("devAddress") != eachInventory.end()) &&
(eachInventory.find("driverType") != eachInventory.end()) &&
(eachInventory.find("busType") != eachInventory.end()))
{
devNameAddr = eachInventory["devAddress"];
driverType = eachInventory["driverType"];
busType = eachInventory["busType"];
// Init all Gpio info variables
std::shared_ptr<GpioEventHandler> gpioObj =
make_shared<GpioEventHandler>(
presencePinName, presenceValue, outputPinName,
outputValue, devNameAddr, driverType, busType,
objectPath, ioContext);
gpioObjects.push_back(gpioObj);
}
}
}
}
}
void GpioEventHandler::toggleGpio()
{
bool presPinVal = getPresencePinValue();
bool isPresent = false;
// preserve the new value
prevPresPinValue = presPinVal;
if (presPinVal == presenceValue)
{
isPresent = true;
}
// if FRU went away set the present property to false
if (!isPresent)
{
inventory::ObjectMap objects;
inventory::InterfaceMap interfaces;
inventory::PropertyMap presProp;
presProp.emplace("Present", false);
interfaces.emplace("xyz.openbmc_project.Inventory.Item", presProp);
objects.emplace(objectPath, move(interfaces));
common::utility::callPIM(move(objects));
}
gpiod::line outputLine = gpiod::find_line(outputPin);
if (!outputLine)
{
cerr << "Error: toggleGpio: couldn't find output line:" << outputPin
<< ". Skipping update\n";
return;
}
outputLine.request({"FRU presence: update the output GPIO pin",
gpiod::line_request::DIRECTION_OUTPUT, 0},
isPresent ? outputValue : (!outputValue));
string cmnd = createBindUnbindDriverCmnd(devNameAddr, busType, driverType,
isPresent ? "bind" : "unbind");
cout << cmnd << endl;
executeCmd(cmnd);
}
void GpioEventHandler::handleTimerExpiry(
const boost::system::error_code& ec,
std::shared_ptr<boost::asio::steady_timer>& timer)
{
if (ec == boost::asio::error::operation_aborted)
{
return;
}
if (ec)
{
std::cerr << "Timer wait failed for gpio pin" << ec.message();
return;
}
if (hasEventOccurred())
{
toggleGpio();
}
timer->expires_at(std::chrono::steady_clock::now() +
std::chrono::seconds(5));
timer->async_wait(boost::bind(&GpioEventHandler::handleTimerExpiry, this,
boost::asio::placeholders::error, timer));
}
void GpioEventHandler::doEventAndTimerSetup(
std::shared_ptr<boost::asio::io_context>& ioContext)
{
prevPresPinValue = getPresencePinValue();
static vector<std::shared_ptr<boost::asio::steady_timer>> timers;
auto timer = make_shared<boost::asio::steady_timer>(
*ioContext, std::chrono::seconds(5));
timer->async_wait(boost::bind(&GpioEventHandler::handleTimerExpiry, this,
boost::asio::placeholders::error, timer));
timers.push_back(timer);
}
} // namespace manager
} // namespace vpd
} // namespace openpower