blob: b7fba2e7d92d95eb1e44b534408fcc9b512501f6 [file] [log] [blame]
/**
* Copyright © 2019 Facebook
* Copyright © 2023 9elements GmbH
*
* 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 "gpio_presence.hpp"
#include <CLI/CLI.hpp>
#include <boost/asio/io_context.hpp>
#include <nlohmann/json.hpp>
#include <phosphor-logging/lg2.hpp>
#include <fstream>
namespace phosphor
{
namespace gpio
{
const std::map<std::string, int> biasMap = {
/**< Set bias as is. */
{"AS_IS", 0},
/**< Disable bias. */
{"DISABLE", GPIOD_LINE_REQUEST_FLAG_BIAS_DISABLE},
/**< Enable pull-up. */
{"PULL_UP", GPIOD_LINE_REQUEST_FLAG_BIAS_PULL_UP},
/**< Enable pull-down. */
{"PULL_DOWN", GPIOD_LINE_REQUEST_FLAG_BIAS_PULL_DOWN}};
}
} // namespace phosphor
int main(int argc, char** argv)
{
boost::asio::io_context io;
CLI::App app{"Monitor gpio presence status"};
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)
{
lg2::error("Failed to open config file: {FILE}", "FILE", gpioFileName);
return -1;
}
nlohmann::json gpioMonObj;
file >> gpioMonObj;
file.close();
std::vector<phosphor::gpio::GpioPresence> gpios;
for (auto& obj : gpioMonObj)
{
/* GPIO Line message */
std::string lineMsg = "GPIO Line ";
/* GPIO line */
gpiod_line* line = nullptr;
/* GPIO line configuration, default to monitor both edge */
struct gpiod_line_request_config config{
"gpio_monitor", GPIOD_LINE_REQUEST_EVENT_BOTH_EDGES, 0};
/* Pretty name of the inventory object */
std::string name;
/* Object path under inventory that will be created */
std::string inventory;
/* List of interfaces to associate to inventory item */
std::vector<std::string> extraInterfaces;
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())
{
lg2::error("Failed to find line name or gpio number: {FILE}",
"FILE", gpioFileName);
return -1;
}
std::string chipIdStr = obj["ChipId"].get<std::string>();
int gpioNum = obj["GpioNum"].get<int>();
lineMsg += chipIdStr + " " + 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"].get<std::string>();
lineMsg += lineName;
line = gpiod_line_find(lineName.c_str());
}
if (line == nullptr)
{
lg2::error("Failed to find the {GPIO}", "GPIO", lineMsg);
continue;
}
/* Parse out inventory argument. */
if (obj.find("Inventory") == obj.end())
{
lg2::error("{GPIO}: Inventory path not specified", "GPIO", lineMsg);
return -1;
}
else
{
inventory = obj["Inventory"].get<std::string>();
}
if (obj.find("Name") == obj.end())
{
lg2::error("{GPIO}: Name path not specified", "GPIO", lineMsg);
return -1;
}
else
{
name = obj["Name"].get<std::string>();
}
/* Parse optional bias */
if (obj.find("Bias") != obj.end())
{
std::string biasName = obj["Bias"].get<std::string>();
auto findBias = phosphor::gpio::biasMap.find(biasName);
if (findBias == phosphor::gpio::biasMap.end())
{
lg2::error("{GPIO}: Bias unknown: {BIAS}", "GPIO", lineMsg,
"BIAS", biasName);
return -1;
}
config.flags = findBias->second;
}
/* Parse optional active level */
if (obj.find("ActiveLow") != obj.end() && obj["ActiveLow"].get<bool>())
{
config.flags |= GPIOD_LINE_REQUEST_FLAG_ACTIVE_LOW;
}
/* Parse optional extra interfaces */
if (obj.find("ExtraInterfaces") != obj.end())
{
obj.at("ExtraInterfaces").get_to(extraInterfaces);
}
/* Create a monitor object and let it do all the rest */
gpios.push_back(phosphor::gpio::GpioPresence(
line, config, io, inventory, extraInterfaces, name, lineMsg));
}
io.run();
return 0;
}