blob: cb32125bcf94e8dfa026a1b3bcccbb985e9c3c0e [file] [log] [blame]
/*
// Copyright (c) 2018 Intel Corporation
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
*/
#pragma once
#include "common.hpp"
#include "gpio.hpp"
#include "xyz/openbmc_project/Chassis/Buttons/ID/server.hpp"
#include "xyz/openbmc_project/Chassis/Common/error.hpp"
#include <unistd.h>
#include <phosphor-logging/elog-errors.hpp>
const static constexpr char* ID_BUTTON = "ID_BTN";
struct IDButton
: sdbusplus::server::object::object<
sdbusplus::xyz::openbmc_project::Chassis::Buttons::server::ID>
{
IDButton(sdbusplus::bus::bus& bus, const char* path, EventPtr& event,
buttonConfig& buttonCfg,
sd_event_io_handler_t handler = IDButton::EventHandler) :
sdbusplus::server::object::object<
sdbusplus::xyz::openbmc_project::Chassis::Buttons::server::ID>(
bus, path),
fd(-1), buttonIFConfig(buttonCfg), bus(bus), event(event),
callbackHandler(handler)
{
int ret = -1;
// config group gpio based on the gpio defs read from the json file
ret = configGroupGpio(bus, buttonIFConfig);
if (ret < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
"ID_BUTTON: failed to config GPIO");
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
// initialize the button io fd from the buttonConfig
// which has fd stored when configGroupGpio is called
fd = buttonIFConfig.gpios[0].fd;
char buf;
::read(fd, &buf, sizeof(buf));
ret = sd_event_add_io(event.get(), nullptr, fd, EPOLLPRI,
callbackHandler, this);
if (ret < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
"ID_BUTTON: failed to add to event loop");
::closeGpio(fd);
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
}
~IDButton()
{
::closeGpio(fd);
}
void simPress() override;
static const std::string getGpioName()
{
return ID_BUTTON;
}
static int EventHandler(sd_event_source* es, int fd, uint32_t revents,
void* userdata)
{
int n = -1;
char buf = '0';
if (!userdata)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
"ID_BUTTON: userdata null!");
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
IDButton* idButton = static_cast<IDButton*>(userdata);
if (!idButton)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
"ID_BUTTON: null pointer!");
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
n = ::lseek(fd, 0, SEEK_SET);
if (n < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
"ID_BUTTON: lseek error!");
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
n = ::read(fd, &buf, sizeof(buf));
if (n < 0)
{
phosphor::logging::log<phosphor::logging::level::ERR>(
"ID_BUTTON: read error!");
throw sdbusplus::xyz::openbmc_project::Chassis::Common::Error::
IOError();
}
if (buf == '0')
{
phosphor::logging::log<phosphor::logging::level::DEBUG>(
"ID_BUTTON: pressed");
// emit pressed signal
idButton->pressed();
}
else
{
phosphor::logging::log<phosphor::logging::level::DEBUG>(
"ID_BUTTON: released");
// released
idButton->released();
}
return 0;
}
private:
int fd;
buttonConfig buttonIFConfig;
sdbusplus::bus::bus& bus;
EventPtr& event;
sd_event_io_handler_t callbackHandler;
};