blob: 83e14b6d51e7abfd7be280e537526c5ec730b218 [file] [log] [blame]
/**
* Copyright © 2019 Facebook
*
* 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.
*/
#include "gpioMon.hpp"
#include <CLI/CLI.hpp>
#include <boost/asio/io_context.hpp>
#include <nlohmann/json.hpp>
#include <phosphor-logging/log.hpp>
#include <fstream>
using namespace phosphor::logging;
namespace phosphor
{
namespace gpio
{
std::map<std::string, int> polarityMap = {
/**< Only watch falling edge events. */
{"FALLING", GPIOD_LINE_REQUEST_EVENT_FALLING_EDGE},
/**< Only watch rising edge events. */
{"RISING", GPIOD_LINE_REQUEST_EVENT_RISING_EDGE},
/**< Monitor both types of events. */
{"BOTH", GPIOD_LINE_REQUEST_EVENT_BOTH_EDGES}};
}
} // namespace phosphor
int main(int argc, char** argv)
{
boost::asio::io_context io;
CLI::App app{"Monitor GPIO line for requested state change"};
std::string gpioFileName;
/* Add an input option */
app.add_option("-c,--config", gpioFileName, "Name of config json file")
->required()
->check(CLI::ExistingFile);
/* Parse input parameter */
try
{
app.parse(argc, argv);
}
catch (const CLI::Error& e)
{
return app.exit(e);
}
/* Get list of gpio config details from json file */
std::ifstream file(gpioFileName);
if (!file)
{
log<level::ERR>("GPIO monitor config file not found",
entry("GPIO_MON_FILE=%s", gpioFileName.c_str()));
return -1;
}
nlohmann::json gpioMonObj;
file >> gpioMonObj;
file.close();
std::vector<std::unique_ptr<phosphor::gpio::GpioMonitor>> gpios;
for (auto& obj : gpioMonObj)
{
/* GPIO Line message */
std::string lineMsg = "GPIO Line ";
/* GPIO line */
gpiod_line* line = NULL;
/* Log message string */
std::string errMsg;
/* GPIO line configuration, default to monitor both edge */
struct gpiod_line_request_config config
{
"gpio_monitor", GPIOD_LINE_REQUEST_EVENT_BOTH_EDGES, 0
};
/* flag to monitor */
bool flag = false;
/* target to start */
std::string target;
/* multi targets to start */
std::map<std::string, std::vector<std::string>> targets;
if (obj.find("LineName") == obj.end())
{
/* If there is no line Name defined then gpio num nd chip
* id must be defined. GpioNum is integer mapping to the
* GPIO key configured by the kernel
*/
if (obj.find("GpioNum") == obj.end() ||
obj.find("ChipId") == obj.end())
{
log<level::ERR>(
"Failed to find line name or gpio number",
entry("GPIO_JSON_FILE_NAME=%s", gpioFileName.c_str()));
return -1;
}
std::string chipIdStr = obj["ChipId"];
int gpioNum = obj["GpioNum"];
lineMsg += std::to_string(gpioNum);
/* Get the GPIO line */
line = gpiod_line_get(chipIdStr.c_str(), gpioNum);
}
else
{
/* Find the GPIO line */
std::string lineName = obj["LineName"];
lineMsg += lineName;
line = gpiod_line_find(lineName.c_str());
}
if (line == NULL)
{
errMsg = "Failed to find the " + lineMsg;
log<level::ERR>(errMsg.c_str());
return -1;
}
/* Get event to be monitored, if it is not defined then
* Both rising falling edge will be monitored.
*/
if (obj.find("EventMon") != obj.end())
{
std::string eventStr = obj["EventMon"];
auto findEvent = phosphor::gpio::polarityMap.find(eventStr);
if (findEvent == phosphor::gpio::polarityMap.end())
{
errMsg = "Incorrect GPIO monitor event defined " + lineMsg;
log<level::ERR>(errMsg.c_str());
return -1;
}
config.request_type = findEvent->second;
}
/* Get flag if monitoring needs to continue after first event */
if (obj.find("Continue") != obj.end())
{
flag = obj["Continue"];
}
/* Parse out target argument. It is fine if the user does not
* pass this if they are not interested in calling into any target
* on meeting a condition.
*/
if (obj.find("Target") != obj.end())
{
target = obj["Target"];
}
/* Parse out the targets argument if multi-targets are needed.*/
if (obj.find("Targets") != obj.end())
{
obj.at("Targets").get_to(targets);
}
/* Create a monitor object and let it do all the rest */
gpios.push_back(std::make_unique<phosphor::gpio::GpioMonitor>(
line, config, io, target, targets, lineMsg, flag));
}
io.run();
return 0;
}