blob: 3e6591f5a3c3b9c3fe4dec34e47b68db5b768475 [file] [log] [blame]
/**
* Copyright © 2021 IBM 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.
*/
#include "config.h"
#include "sdbusplus.hpp"
#include <CLI/CLI.hpp>
#include <nlohmann/json.hpp>
#include <sdbusplus/bus.hpp>
#include <chrono>
#include <filesystem>
#include <iomanip>
#include <iostream>
using SDBusPlus = phosphor::fan::util::SDBusPlus;
constexpr auto systemdMgrIface = "org.freedesktop.systemd1.Manager";
constexpr auto systemdPath = "/org/freedesktop/systemd1";
constexpr auto systemdService = "org.freedesktop.systemd1";
constexpr auto phosphorServiceName = "phosphor-fan-control@0.service";
constexpr auto dumpFile = "/tmp/fan_control_dump.json";
enum
{
FAN_NAMES = 0,
PATH_MAP = 1,
IFACES = 2,
METHOD = 3
};
struct DumpQuery
{
std::string section;
std::string name;
std::vector<std::string> properties;
};
/**
* @function extracts fan name from dbus path string (last token where
* delimiter is the / character), with proper bounds checking.
* @param[in] path - D-Bus path
* @return just the fan name.
*/
std::string justFanName(std::string const& path)
{
std::string fanName;
auto itr = path.rfind("/");
if (itr != std::string::npos && itr < path.size())
{
fanName = path.substr(1 + itr);
}
return fanName;
}
/**
* @function produces subtree paths whose names match fan token names.
* @param[in] path - D-Bus path to obtain subtree from
* @param[in] iface - interface to obtain subTreePaths from
* @param[in] fans - label matching tokens to filter by
* @param[in] shortPath - flag to shorten fan token
* @return map of paths by fan name
*/
std::map<std::string, std::vector<std::string>>
getPathsFromIface(const std::string& path, const std::string& iface,
const std::vector<std::string>& fans,
bool shortPath = false)
{
std::map<std::string, std::vector<std::string>> dest;
for (auto& path :
SDBusPlus::getSubTreePathsRaw(SDBusPlus::getBus(), path, iface, 0))
{
for (auto& fan : fans)
{
if (shortPath)
{
if (fan == justFanName(path))
{
dest[fan].push_back(path);
}
}
else if (std::string::npos != path.find(fan + "_"))
{
dest[fan].push_back(path);
}
}
}
return dest;
}
/**
* @function consolidated function to load dbus paths and fan names
*/
auto loadDBusData()
{
auto& bus{SDBusPlus::getBus()};
std::vector<std::string> fanNames;
// paths by D-bus interface,fan name
std::map<std::string, std::map<std::string, std::vector<std::string>>>
pathMap;
std::string method("RPM");
std::map<const std::string, const std::string> interfaces{
{"FanSpeed", "xyz.openbmc_project.Control.FanSpeed"},
{"FanPwm", "xyz.openbmc_project.Control.FanPwm"},
{"SensorValue", "xyz.openbmc_project.Sensor.Value"},
{"Item", "xyz.openbmc_project.Inventory.Item"},
{"OpStatus", "xyz.openbmc_project.State.Decorator.OperationalStatus"}};
std::map<const std::string, const std::string> paths{
{"motherboard",
"/xyz/openbmc_project/inventory/system/chassis/motherboard"},
{"tach", "/xyz/openbmc_project/sensors/fan_tach"}};
// build a list of all fans
for (auto& path : SDBusPlus::getSubTreePathsRaw(bus, paths["tach"],
interfaces["FanSpeed"], 0))
{
// special case where we build the list of fans
auto fan = justFanName(path);
fan = fan.substr(0, fan.rfind("_"));
fanNames.push_back(fan);
}
// retry with PWM mode if none found
if (0 == fanNames.size())
{
method = "PWM";
for (auto& path : SDBusPlus::getSubTreePathsRaw(
bus, paths["tach"], interfaces["FanPwm"], 0))
{
// special case where we build the list of fans
auto fan = justFanName(path);
fan = fan.substr(0, fan.rfind("_"));
fanNames.push_back(fan);
}
}
// load tach sensor paths for each fan
pathMap["tach"] =
getPathsFromIface(paths["tach"], interfaces["SensorValue"], fanNames);
// load inventory Item data for each fan
pathMap["inventory"] = getPathsFromIface(
paths["motherboard"], interfaces["Item"], fanNames, true);
// load operational status data for each fan
pathMap["opstatus"] = getPathsFromIface(
paths["motherboard"], interfaces["OpStatus"], fanNames, true);
return std::make_tuple(fanNames, pathMap, interfaces, method);
}
/**
* @function gets the states of phosphor-fanctl. equivalent to
* "systemctl status phosphor-fan-control@0"
* @return a list of several (sub)states of fanctl (loaded,
* active, running) as well as D-Bus properties representing
* BMC states (bmc state,chassis power state, host state)
*/
std::array<std::string, 6> getStates()
{
using DBusTuple =
std::tuple<std::string, std::string, std::string, std::string,
std::string, std::string, sdbusplus::message::object_path,
uint32_t, std::string, sdbusplus::message::object_path>;
std::array<std::string, 6> ret;
std::vector<std::string> services{phosphorServiceName};
try
{
auto fields{SDBusPlus::callMethodAndRead<std::vector<DBusTuple>>(
systemdService, systemdPath, systemdMgrIface, "ListUnitsByNames",
services)};
if (fields.size() > 0)
{
ret[0] = std::get<2>(fields[0]);
ret[1] = std::get<3>(fields[0]);
ret[2] = std::get<4>(fields[0]);
}
else
{
std::cout << "No units found for systemd service: " << services[0]
<< std::endl;
}
}
catch (const std::exception& e)
{
std::cerr << "Failure retrieving phosphor-fan-control states: "
<< e.what() << std::endl;
}
std::string path("/xyz/openbmc_project/state/bmc0");
std::string iface("xyz.openbmc_project.State.BMC");
ret[3] =
SDBusPlus::getProperty<std::string>(path, iface, "CurrentBMCState");
path = "/xyz/openbmc_project/state/chassis0";
iface = "xyz.openbmc_project.State.Chassis";
ret[4] =
SDBusPlus::getProperty<std::string>(path, iface, "CurrentPowerState");
path = "/xyz/openbmc_project/state/host0";
iface = "xyz.openbmc_project.State.Host";
ret[5] =
SDBusPlus::getProperty<std::string>(path, iface, "CurrentHostState");
return ret;
}
/**
* @function helper to determine interface type from a given control method
*/
std::string ifaceTypeFromMethod(const std::string& method)
{
return (method == "RPM" ? "FanSpeed" : "FanPwm");
}
/**
* @function performs the "status" command from the cmdline.
* get states and sensor data and output to the console
*/
void status()
{
using std::cout;
using std::endl;
using std::setw;
auto busData = loadDBusData();
auto& method = std::get<METHOD>(busData);
std::string property;
// get the state,substate of fan-control and obmc
auto states = getStates();
// print the header
cout << "Fan Control Service State : " << states[0] << ", " << states[1]
<< "(" << states[2] << ")" << endl;
cout << endl;
cout << "CurrentBMCState : " << states[3] << endl;
cout << "CurrentPowerState : " << states[4] << endl;
cout << "CurrentHostState : " << states[5] << endl;
cout << endl;
cout << "FAN "
<< "TARGET(" << method << ") FEEDBACKS(RPM) PRESENT"
<< " FUNCTIONAL" << endl;
cout << "==============================================================="
<< endl;
auto& fanNames{std::get<FAN_NAMES>(busData)};
auto& pathMap{std::get<PATH_MAP>(busData)};
auto& interfaces{std::get<IFACES>(busData)};
for (auto& fan : fanNames)
{
cout << setw(8) << std::left << fan << std::right << setw(13);
// get the target RPM
property = "Target";
cout << SDBusPlus::getProperty<uint64_t>(
pathMap["tach"][fan][0],
interfaces[ifaceTypeFromMethod(method)], property)
<< setw(19);
// get the sensor RPM
property = "Value";
std::ostringstream output;
int numRotors = pathMap["tach"][fan].size();
// print tach readings for each rotor
for (auto& path : pathMap["tach"][fan])
{
output << SDBusPlus::getProperty<double>(
path, interfaces["SensorValue"], property);
// dont print slash on last rotor
if (--numRotors)
output << "/";
}
cout << output.str() << setw(10);
// print the Present property
property = "Present";
auto itFan = pathMap["inventory"].find(fan);
if (itFan != pathMap["inventory"].end())
{
for (auto& path : itFan->second)
{
try
{
cout << std::boolalpha
<< SDBusPlus::getProperty<bool>(
path, interfaces["Item"], property);
}
catch (const phosphor::fan::util::DBusError&)
{
cout << "Unknown";
}
}
}
else
{
cout << "Unknown";
}
cout << setw(13);
// and the functional property
property = "Functional";
itFan = pathMap["opstatus"].find(fan);
if (itFan != pathMap["opstatus"].end())
{
for (auto& path : itFan->second)
{
try
{
cout << std::boolalpha
<< SDBusPlus::getProperty<bool>(
path, interfaces["OpStatus"], property);
}
catch (const phosphor::fan::util::DBusError&)
{
cout << "Unknown";
}
}
}
else
{
cout << "Unknown";
}
cout << endl;
}
}
/**
* @function print target RPM/PWM and tach readings from each fan
*/
void get()
{
using std::cout;
using std::endl;
using std::setw;
auto busData = loadDBusData();
auto& fanNames{std::get<FAN_NAMES>(busData)};
auto& pathMap{std::get<PATH_MAP>(busData)};
auto& interfaces{std::get<IFACES>(busData)};
auto& method = std::get<METHOD>(busData);
std::string property;
// print the header
cout << "TARGET SENSOR" << setw(11) << "TARGET(" << method
<< ") FEEDBACK SENSOR FEEDBACK(RPM)" << endl;
cout << "==============================================================="
<< endl;
for (auto& fan : fanNames)
{
if (pathMap["tach"][fan].size() == 0)
continue;
// print just the sensor name
auto shortPath = pathMap["tach"][fan][0];
shortPath = justFanName(shortPath);
cout << setw(13) << std::left << shortPath << std::right << setw(15);
// print its target RPM/PWM
property = "Target";
cout << SDBusPlus::getProperty<uint64_t>(
pathMap["tach"][fan][0], interfaces[ifaceTypeFromMethod(method)],
property);
// print readings for each rotor
property = "Value";
auto indent = 0;
for (auto& path : pathMap["tach"][fan])
{
cout << setw(18 + indent) << justFanName(path) << setw(17)
<< SDBusPlus::getProperty<double>(
path, interfaces["SensorValue"], property)
<< endl;
if (0 == indent)
indent = 28;
}
}
}
/**
* @function set fan[s] to a target RPM
*/
void set(uint64_t target, std::vector<std::string>& fanList)
{
auto busData = loadDBusData();
auto& bus{SDBusPlus::getBus()};
auto& pathMap{std::get<PATH_MAP>(busData)};
auto& interfaces{std::get<IFACES>(busData)};
auto& method = std::get<METHOD>(busData);
std::string ifaceType(method == "RPM" ? "FanSpeed" : "FanPwm");
// stop the fan-control service
SDBusPlus::callMethodAndRead<sdbusplus::message::object_path>(
systemdService, systemdPath, systemdMgrIface, "StopUnit",
phosphorServiceName, "replace");
if (fanList.size() == 0)
{
fanList = std::get<FAN_NAMES>(busData);
}
for (auto& fan : fanList)
{
try
{
auto paths(pathMap["tach"].find(fan));
if (pathMap["tach"].end() == paths)
{
// try again, maybe it was a sensor name instead of a fan name
for (const auto& [fanName, sensors] : pathMap["tach"])
{
for (const auto& path : sensors)
{
std::string sensor(path.substr(path.rfind("/")));
if (sensor.size() > 0)
{
sensor = sensor.substr(1);
if (sensor == fan)
{
paths = pathMap["tach"].find(fanName);
break;
}
}
}
}
}
if (pathMap["tach"].end() == paths)
{
std::cout << "Could not find tach path for fan: " << fan
<< std::endl;
continue;
}
// set the target RPM
SDBusPlus::setProperty<uint64_t>(bus, paths->second[0],
interfaces[ifaceType], "Target",
std::move(target));
}
catch (const phosphor::fan::util::DBusPropertyError& e)
{
std::cerr << "Cannot set target rpm for " << fan
<< " caught D-Bus exception: " << e.what() << std::endl;
}
}
}
/**
* @function restart fan-control to allow it to manage fan speeds
*/
void resume()
{
try
{
auto retval =
SDBusPlus::callMethodAndRead<sdbusplus::message::object_path>(
systemdService, systemdPath, systemdMgrIface, "StartUnit",
phosphorServiceName, "replace");
}
catch (const phosphor::fan::util::DBusMethodError& e)
{
std::cerr << "Unable to start fan control: " << e.what() << std::endl;
}
}
/**
* @function force reload of control files by sending HUP signal
*/
void reload()
{
try
{
SDBusPlus::callMethod(systemdService, systemdPath, systemdMgrIface,
"KillUnit", phosphorServiceName, "main", SIGHUP);
}
catch (const phosphor::fan::util::DBusPropertyError& e)
{
std::cerr << "Unable to reload configuration files: " << e.what()
<< std::endl;
}
}
/**
* @function dump the FlightRecorder log data
*/
void dumpFanControl()
{
namespace fs = std::filesystem;
try
{
// delete existing file
if (fs::exists(dumpFile))
{
std::filesystem::remove(dumpFile);
}
SDBusPlus::callMethod(systemdService, systemdPath, systemdMgrIface,
"KillUnit", phosphorServiceName, "main", SIGUSR1);
bool done = false;
do
{
// wait for file to be detected
sleep(1);
if (fs::exists(dumpFile))
{
try
{
auto unused{nlohmann::json::parse(std::ifstream{dumpFile})};
done = true;
}
catch (...)
{
// TODO: maybe have a max-retries counter and fail after N
// tries
}
}
} while (!done);
std::cout << "Fan control dump written to: " << dumpFile << std::endl;
}
catch (const phosphor::fan::util::DBusPropertyError& e)
{
std::cerr << "Unable to dump fan control: " << e.what() << std::endl;
}
}
/**
* @function Query items in the dump file
*/
void queryDumpFile(const DumpQuery& dq)
{
nlohmann::json output;
std::ifstream file{dumpFile};
if (!file.good())
{
std::cerr << "Unable to open dump file, please run 'fanctl dump'.\n";
return;
}
auto dumpData = nlohmann::json::parse(file);
if (!dumpData.contains(dq.section))
{
std::cerr << "Error: Dump file does not contain " << dq.section
<< " section"
<< "\n";
return;
}
const auto& section = dumpData.at(dq.section);
if (section.is_array())
{
for (const auto& entry : section)
{
if (!entry.is_string() || dq.name.empty() ||
(entry.get<std::string>().find(dq.name) != std::string::npos))
{
output[dq.section].push_back(entry);
}
}
std::cout << std::setw(4) << output << "\n";
return;
}
for (const auto& [key1, values1] : section.items())
{
if (dq.name.empty() || (key1.find(dq.name) != std::string::npos))
{
// If no properties specified, print the whole JSON value
if (dq.properties.empty())
{
output[key1] = values1;
continue;
}
// Look for properties both one and two levels down.
// Future improvement: Use recursion.
for (const auto& [key2, values2] : values1.items())
{
for (const auto& prop : dq.properties)
{
if (prop == key2)
{
output[key1][prop] = values2;
}
}
for (const auto& [key3, values3] : values2.items())
{
for (const auto& prop : dq.properties)
{
if (prop == key3)
{
output[key1][prop] = values3;
}
}
}
}
}
}
if (!output.empty())
{
std::cout << std::setw(4) << output << "\n";
}
}
/**
* @function setup the CLI object to accept all options
*/
void initCLI(CLI::App& app, uint64_t& target, std::vector<std::string>& fanList,
DumpQuery& dq)
{
app.set_help_flag("-h,--help", "Print this help page and exit.");
// App requires only 1 subcommand to be given
app.require_subcommand(1);
// This represents the command given
auto commands = app.add_option_group("Commands");
// status method
std::string strHelp("Prints fan target/tach readings, present/functional "
"states, and fan-monitor/BMC/Power service status");
auto cmdStatus = commands->add_subcommand("status", strHelp);
cmdStatus->set_help_flag("-h, --help", strHelp);
cmdStatus->require_option(0);
// get method
strHelp = "Get the current fan target and feedback speeds for all rotors";
auto cmdGet = commands->add_subcommand("get", strHelp);
cmdGet->set_help_flag("-h, --help", strHelp);
cmdGet->require_option(0);
// set method
strHelp = "Set target (all rotors) for one-or-more fans";
auto cmdSet = commands->add_subcommand("set", strHelp);
strHelp = R"(set <TARGET> [TARGET SENSOR(S)]
<TARGET>
- RPM/PWM target to set the fans
[TARGET SENSOR LIST]
- list of target sensors to set)";
cmdSet->set_help_flag("-h, --help", strHelp);
cmdSet->add_option("target", target, "RPM/PWM target to set the fans");
cmdSet->add_option(
"fan list", fanList,
"[optional] list of 1+ fans to set target RPM/PWM (default: all)");
cmdSet->require_option();
#ifdef CONTROL_USE_JSON
strHelp = "Reload phosphor-fan configuration files";
auto cmdReload = commands->add_subcommand("reload", strHelp);
cmdReload->set_help_flag("-h, --help", strHelp);
cmdReload->require_option(0);
#endif
strHelp = "Resume running phosphor-fan-control";
auto cmdResume = commands->add_subcommand("resume", strHelp);
cmdResume->set_help_flag("-h, --help", strHelp);
cmdResume->require_option(0);
#ifdef CONTROL_USE_JSON
// Dump method
auto cmdDump = commands->add_subcommand(
"dump", "Dump the FlightRecorder diagnostic log");
cmdDump->set_help_flag("-h, --help",
"Dump the FlightRecorder diagnostic log");
cmdDump->require_option(0);
// Query dump
auto cmdDumpQuery =
commands->add_subcommand("query_dump", "Query the dump file");
cmdDumpQuery->set_help_flag("-h, --help", "Query the dump file");
cmdDumpQuery
->add_option("-s, --section", dq.section, "Dump file section name")
->required();
cmdDumpQuery->add_option("-n, --name", dq.name,
"Optional dump file entry name (or substring)");
cmdDumpQuery->add_option("-p, --properties", dq.properties,
"Optional list of dump file property names");
#endif
}
/**
* @function main entry point for the application
*/
int main(int argc, char* argv[])
{
auto rc = 0;
uint64_t target{0U};
std::vector<std::string> fanList;
DumpQuery dq;
try
{
CLI::App app{"Manually control, get fan tachs, view status, and resume "
"automatic control of all fans within a chassis. Full "
"documentation can be found at the readme:\n"
"https://github.com/openbmc/phosphor-fan-presence/tree/"
"master/docs/control/fanctl"};
initCLI(app, target, fanList, dq);
CLI11_PARSE(app, argc, argv);
if (app.got_subcommand("get"))
{
get();
}
else if (app.got_subcommand("set"))
{
set(target, fanList);
}
#ifdef CONTROL_USE_JSON
else if (app.got_subcommand("reload"))
{
reload();
}
#endif
else if (app.got_subcommand("resume"))
{
resume();
}
else if (app.got_subcommand("status"))
{
status();
}
else if (app.got_subcommand("dump"))
{
#ifdef CONTROL_USE_JSON
dumpFanControl();
#else
std::ofstream(dumpFile)
<< "{\n\"msg\": \"Unable to create dump on "
"non-JSON config based system\"\n}";
#endif
}
#ifdef CONTROL_USE_JSON
else if (app.got_subcommand("query_dump"))
{
queryDumpFile(dq);
}
#endif
}
catch (const std::exception& e)
{
rc = -1;
std::cerr << argv[0] << " failed: " << e.what() << std::endl;
}
return rc;
}