fanctl: Add option to display sensors
Add a new 'sensors' option to fanctl to display sensor values for debug,
which the code gets straight from D-Bus.
The help:
$ fanctl sensors -h
```
Retrieve sensor values
Usage: fanctl sensors [OPTIONS]
Options:
-h,--help Retrieve sensor values
-t,--type TEXT Only show sensors of this type (i.e. 'temperature'). Optional
-n,--name TEXT Only show sensors with this string in the name. Optional
-v,--verbose Verbose: Use sensor object path for the name
```
Example output:
$ fanctl sensors
Altitude: 110.901
Ambient_0_Temp: 22.562
Ambient_1_Temp: 22.604
Ambient_2_Temp: 22.4047
Ambient_Virtual_Temp: 22.562
Battery_Voltage: 3.144
NVMe_1_Temp: 31
NVMe_2_Temp: 31
NVMe_JBOF_Card_C10_Local_Temp: 33.938
NVMe_JBOF_Card_C10_Temp: 44
...
Display the object path for the sensor name:
```
$ fanctl sensors -v
/xyz/openbmc_project/sensors/altitude/Altitude: 110.901
/xyz/openbmc_project/sensors/current/ps0_output_current: 20.375
/xyz/openbmc_project/sensors/current/ps1_output_current: 18.218
/xyz/openbmc_project/sensors/current/vcs_p0_dcm0_rail_iout: 1.75
/xyz/openbmc_project/sensors/current/vcs_p0_dcm0_rail_iout_peak: 6.5
/xyz/openbmc_project/sensors/current/vcs_p0_dcm0_rail_iout_valley: 0
/xyz/openbmc_project/sensors/current/vcs_p0_dcm1_rail_iout: 0.75
/xyz/openbmc_project/sensors/current/vcs_p0_dcm1_rail_iout_peak: 3
/xyz/openbmc_project/sensors/current/vcs_p0_dcm1_rail_iout_valley: 0
/xyz/openbmc_project/sensors/current/vcs_p1_dcm0_rail_iout: 2.75
...
```
Display only voltage sensors:
```
$ fanctl sensors -t voltage
Battery_Voltage: 3.144
ps0_input_voltage: 207
ps0_input_voltage_rating: 220
ps0_output_voltage: 12.296
...
```
Display only sensors that contain the substring 'core':
$ fanctl sensors -n core
```
proc0_core0_0_temp: 44
proc0_core0_1_temp: 44
proc0_core10_0_temp: 45
proc0_core10_1_temp: 45
...
```
Signed-off-by: Matt Spinler <spinler@us.ibm.com>
Change-Id: Ib93f00c457a6972e9eba1f74adca904baa2bf28c
diff --git a/control/fanctl.cpp b/control/fanctl.cpp
index 9789c24..9c7b1de 100644
--- a/control/fanctl.cpp
+++ b/control/fanctl.cpp
@@ -22,7 +22,6 @@
#include <nlohmann/json.hpp>
#include <sdbusplus/bus.hpp>
-#include <chrono>
#include <filesystem>
#include <iomanip>
#include <iostream>
@@ -51,6 +50,21 @@
bool dump{false};
};
+struct SensorOpts
+{
+ std::string type;
+ std::string name;
+ bool verbose{false};
+};
+
+struct SensorOutput
+{
+ std::string name;
+ double value;
+ bool functional;
+ bool available;
+};
+
/**
* @function extracts fan name from dbus path string (last token where
* delimiter is the / character), with proper bounds checking.
@@ -663,10 +677,197 @@
}
/**
+ * @function Get the sensor type based on the sensor name
+ *
+ * @param sensor The sensor object path
+ */
+std::string getSensorType(const std::string& sensor)
+{
+ // Get type from /xyz/openbmc_project/sensors/<type>/<name>
+ try
+ {
+ auto type =
+ sensor.substr(std::string{"/xyz/openbmc_project/sensors/"}.size());
+ return type.substr(0, type.find_first_of('/'));
+ }
+ catch (const std::exception& e)
+ {
+ std::cerr << "Failed extracting type from sensor " << sensor << ": "
+ << e.what() << "\n";
+ }
+ return std::string{};
+}
+
+/**
+ * @function Print the sensors passed in
+ *
+ * @param sensors The sensors to print
+ */
+void printSensors(const std::vector<SensorOutput>& sensors)
+{
+ size_t maxNameSize = 0;
+
+ std::ranges::for_each(sensors, [&maxNameSize](const auto& s) {
+ maxNameSize = std::max(maxNameSize, s.name.size());
+ });
+
+ std::ranges::for_each(sensors, [maxNameSize](const auto& sensor) {
+ auto nameField = sensor.name + ':';
+ std::cout << std::left << std::setw(maxNameSize + 2) << nameField
+ << sensor.value;
+ if (!sensor.functional)
+ {
+ std::cout << " (Functional=false)";
+ }
+
+ if (!sensor.available)
+ {
+ std::cout << " (Available=false)";
+ }
+ std::cout << "\n";
+ });
+}
+
+/**
+ * @function Extracts the sensor out of the GetManagedObjects output
+ * for the one object path passed in.
+ *
+ * @param object The GetManagedObjects output for a single object path
+ * @param opts The sensor options
+ * @param[out] sensors Filled in with the sensor data
+ */
+void extractSensorData(const auto& object, const SensorOpts& opts,
+ std::vector<SensorOutput>& sensors)
+{
+ auto it = object.second.find("xyz.openbmc_project.Sensor.Value");
+ if (it == object.second.end())
+ {
+ return;
+ }
+ auto value = std::get<double>(it->second.at("Value"));
+
+ // Use the full D-Bus path of the sensor for the name if verbose
+ std::string name = object.first.str;
+ name = name.substr(name.find_last_of('/') + 1);
+ std::string printName = name;
+ if (opts.verbose)
+ {
+ printName = object.first.str;
+ }
+
+ // Apply the name filter
+ if (!opts.name.empty())
+ {
+ if (!name.contains(opts.name))
+ {
+ return;
+ }
+ }
+
+ // Apply the type filter
+ if (!opts.type.empty())
+ {
+ if (opts.type != getSensorType(object.first.str))
+ {
+ return;
+ }
+ }
+
+ bool functional = true;
+ it = object.second.find(
+ "xyz.openbmc_project.State.Decorator.OperationalStatus");
+ if (it != object.second.end())
+ {
+ functional = std::get<bool>(it->second.at("Functional"));
+ }
+
+ bool available = true;
+ it = object.second.find("xyz.openbmc_project.State.Decorator.Availability");
+ if (it != object.second.end())
+ {
+ available = std::get<bool>(it->second.at("Available"));
+ }
+
+ sensors.emplace_back(printName, value, functional, available);
+}
+
+/**
+ * @function Call GetManagedObjects on all sensor object managers and then
+ * print the sensor values.
+ *
+ * @param sensorManagers map<service, path> of sensor ObjectManagers
+ * @param opts The sensor options
+ */
+void readSensorsAndPrint(std::map<std::string, std::string>& sensorManagers,
+ const SensorOpts& opts)
+{
+ std::vector<SensorOutput> sensors;
+
+ using PropertyVariantType =
+ std::variant<bool, int32_t, int64_t, double, std::string>;
+
+ std::ranges::for_each(sensorManagers, [&opts, &sensors](const auto& entry) {
+ auto values = SDBusPlus::getManagedObjects<PropertyVariantType>(
+ SDBusPlus::getBus(), entry.first, entry.second);
+
+ // Pull out the sensor details
+ std::ranges::for_each(values, [&opts, &sensors](const auto& sensor) {
+ extractSensorData(sensor, opts, sensors);
+ });
+ });
+
+ std::ranges::sort(sensors, [](const auto& left, const auto& right) {
+ return left.name < right.name;
+ });
+
+ printSensors(sensors);
+}
+
+/**
+ * @function Prints sensor values
+ *
+ * @param opts The sensor options
+ */
+void displaySensors(const SensorOpts& opts)
+{
+ // Find the services that provide sensors
+ auto sensorObjects = SDBusPlus::getSubTreeRaw(
+ SDBusPlus::getBus(), "/", "xyz.openbmc_project.Sensor.Value", 0);
+
+ std::set<std::string> sensorServices;
+
+ std::ranges::for_each(sensorObjects, [&sensorServices](const auto& object) {
+ sensorServices.insert(object.second.begin()->first);
+ });
+
+ // Find the ObjectManagers for those services
+ auto objectManagers = SDBusPlus::getSubTreeRaw(
+ SDBusPlus::getBus(), "/", "org.freedesktop.DBus.ObjectManager", 0);
+
+ std::map<std::string, std::string> managers;
+
+ std::ranges::for_each(
+ objectManagers, [&sensorServices, &managers](const auto& object) {
+ // Check every service on this path
+ std::ranges::for_each(
+ object.second, [&managers, path = object.first,
+ &sensorServices](const auto& entry) {
+ // Check if this service provides sensors
+ if (std::ranges::contains(sensorServices, entry.first))
+ {
+ managers[entry.first] = path;
+ }
+ });
+ });
+
+ readSensorsAndPrint(managers, opts);
+}
+
+/**
* @function setup the CLI object to accept all options
*/
void initCLI(CLI::App& app, uint64_t& target, std::vector<std::string>& fanList,
- [[maybe_unused]] DumpQuery& dq)
+ [[maybe_unused]] DumpQuery& dq, SensorOpts& sensorOpts)
{
app.set_help_flag("-h,--help", "Print this help page and exit.");
@@ -738,6 +939,18 @@
cmdDumpQuery->add_flag("-d, --dump", dq.dump,
"Force a dump before the query");
#endif
+
+ auto cmdSensors =
+ commands->add_subcommand("sensors", "Retrieve sensor values");
+ cmdSensors->set_help_flag("-h, --help", "Retrieve sensor values");
+ cmdSensors->add_option(
+ "-t, --type", sensorOpts.type,
+ "Only show sensors of this type (i.e. 'temperature'). Optional");
+ cmdSensors->add_option(
+ "-n, --name", sensorOpts.name,
+ "Only show sensors with this string in the name. Optional");
+ cmdSensors->add_flag("-v, --verbose", sensorOpts.verbose,
+ "Verbose: Use sensor object path for the name");
}
/**
@@ -749,6 +962,7 @@
uint64_t target{0U};
std::vector<std::string> fanList;
DumpQuery dq;
+ SensorOpts sensorOpts;
try
{
@@ -758,7 +972,7 @@
"https://github.com/openbmc/phosphor-fan-presence/tree/"
"master/docs/control/fanctl"};
- initCLI(app, target, fanList, dq);
+ initCLI(app, target, fanList, dq, sensorOpts);
CLI11_PARSE(app, argc, argv);
@@ -804,6 +1018,10 @@
queryDumpFile(dq);
}
#endif
+ else if (app.got_subcommand("sensors"))
+ {
+ displaySensors(sensorOpts);
+ }
}
catch (const std::exception& e)
{