blob: b3a98cc552566f661b892f711bcd978719043b3a [file] [log] [blame]
#pragma once
#include "button_config.hpp"
#include "common.hpp"
#include "xyz/openbmc_project/Chassis/Common/error.hpp"
#include <phosphor-logging/elog-errors.hpp>
// This is the base class for all the button interface types
//
class ButtonIface
{
public:
ButtonIface(sdbusplus::bus_t& bus, EventPtr& event, ButtonConfig& buttonCfg,
sd_event_io_handler_t handler = ButtonIface::EventHandler) :
bus(bus),
event(event), config(buttonCfg), callbackHandler(handler)
{
int ret = -1;
std::string configType;
// config group gpio or cpld based on the defs read from the json file
if (buttonCfg.type == ConfigType::gpio)
{
configType = "GPIO";
ret = configGroupGpio(config);
}
else if (buttonCfg.type == ConfigType::cpld)
{
configType = "CPLD";
ret = configCpld(config);
}
if (ret < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
(getFormFactorType() + " : failed to config " + configType)
.c_str());
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
}
virtual ~ButtonIface() {}
/**
* @brief This method is called from sd-event provided callback function
* callbackHandler if platform specific event handling is needed then a
* derived class instance with its specific evend handling logic along with
* init() function can be created to override the default event handling.
*/
virtual void handleEvent(sd_event_source* es, int fd, uint32_t revents) = 0;
static int EventHandler(sd_event_source* es, int fd, uint32_t revents,
void* userdata)
{
if (userdata)
{
ButtonIface* buttonIface = static_cast<ButtonIface*>(userdata);
buttonIface->handleEvent(es, fd, revents);
}
return 0;
}
std::string getFormFactorType() const
{
return config.formFactorName;
}
protected:
/**
* @brief oem specific initialization can be done under init function.
* if platform specific initialization is needed then
* a derived class instance with its own init function to override the
* default init() method can be added.
*/
virtual void init()
{
// initialize the button io fd from the ButtonConfig
// which has fd stored when configGroupGpio or configCpld is called
for (auto fd : config.fds)
{
char buf;
int ret = ::read(fd, &buf, sizeof(buf));
if (ret < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
(getFormFactorType() + " : read error!").c_str());
}
ret = sd_event_add_io(event.get(), nullptr, fd, EPOLLPRI,
callbackHandler, this);
if (ret < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
(getFormFactorType() + " : failed to add to event loop")
.c_str());
if (fd > 0)
{
::close(fd);
}
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
}
}
/**
* @brief similar to init() oem specific deinitialization can be done under
* deInit function. if platform specific deinitialization is needed then a
* derived class instance with its own init function to override the default
* deinit() method can be added.
*/
virtual void deInit()
{
for (auto fd : config.fds)
{
if (fd > 0)
{
::close(fd);
}
}
}
sdbusplus::bus_t& bus;
EventPtr& event;
ButtonConfig config;
sd_event_io_handler_t callbackHandler;
};