blob: ff8bf8e75f4809685939c2e5570268b0c23c0a4d [file] [log] [blame]
#pragma once
#include "dbus-sensor_config.h"
#include <SensorPaths.hpp>
#include <Thresholds.hpp>
#include <Utils.hpp>
#include <sdbusplus/asio/object_server.hpp>
#include <limits>
#include <memory>
#include <string>
#include <vector>
constexpr size_t sensorFailedPollTimeMs = 5000;
// Enable useful logging with sensor instrumentation
// This is intentionally not DEBUG, avoid clash with usage in .cpp files
constexpr bool enableInstrumentation = false;
constexpr const char* sensorValueInterface = "xyz.openbmc_project.Sensor.Value";
constexpr const char* valueMutabilityInterfaceName =
"xyz.openbmc_project.Sensor.ValueMutability";
constexpr const char* availableInterfaceName =
"xyz.openbmc_project.State.Decorator.Availability";
constexpr const char* operationalInterfaceName =
"xyz.openbmc_project.State.Decorator.OperationalStatus";
constexpr const size_t errorThreshold = 5;
struct SensorInstrumentation
{
// These are for instrumentation for debugging
int numCollectsGood = 0;
int numCollectsMiss = 0;
int numStreakGreats = 0;
int numStreakMisses = 0;
double minCollected = 0.0;
double maxCollected = 0.0;
};
struct Sensor
{
Sensor(const std::string& name,
std::vector<thresholds::Threshold>&& thresholdData,
const std::string& configurationPath, const std::string& objectType,
bool isSettable, bool isMutable, const double max, const double min,
std::shared_ptr<sdbusplus::asio::connection>& conn,
PowerState readState = PowerState::always) :
name(sensor_paths::escapePathForDbus(name)),
configurationPath(configurationPath), objectType(objectType),
isSensorSettable(isSettable), isValueMutable(isMutable), maxValue(max),
minValue(min), thresholds(std::move(thresholdData)),
hysteresisTrigger((max - min) * 0.01),
hysteresisPublish((max - min) * 0.0001), dbusConnection(conn),
readState(readState), errCount(0),
instrumentation(enableInstrumentation
? std::make_unique<SensorInstrumentation>()
: nullptr)
{}
virtual ~Sensor() = default;
virtual void checkThresholds(void) = 0;
std::string name;
std::string configurationPath;
std::string objectType;
bool isSensorSettable;
/* A flag indicates if properties of xyz.openbmc_project.Sensor.Value
* interface are mutable. If mutable, then
* xyz.openbmc_project.Sensor.ValueMutability interface will be
* instantiated.
*/
bool isValueMutable;
double maxValue;
double minValue;
std::vector<thresholds::Threshold> thresholds;
std::shared_ptr<sdbusplus::asio::dbus_interface> sensorInterface;
std::shared_ptr<sdbusplus::asio::dbus_interface> thresholdInterfaceWarning;
std::shared_ptr<sdbusplus::asio::dbus_interface> thresholdInterfaceCritical;
std::shared_ptr<sdbusplus::asio::dbus_interface> association;
std::shared_ptr<sdbusplus::asio::dbus_interface> availableInterface;
std::shared_ptr<sdbusplus::asio::dbus_interface> operationalInterface;
std::shared_ptr<sdbusplus::asio::dbus_interface> valueMutabilityInterface;
double value = std::numeric_limits<double>::quiet_NaN();
double rawValue = std::numeric_limits<double>::quiet_NaN();
bool overriddenState = false;
bool internalSet = false;
double hysteresisTrigger;
double hysteresisPublish;
std::shared_ptr<sdbusplus::asio::connection> dbusConnection;
PowerState readState;
size_t errCount;
std::unique_ptr<SensorInstrumentation> instrumentation;
// This member variable provides a hook that can be used to receive
// notification whenever this Sensor's value is externally set via D-Bus.
// If interested, assign your own lambda to this variable, during
// construction of your Sensor subclass. See ExternalSensor for example.
std::function<void()> externalSetHook;
struct ThresholdProperty
{
thresholds::Level level;
thresholds::Direction direction;
uint8_t sevOrder;
const char* levelProperty;
const char* alarmProperty;
const char* dirOrder;
};
constexpr static std::array<ThresholdProperty, 4> thresProp = {
{{thresholds::Level::WARNING, thresholds::Direction::HIGH, 0,
"WarningHigh", "WarningAlarmHigh", "greater than"},
{thresholds::Level::WARNING, thresholds::Direction::LOW, 0,
"WarningLow", "WarningAlarmLow", "less than"},
{thresholds::Level::CRITICAL, thresholds::Direction::HIGH, 1,
"CriticalHigh", "CriticalAlarmHigh", "greater than"},
{thresholds::Level::CRITICAL, thresholds::Direction::LOW, 1,
"CriticalLow", "CriticalAlarmLow", "less than"}}};
void updateInstrumentation(double readValue)
{
// Do nothing if this feature is not enabled
if constexpr (!enableInstrumentation)
{
return;
}
if (!instrumentation)
{
return;
}
// Save some typing
auto& inst = *instrumentation;
// Show constants if first reading (even if unsuccessful)
if ((inst.numCollectsGood == 0) && (inst.numCollectsMiss == 0))
{
std::cerr << "Sensor " << name << ": Configuration min=" << minValue
<< ", max=" << maxValue << ", type=" << objectType
<< ", path=" << configurationPath << "\n";
}
// Sensors can use "nan" to indicate unavailable reading
if (!std::isfinite(readValue))
{
// Only show this if beginning a new streak
if (inst.numStreakMisses == 0)
{
std::cerr << "Sensor " << name
<< ": Missing reading, Reading counts good="
<< inst.numCollectsGood
<< ", miss=" << inst.numCollectsMiss
<< ", Prior good streak=" << inst.numStreakGreats
<< "\n";
}
inst.numStreakGreats = 0;
++(inst.numCollectsMiss);
++(inst.numStreakMisses);
return;
}
// Only show this if beginning a new streak and not the first time
if ((inst.numStreakGreats == 0) && (inst.numCollectsGood != 0))
{
std::cerr << "Sensor " << name
<< ": Recovered reading, Reading counts good="
<< inst.numCollectsGood
<< ", miss=" << inst.numCollectsMiss
<< ", Prior miss streak=" << inst.numStreakMisses << "\n";
}
// Initialize min/max if the first successful reading
if (inst.numCollectsGood == 0)
{
std::cerr << "Sensor " << name << ": First reading=" << readValue
<< "\n";
inst.minCollected = readValue;
inst.maxCollected = readValue;
}
inst.numStreakMisses = 0;
++(inst.numCollectsGood);
++(inst.numStreakGreats);
// Only provide subsequent output if new min/max established
if (readValue < inst.minCollected)
{
std::cerr << "Sensor " << name << ": Lowest reading=" << readValue
<< "\n";
inst.minCollected = readValue;
}
if (readValue > inst.maxCollected)
{
std::cerr << "Sensor " << name << ": Highest reading=" << readValue
<< "\n";
inst.maxCollected = readValue;
}
}
int setSensorValue(const double& newValue, double& oldValue)
{
if (!internalSet)
{
if (insecureSensorOverride == false)
{ // insecure sesnor override.
if (isSensorSettable == false)
{ // sensor is not settable.
if (getManufacturingMode() == false)
{ // manufacture mode is not enable.
std::cerr << "Sensor " << name
<< ": Not allowed to set property value.\n";
return -EACCES;
}
}
}
oldValue = newValue;
overriddenState = true;
// check thresholds for external set
value = newValue;
checkThresholds();
// Trigger the hook, as an external set has just happened
if (externalSetHook)
{
externalSetHook();
}
}
else if (!overriddenState)
{
oldValue = newValue;
}
return 1;
}
void
setInitialProperties(std::shared_ptr<sdbusplus::asio::connection>& conn,
const std::string& unit,
const std::string& label = std::string(),
size_t thresholdSize = 0)
{
if (readState == PowerState::on || readState == PowerState::biosPost)
{
setupPowerMatch(conn);
}
createAssociation(association, configurationPath);
sensorInterface->register_property("Unit", unit);
sensorInterface->register_property("MaxValue", maxValue);
sensorInterface->register_property("MinValue", minValue);
sensorInterface->register_property(
"Value", value, [&](const double& newValue, double& oldValue) {
return setSensorValue(newValue, oldValue);
});
for (auto& threshold : thresholds)
{
if (std::isnan(threshold.hysteresis))
{
threshold.hysteresis = hysteresisTrigger;
}
std::shared_ptr<sdbusplus::asio::dbus_interface> iface;
if (threshold.level == thresholds::Level::CRITICAL)
{
iface = thresholdInterfaceCritical;
}
else if (threshold.level == thresholds::Level::WARNING)
{
iface = thresholdInterfaceWarning;
}
else
{
std::cerr << "Unknown threshold level"
<< static_cast<int>(threshold.level) << "\n";
continue;
}
if (!iface)
{
std::cout << "trying to set uninitialized interface\n";
continue;
}
std::string level =
propertyLevel(threshold.level, threshold.direction);
std::string alarm =
propertyAlarm(threshold.level, threshold.direction);
if ((level.empty()) || (alarm.empty()))
{
continue;
}
size_t thresSize =
label.empty() ? thresholds.size() : thresholdSize;
iface->register_property(
level, threshold.value,
[&, label, thresSize](const double& request, double& oldValue) {
oldValue = request; // todo, just let the config do this?
threshold.value = request;
thresholds::persistThreshold(configurationPath, objectType,
threshold, conn, thresSize,
label);
// Invalidate previously remembered value,
// so new thresholds will be checked during next update,
// even if sensor reading remains unchanged.
value = std::numeric_limits<double>::quiet_NaN();
// Although tempting, don't call checkThresholds() from here
// directly. Let the regular sensor monitor call the same
// using updateValue(), which can check conditions like
// poweron, etc., before raising any event.
return 1;
});
iface->register_property(alarm, false);
}
if (!sensorInterface->initialize())
{
std::cerr << "error initializing value interface\n";
}
if (thresholdInterfaceWarning &&
!thresholdInterfaceWarning->initialize(true))
{
std::cerr << "error initializing warning threshold interface\n";
}
if (thresholdInterfaceCritical &&
!thresholdInterfaceCritical->initialize(true))
{
std::cerr << "error initializing critical threshold interface\n";
}
if (isValueMutable)
{
valueMutabilityInterface =
std::make_shared<sdbusplus::asio::dbus_interface>(
conn, sensorInterface->get_object_path(),
valueMutabilityInterfaceName);
valueMutabilityInterface->register_property("Mutable", true);
if (!valueMutabilityInterface->initialize())
{
std::cerr
<< "error initializing sensor value mutability interface\n";
valueMutabilityInterface = nullptr;
}
}
if (!availableInterface)
{
availableInterface =
std::make_shared<sdbusplus::asio::dbus_interface>(
conn, sensorInterface->get_object_path(),
availableInterfaceName);
availableInterface->register_property(
"Available", true, [this](const bool propIn, bool& old) {
if (propIn == old)
{
return 1;
}
old = propIn;
if (!propIn)
{
updateValue(std::numeric_limits<double>::quiet_NaN());
}
return 1;
});
availableInterface->initialize();
}
if (!operationalInterface)
{
operationalInterface =
std::make_shared<sdbusplus::asio::dbus_interface>(
conn, sensorInterface->get_object_path(),
operationalInterfaceName);
operationalInterface->register_property("Functional", true);
operationalInterface->initialize();
}
}
std::string propertyLevel(const thresholds::Level lev,
const thresholds::Direction dir)
{
for (ThresholdProperty prop : thresProp)
{
if ((prop.level == lev) && (prop.direction == dir))
{
return prop.levelProperty;
}
}
return "";
}
std::string propertyAlarm(const thresholds::Level lev,
const thresholds::Direction dir)
{
for (ThresholdProperty prop : thresProp)
{
if ((prop.level == lev) && (prop.direction == dir))
{
return prop.alarmProperty;
}
}
return "";
}
bool readingStateGood()
{
if (readState == PowerState::on && !isPowerOn())
{
return false;
}
if (readState == PowerState::biosPost &&
(!hasBiosPost() || !isPowerOn()))
{
return false;
}
return true;
}
void markFunctional(bool isFunctional)
{
if (operationalInterface)
{
operationalInterface->set_property("Functional", isFunctional);
}
if (isFunctional)
{
errCount = 0;
}
else
{
updateValue(std::numeric_limits<double>::quiet_NaN());
}
}
void markAvailable(bool isAvailable)
{
if (availableInterface)
{
availableInterface->set_property("Available", isAvailable);
errCount = 0;
}
}
void incrementError()
{
if (!readingStateGood())
{
markAvailable(false);
return;
}
if (errCount >= errorThreshold)
{
return;
}
errCount++;
if (errCount == errorThreshold)
{
std::cerr << "Sensor " << name << " reading error!\n";
markFunctional(false);
}
}
void updateValue(const double& newValue)
{
// Ignore if overriding is enabled
if (overriddenState)
{
return;
}
if (!readingStateGood())
{
markAvailable(false);
updateValueProperty(std::numeric_limits<double>::quiet_NaN());
return;
}
updateValueProperty(newValue);
updateInstrumentation(newValue);
// Always check thresholds after changing the value,
// as the test against hysteresisTrigger now takes place in
// the thresholds::checkThresholds() method,
// which is called by checkThresholds() below,
// in all current implementations of sensors that have thresholds.
checkThresholds();
if (!std::isnan(newValue))
{
markFunctional(true);
markAvailable(true);
}
}
void updateProperty(
std::shared_ptr<sdbusplus::asio::dbus_interface>& interface,
double& oldValue, const double& newValue, const char* dbusPropertyName)
{
if (requiresUpdate(oldValue, newValue))
{
oldValue = newValue;
if (interface &&
!(interface->set_property(dbusPropertyName, newValue)))
{
std::cerr << "error setting property " << dbusPropertyName
<< " to " << newValue << "\n";
}
}
}
bool requiresUpdate(const double& lVal, const double& rVal)
{
if (std::isnan(lVal) || std::isnan(rVal))
{
return true;
}
double diff = std::abs(lVal - rVal);
if (diff > hysteresisPublish)
{
return true;
}
return false;
}
private:
void updateValueProperty(const double& newValue)
{
// Indicate that it is internal set call, not an external overwrite
internalSet = true;
updateProperty(sensorInterface, value, newValue, "Value");
internalSet = false;
}
};