146a9a5b9SPatrick Rudolph /**
246a9a5b9SPatrick Rudolph * Copyright © 2019 Facebook
346a9a5b9SPatrick Rudolph * Copyright © 2023 9elements GmbH
446a9a5b9SPatrick Rudolph *
546a9a5b9SPatrick Rudolph * Licensed under the Apache License, Version 2.0 (the "License");
646a9a5b9SPatrick Rudolph * you may not use this file except in compliance with the License.
746a9a5b9SPatrick Rudolph * You may obtain a copy of the License at
846a9a5b9SPatrick Rudolph *
946a9a5b9SPatrick Rudolph * http://www.apache.org/licenses/LICENSE-2.0
1046a9a5b9SPatrick Rudolph *
1146a9a5b9SPatrick Rudolph * Unless required by applicable law or agreed to in writing, software
1246a9a5b9SPatrick Rudolph * distributed under the License is distributed on an "AS IS" BASIS,
1346a9a5b9SPatrick Rudolph * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1446a9a5b9SPatrick Rudolph * See the License for the specific language governing permissions and
1546a9a5b9SPatrick Rudolph * limitations under the License.
1646a9a5b9SPatrick Rudolph */
1746a9a5b9SPatrick Rudolph
1846a9a5b9SPatrick Rudolph #include "gpio_presence.hpp"
1946a9a5b9SPatrick Rudolph
2046a9a5b9SPatrick Rudolph #include <CLI/CLI.hpp>
2146a9a5b9SPatrick Rudolph #include <boost/asio/io_context.hpp>
2246a9a5b9SPatrick Rudolph #include <nlohmann/json.hpp>
2346a9a5b9SPatrick Rudolph #include <phosphor-logging/lg2.hpp>
2446a9a5b9SPatrick Rudolph
2546a9a5b9SPatrick Rudolph #include <fstream>
2646a9a5b9SPatrick Rudolph
2746a9a5b9SPatrick Rudolph namespace phosphor
2846a9a5b9SPatrick Rudolph {
2946a9a5b9SPatrick Rudolph namespace gpio
3046a9a5b9SPatrick Rudolph {
3146a9a5b9SPatrick Rudolph
3246a9a5b9SPatrick Rudolph const std::map<std::string, int> biasMap = {
3346a9a5b9SPatrick Rudolph /**< Set bias as is. */
3446a9a5b9SPatrick Rudolph {"AS_IS", 0},
3546a9a5b9SPatrick Rudolph /**< Disable bias. */
3646a9a5b9SPatrick Rudolph {"DISABLE", GPIOD_LINE_REQUEST_FLAG_BIAS_DISABLE},
3746a9a5b9SPatrick Rudolph /**< Enable pull-up. */
3846a9a5b9SPatrick Rudolph {"PULL_UP", GPIOD_LINE_REQUEST_FLAG_BIAS_PULL_UP},
3946a9a5b9SPatrick Rudolph /**< Enable pull-down. */
4046a9a5b9SPatrick Rudolph {"PULL_DOWN", GPIOD_LINE_REQUEST_FLAG_BIAS_PULL_DOWN}};
4146a9a5b9SPatrick Rudolph }
4246a9a5b9SPatrick Rudolph } // namespace phosphor
4346a9a5b9SPatrick Rudolph
main(int argc,char ** argv)4446a9a5b9SPatrick Rudolph int main(int argc, char** argv)
4546a9a5b9SPatrick Rudolph {
4646a9a5b9SPatrick Rudolph boost::asio::io_context io;
4746a9a5b9SPatrick Rudolph
4846a9a5b9SPatrick Rudolph CLI::App app{"Monitor gpio presence status"};
4946a9a5b9SPatrick Rudolph
5046a9a5b9SPatrick Rudolph std::string gpioFileName;
5146a9a5b9SPatrick Rudolph
5246a9a5b9SPatrick Rudolph /* Add an input option */
5346a9a5b9SPatrick Rudolph app.add_option("-c,--config", gpioFileName, "Name of config json file")
5446a9a5b9SPatrick Rudolph ->required()
5546a9a5b9SPatrick Rudolph ->check(CLI::ExistingFile);
5646a9a5b9SPatrick Rudolph
5746a9a5b9SPatrick Rudolph /* Parse input parameter */
5846a9a5b9SPatrick Rudolph try
5946a9a5b9SPatrick Rudolph {
6046a9a5b9SPatrick Rudolph app.parse(argc, argv);
6146a9a5b9SPatrick Rudolph }
6246a9a5b9SPatrick Rudolph catch (const CLI::Error& e)
6346a9a5b9SPatrick Rudolph {
6446a9a5b9SPatrick Rudolph return app.exit(e);
6546a9a5b9SPatrick Rudolph }
6646a9a5b9SPatrick Rudolph
6746a9a5b9SPatrick Rudolph /* Get list of gpio config details from json file */
6846a9a5b9SPatrick Rudolph std::ifstream file(gpioFileName);
6946a9a5b9SPatrick Rudolph if (!file)
7046a9a5b9SPatrick Rudolph {
7146a9a5b9SPatrick Rudolph lg2::error("Failed to open config file: {FILE}", "FILE", gpioFileName);
7246a9a5b9SPatrick Rudolph return -1;
7346a9a5b9SPatrick Rudolph }
7446a9a5b9SPatrick Rudolph
7546a9a5b9SPatrick Rudolph nlohmann::json gpioMonObj;
7646a9a5b9SPatrick Rudolph file >> gpioMonObj;
7746a9a5b9SPatrick Rudolph file.close();
7846a9a5b9SPatrick Rudolph
7946a9a5b9SPatrick Rudolph std::vector<phosphor::gpio::GpioPresence> gpios;
8046a9a5b9SPatrick Rudolph
8146a9a5b9SPatrick Rudolph for (auto& obj : gpioMonObj)
8246a9a5b9SPatrick Rudolph {
8346a9a5b9SPatrick Rudolph /* GPIO Line message */
8446a9a5b9SPatrick Rudolph std::string lineMsg = "GPIO Line ";
8546a9a5b9SPatrick Rudolph
8646a9a5b9SPatrick Rudolph /* GPIO line */
87cf33c593SJayanth Othayoth gpiod_line* line = nullptr;
8846a9a5b9SPatrick Rudolph
8946a9a5b9SPatrick Rudolph /* GPIO line configuration, default to monitor both edge */
90*1c88803fSPatrick Williams struct gpiod_line_request_config config{
91*1c88803fSPatrick Williams "gpio_monitor", GPIOD_LINE_REQUEST_EVENT_BOTH_EDGES, 0};
9246a9a5b9SPatrick Rudolph
9346a9a5b9SPatrick Rudolph /* Pretty name of the inventory object */
9446a9a5b9SPatrick Rudolph std::string name;
9546a9a5b9SPatrick Rudolph
9646a9a5b9SPatrick Rudolph /* Object path under inventory that will be created */
9746a9a5b9SPatrick Rudolph std::string inventory;
9846a9a5b9SPatrick Rudolph
9946a9a5b9SPatrick Rudolph /* List of interfaces to associate to inventory item */
10046a9a5b9SPatrick Rudolph std::vector<std::string> extraInterfaces;
10146a9a5b9SPatrick Rudolph
10246a9a5b9SPatrick Rudolph if (obj.find("LineName") == obj.end())
10346a9a5b9SPatrick Rudolph {
10446a9a5b9SPatrick Rudolph /* If there is no line Name defined then gpio num nd chip
10546a9a5b9SPatrick Rudolph * id must be defined. GpioNum is integer mapping to the
10646a9a5b9SPatrick Rudolph * GPIO key configured by the kernel
10746a9a5b9SPatrick Rudolph */
10846a9a5b9SPatrick Rudolph if (obj.find("GpioNum") == obj.end() ||
10946a9a5b9SPatrick Rudolph obj.find("ChipId") == obj.end())
11046a9a5b9SPatrick Rudolph {
11146a9a5b9SPatrick Rudolph lg2::error("Failed to find line name or gpio number: {FILE}",
11246a9a5b9SPatrick Rudolph "FILE", gpioFileName);
11346a9a5b9SPatrick Rudolph return -1;
11446a9a5b9SPatrick Rudolph }
11546a9a5b9SPatrick Rudolph
11646a9a5b9SPatrick Rudolph std::string chipIdStr = obj["ChipId"].get<std::string>();
11746a9a5b9SPatrick Rudolph int gpioNum = obj["GpioNum"].get<int>();
11846a9a5b9SPatrick Rudolph
11946a9a5b9SPatrick Rudolph lineMsg += chipIdStr + " " + std::to_string(gpioNum);
12046a9a5b9SPatrick Rudolph
12146a9a5b9SPatrick Rudolph /* Get the GPIO line */
12246a9a5b9SPatrick Rudolph line = gpiod_line_get(chipIdStr.c_str(), gpioNum);
12346a9a5b9SPatrick Rudolph }
12446a9a5b9SPatrick Rudolph else
12546a9a5b9SPatrick Rudolph {
12646a9a5b9SPatrick Rudolph /* Find the GPIO line */
12746a9a5b9SPatrick Rudolph std::string lineName = obj["LineName"].get<std::string>();
12846a9a5b9SPatrick Rudolph lineMsg += lineName;
12946a9a5b9SPatrick Rudolph line = gpiod_line_find(lineName.c_str());
13046a9a5b9SPatrick Rudolph }
13146a9a5b9SPatrick Rudolph
132cf33c593SJayanth Othayoth if (line == nullptr)
13346a9a5b9SPatrick Rudolph {
13446a9a5b9SPatrick Rudolph lg2::error("Failed to find the {GPIO}", "GPIO", lineMsg);
13538cd74c8SYang Chen continue;
13646a9a5b9SPatrick Rudolph }
13746a9a5b9SPatrick Rudolph
13846a9a5b9SPatrick Rudolph /* Parse out inventory argument. */
13946a9a5b9SPatrick Rudolph if (obj.find("Inventory") == obj.end())
14046a9a5b9SPatrick Rudolph {
14146a9a5b9SPatrick Rudolph lg2::error("{GPIO}: Inventory path not specified", "GPIO", lineMsg);
14246a9a5b9SPatrick Rudolph return -1;
14346a9a5b9SPatrick Rudolph }
14446a9a5b9SPatrick Rudolph else
14546a9a5b9SPatrick Rudolph {
14646a9a5b9SPatrick Rudolph inventory = obj["Inventory"].get<std::string>();
14746a9a5b9SPatrick Rudolph }
14846a9a5b9SPatrick Rudolph
14946a9a5b9SPatrick Rudolph if (obj.find("Name") == obj.end())
15046a9a5b9SPatrick Rudolph {
15146a9a5b9SPatrick Rudolph lg2::error("{GPIO}: Name path not specified", "GPIO", lineMsg);
15246a9a5b9SPatrick Rudolph return -1;
15346a9a5b9SPatrick Rudolph }
15446a9a5b9SPatrick Rudolph else
15546a9a5b9SPatrick Rudolph {
15646a9a5b9SPatrick Rudolph name = obj["Name"].get<std::string>();
15746a9a5b9SPatrick Rudolph }
15846a9a5b9SPatrick Rudolph
15946a9a5b9SPatrick Rudolph /* Parse optional bias */
16046a9a5b9SPatrick Rudolph if (obj.find("Bias") != obj.end())
16146a9a5b9SPatrick Rudolph {
16246a9a5b9SPatrick Rudolph std::string biasName = obj["Bias"].get<std::string>();
16346a9a5b9SPatrick Rudolph auto findBias = phosphor::gpio::biasMap.find(biasName);
16446a9a5b9SPatrick Rudolph if (findBias == phosphor::gpio::biasMap.end())
16546a9a5b9SPatrick Rudolph {
16646a9a5b9SPatrick Rudolph lg2::error("{GPIO}: Bias unknown: {BIAS}", "GPIO", lineMsg,
16746a9a5b9SPatrick Rudolph "BIAS", biasName);
16846a9a5b9SPatrick Rudolph return -1;
16946a9a5b9SPatrick Rudolph }
17046a9a5b9SPatrick Rudolph
17146a9a5b9SPatrick Rudolph config.flags = findBias->second;
17246a9a5b9SPatrick Rudolph }
17346a9a5b9SPatrick Rudolph
17446a9a5b9SPatrick Rudolph /* Parse optional active level */
17546a9a5b9SPatrick Rudolph if (obj.find("ActiveLow") != obj.end() && obj["ActiveLow"].get<bool>())
17646a9a5b9SPatrick Rudolph {
17746a9a5b9SPatrick Rudolph config.flags |= GPIOD_LINE_REQUEST_FLAG_ACTIVE_LOW;
17846a9a5b9SPatrick Rudolph }
17946a9a5b9SPatrick Rudolph
18046a9a5b9SPatrick Rudolph /* Parse optional extra interfaces */
18146a9a5b9SPatrick Rudolph if (obj.find("ExtraInterfaces") != obj.end())
18246a9a5b9SPatrick Rudolph {
18346a9a5b9SPatrick Rudolph obj.at("ExtraInterfaces").get_to(extraInterfaces);
18446a9a5b9SPatrick Rudolph }
18546a9a5b9SPatrick Rudolph
18646a9a5b9SPatrick Rudolph /* Create a monitor object and let it do all the rest */
18746a9a5b9SPatrick Rudolph gpios.push_back(phosphor::gpio::GpioPresence(
18846a9a5b9SPatrick Rudolph line, config, io, inventory, extraInterfaces, name, lineMsg));
18946a9a5b9SPatrick Rudolph }
19046a9a5b9SPatrick Rudolph io.run();
19146a9a5b9SPatrick Rudolph
19246a9a5b9SPatrick Rudolph return 0;
19346a9a5b9SPatrick Rudolph }
194