xref: /openbmc/pldm/platform-mc/numeric_sensor.cpp (revision 77e6fe7ae064915e73760a94ae2c674bc6d3ef8a)
1 #include "numeric_sensor.hpp"
2 
3 #include "libpldm/platform.h"
4 
5 #include "common/utils.hpp"
6 #include "requester/handler.hpp"
7 
8 #include <limits>
9 #include <regex>
10 
11 PHOSPHOR_LOG2_USING;
12 
13 namespace pldm
14 {
15 namespace platform_mc
16 {
17 
18 NumericSensor::NumericSensor(
19     const pldm_tid_t tid, const bool sensorDisabled,
20     std::shared_ptr<pldm_numeric_sensor_value_pdr> pdr, std::string& sensorName,
21     std::string& associationPath) : tid(tid), sensorName(sensorName)
22 {
23     if (!pdr)
24     {
25         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
26     }
27 
28     sensorId = pdr->sensor_id;
29     std::string path;
30     SensorUnit sensorUnit = SensorUnit::DegreesC;
31 
32     switch (pdr->base_unit)
33     {
34         case PLDM_SENSOR_UNIT_DEGRESS_C:
35             sensorNameSpace = "/xyz/openbmc_project/sensors/temperature/";
36             sensorUnit = SensorUnit::DegreesC;
37             break;
38         case PLDM_SENSOR_UNIT_VOLTS:
39             sensorNameSpace = "/xyz/openbmc_project/sensors/voltage/";
40             sensorUnit = SensorUnit::Volts;
41             break;
42         case PLDM_SENSOR_UNIT_AMPS:
43             sensorNameSpace = "/xyz/openbmc_project/sensors/current/";
44             sensorUnit = SensorUnit::Amperes;
45             break;
46         case PLDM_SENSOR_UNIT_RPM:
47             sensorNameSpace = "/xyz/openbmc_project/sensors/fan_pwm/";
48             sensorUnit = SensorUnit::RPMS;
49             break;
50         case PLDM_SENSOR_UNIT_WATTS:
51             sensorNameSpace = "/xyz/openbmc_project/sensors/power/";
52             sensorUnit = SensorUnit::Watts;
53             break;
54         case PLDM_SENSOR_UNIT_JOULES:
55             sensorNameSpace = "/xyz/openbmc_project/sensors/energy/";
56             sensorUnit = SensorUnit::Joules;
57             break;
58         case PLDM_SENSOR_UNIT_PERCENTAGE:
59             sensorNameSpace = "/xyz/openbmc_project/sensors/utilization/";
60             sensorUnit = SensorUnit::Percent;
61             break;
62         default:
63             lg2::error("Sensor {NAME} has Invalid baseUnit {UNIT}.", "NAME",
64                        sensorName, "UNIT", pdr->base_unit);
65             throw sdbusplus::xyz::openbmc_project::Common::Error::
66                 InvalidArgument();
67             break;
68     }
69 
70     path = sensorNameSpace + sensorName;
71     try
72     {
73         auto service = pldm::utils::DBusHandler().getService(
74             path.c_str(), "xyz.openbmc_project.Sensor.Value");
75         if (!service.empty())
76         {
77             throw sdbusplus::xyz::openbmc_project::Common::Error::
78                 TooManyResources();
79         }
80     }
81     catch (const std::exception&)
82     {
83         /* The sensor object path is not created */
84     }
85 
86     auto& bus = pldm::utils::DBusHandler::getBus();
87     try
88     {
89         associationDefinitionsIntf =
90             std::make_unique<AssociationDefinitionsInft>(bus, path.c_str());
91     }
92     catch (const sdbusplus::exception_t& e)
93     {
94         lg2::error(
95             "Failed to create association interface for numeric sensor {PATH} error - {ERROR}",
96             "PATH", path, "ERROR", e);
97         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
98     }
99 
100     associationDefinitionsIntf->associations(
101         {{"chassis", "all_sensors", associationPath}});
102 
103     double maxValue = std::numeric_limits<double>::quiet_NaN();
104     double minValue = std::numeric_limits<double>::quiet_NaN();
105 
106     switch (pdr->sensor_data_size)
107     {
108         case PLDM_SENSOR_DATA_SIZE_UINT8:
109             maxValue = pdr->max_readable.value_u8;
110             minValue = pdr->min_readable.value_u8;
111             hysteresis = pdr->hysteresis.value_u8;
112             break;
113         case PLDM_SENSOR_DATA_SIZE_SINT8:
114             maxValue = pdr->max_readable.value_s8;
115             minValue = pdr->min_readable.value_s8;
116             hysteresis = pdr->hysteresis.value_s8;
117             break;
118         case PLDM_SENSOR_DATA_SIZE_UINT16:
119             maxValue = pdr->max_readable.value_u16;
120             minValue = pdr->min_readable.value_u16;
121             hysteresis = pdr->hysteresis.value_u16;
122             break;
123         case PLDM_SENSOR_DATA_SIZE_SINT16:
124             maxValue = pdr->max_readable.value_s16;
125             minValue = pdr->min_readable.value_s16;
126             hysteresis = pdr->hysteresis.value_s16;
127             break;
128         case PLDM_SENSOR_DATA_SIZE_UINT32:
129             maxValue = pdr->max_readable.value_u32;
130             minValue = pdr->min_readable.value_u32;
131             hysteresis = pdr->hysteresis.value_u32;
132             break;
133         case PLDM_SENSOR_DATA_SIZE_SINT32:
134             maxValue = pdr->max_readable.value_s32;
135             minValue = pdr->min_readable.value_s32;
136             hysteresis = pdr->hysteresis.value_s32;
137             break;
138     }
139 
140     bool hasCriticalThresholds = false;
141     bool hasWarningThresholds = false;
142     double criticalHigh = std::numeric_limits<double>::quiet_NaN();
143     double criticalLow = std::numeric_limits<double>::quiet_NaN();
144     double warningHigh = std::numeric_limits<double>::quiet_NaN();
145     double warningLow = std::numeric_limits<double>::quiet_NaN();
146 
147     if (pdr->supported_thresholds.bits.bit0)
148     {
149         hasWarningThresholds = true;
150         switch (pdr->range_field_format)
151         {
152             case PLDM_RANGE_FIELD_FORMAT_UINT8:
153                 warningHigh = pdr->warning_high.value_u8;
154                 break;
155             case PLDM_RANGE_FIELD_FORMAT_SINT8:
156                 warningHigh = pdr->warning_high.value_s8;
157                 break;
158             case PLDM_RANGE_FIELD_FORMAT_UINT16:
159                 warningHigh = pdr->warning_high.value_u16;
160                 break;
161             case PLDM_RANGE_FIELD_FORMAT_SINT16:
162                 warningHigh = pdr->warning_high.value_s16;
163                 break;
164             case PLDM_RANGE_FIELD_FORMAT_UINT32:
165                 warningHigh = pdr->warning_high.value_u32;
166                 break;
167             case PLDM_RANGE_FIELD_FORMAT_SINT32:
168                 warningHigh = pdr->warning_high.value_s32;
169                 break;
170             case PLDM_RANGE_FIELD_FORMAT_REAL32:
171                 warningHigh = pdr->warning_high.value_f32;
172                 break;
173         }
174     }
175 
176     if (pdr->supported_thresholds.bits.bit3)
177     {
178         hasWarningThresholds = true;
179         switch (pdr->range_field_format)
180         {
181             case PLDM_RANGE_FIELD_FORMAT_UINT8:
182                 warningLow = pdr->warning_low.value_u8;
183                 break;
184             case PLDM_RANGE_FIELD_FORMAT_SINT8:
185                 warningLow = pdr->warning_low.value_s8;
186                 break;
187             case PLDM_RANGE_FIELD_FORMAT_UINT16:
188                 warningLow = pdr->warning_low.value_u16;
189                 break;
190             case PLDM_RANGE_FIELD_FORMAT_SINT16:
191                 warningLow = pdr->warning_low.value_s16;
192                 break;
193             case PLDM_RANGE_FIELD_FORMAT_UINT32:
194                 warningLow = pdr->warning_low.value_u32;
195                 break;
196             case PLDM_RANGE_FIELD_FORMAT_SINT32:
197                 warningLow = pdr->warning_low.value_s32;
198                 break;
199             case PLDM_RANGE_FIELD_FORMAT_REAL32:
200                 warningLow = pdr->warning_low.value_f32;
201                 break;
202         }
203     }
204 
205     if (pdr->supported_thresholds.bits.bit1)
206     {
207         hasCriticalThresholds = true;
208         switch (pdr->range_field_format)
209         {
210             case PLDM_RANGE_FIELD_FORMAT_UINT8:
211                 criticalHigh = pdr->critical_high.value_u8;
212                 break;
213             case PLDM_RANGE_FIELD_FORMAT_SINT8:
214                 criticalHigh = pdr->critical_high.value_s8;
215                 break;
216             case PLDM_RANGE_FIELD_FORMAT_UINT16:
217                 criticalHigh = pdr->critical_high.value_u16;
218                 break;
219             case PLDM_RANGE_FIELD_FORMAT_SINT16:
220                 criticalHigh = pdr->critical_high.value_s16;
221                 break;
222             case PLDM_RANGE_FIELD_FORMAT_UINT32:
223                 criticalHigh = pdr->critical_high.value_u32;
224                 break;
225             case PLDM_RANGE_FIELD_FORMAT_SINT32:
226                 criticalHigh = pdr->critical_high.value_s32;
227                 break;
228             case PLDM_RANGE_FIELD_FORMAT_REAL32:
229                 criticalHigh = pdr->critical_high.value_f32;
230                 break;
231         }
232     }
233 
234     if (pdr->supported_thresholds.bits.bit4)
235     {
236         hasCriticalThresholds = true;
237         switch (pdr->range_field_format)
238         {
239             case PLDM_RANGE_FIELD_FORMAT_UINT8:
240                 criticalLow = pdr->critical_low.value_u8;
241                 break;
242             case PLDM_RANGE_FIELD_FORMAT_SINT8:
243                 criticalLow = pdr->critical_low.value_s8;
244                 break;
245             case PLDM_RANGE_FIELD_FORMAT_UINT16:
246                 criticalLow = pdr->critical_low.value_u16;
247                 break;
248             case PLDM_RANGE_FIELD_FORMAT_SINT16:
249                 criticalLow = pdr->critical_low.value_s16;
250                 break;
251             case PLDM_RANGE_FIELD_FORMAT_UINT32:
252                 criticalLow = pdr->critical_low.value_u32;
253                 break;
254             case PLDM_RANGE_FIELD_FORMAT_SINT32:
255                 criticalLow = pdr->critical_low.value_s32;
256                 break;
257             case PLDM_RANGE_FIELD_FORMAT_REAL32:
258                 criticalLow = pdr->critical_low.value_f32;
259                 break;
260         }
261     }
262 
263     resolution = pdr->resolution;
264     offset = pdr->offset;
265     baseUnitModifier = pdr->unit_modifier;
266     timeStamp = 0;
267 
268     /**
269      * DEFAULT_SENSOR_UPDATER_INTERVAL is in milliseconds
270      * updateTime is in microseconds
271      */
272     updateTime = static_cast<uint64_t>(DEFAULT_SENSOR_UPDATER_INTERVAL * 1000);
273     if (!std::isnan(pdr->update_interval))
274     {
275         updateTime = pdr->update_interval * 1000000;
276     }
277 
278     try
279     {
280         valueIntf = std::make_unique<ValueIntf>(bus, path.c_str());
281     }
282     catch (const sdbusplus::exception_t& e)
283     {
284         lg2::error(
285             "Failed to create Value interface for numeric sensor {PATH} error - {ERROR}",
286             "PATH", path, "ERROR", e);
287         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
288     }
289     valueIntf->maxValue(unitModifier(conversionFormula(maxValue)));
290     valueIntf->minValue(unitModifier(conversionFormula(minValue)));
291     hysteresis = unitModifier(conversionFormula(hysteresis));
292     valueIntf->unit(sensorUnit);
293 
294     try
295     {
296         availabilityIntf =
297             std::make_unique<AvailabilityIntf>(bus, path.c_str());
298     }
299     catch (const sdbusplus::exception_t& e)
300     {
301         lg2::error(
302             "Failed to create Availability interface for numeric sensor {PATH} error - {ERROR}",
303             "PATH", path, "ERROR", e);
304         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
305     }
306     availabilityIntf->available(true);
307 
308     try
309     {
310         operationalStatusIntf =
311             std::make_unique<OperationalStatusIntf>(bus, path.c_str());
312     }
313     catch (const sdbusplus::exception_t& e)
314     {
315         lg2::error(
316             "Failed to create Operation status interface for numeric sensor {PATH} error - {ERROR}",
317             "PATH", path, "ERROR", e);
318         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
319     }
320     operationalStatusIntf->functional(!sensorDisabled);
321 
322     if (hasWarningThresholds)
323     {
324         try
325         {
326             thresholdWarningIntf =
327                 std::make_unique<ThresholdWarningIntf>(bus, path.c_str());
328         }
329         catch (const sdbusplus::exception_t& e)
330         {
331             lg2::error(
332                 "Failed to create Threshold warning interface for numeric sensor {PATH} error - {ERROR}",
333                 "PATH", path, "ERROR", e);
334             throw sdbusplus::xyz::openbmc_project::Common::Error::
335                 InvalidArgument();
336         }
337         thresholdWarningIntf->warningHigh(unitModifier(warningHigh));
338         thresholdWarningIntf->warningLow(unitModifier(warningLow));
339     }
340 
341     if (hasCriticalThresholds)
342     {
343         try
344         {
345             thresholdCriticalIntf =
346                 std::make_unique<ThresholdCriticalIntf>(bus, path.c_str());
347         }
348         catch (const sdbusplus::exception_t& e)
349         {
350             lg2::error(
351                 "Failed to create Threshold critical interface for numeric sensor {PATH} error - {ERROR}",
352                 "PATH", path, "ERROR", e);
353             throw sdbusplus::xyz::openbmc_project::Common::Error::
354                 InvalidArgument();
355         }
356         thresholdCriticalIntf->criticalHigh(unitModifier(criticalHigh));
357         thresholdCriticalIntf->criticalLow(unitModifier(criticalLow));
358     }
359 }
360 
361 NumericSensor::NumericSensor(
362     const pldm_tid_t tid, const bool sensorDisabled,
363     std::shared_ptr<pldm_compact_numeric_sensor_pdr> pdr,
364     std::string& sensorName, std::string& associationPath) :
365     tid(tid), sensorName(sensorName)
366 {
367     if (!pdr)
368     {
369         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
370     }
371 
372     sensorId = pdr->sensor_id;
373     std::string path;
374     SensorUnit sensorUnit = SensorUnit::DegreesC;
375 
376     switch (pdr->base_unit)
377     {
378         case PLDM_SENSOR_UNIT_DEGRESS_C:
379             sensorNameSpace = "/xyz/openbmc_project/sensors/temperature/";
380             sensorUnit = SensorUnit::DegreesC;
381             break;
382         case PLDM_SENSOR_UNIT_VOLTS:
383             sensorNameSpace = "/xyz/openbmc_project/sensors/voltage/";
384             sensorUnit = SensorUnit::Volts;
385             break;
386         case PLDM_SENSOR_UNIT_AMPS:
387             sensorNameSpace = "/xyz/openbmc_project/sensors/current/";
388             sensorUnit = SensorUnit::Amperes;
389             break;
390         case PLDM_SENSOR_UNIT_RPM:
391             sensorNameSpace = "/xyz/openbmc_project/sensors/fan_pwm/";
392             sensorUnit = SensorUnit::RPMS;
393             break;
394         case PLDM_SENSOR_UNIT_WATTS:
395             sensorNameSpace = "/xyz/openbmc_project/sensors/power/";
396             sensorUnit = SensorUnit::Watts;
397             break;
398         case PLDM_SENSOR_UNIT_JOULES:
399             sensorNameSpace = "/xyz/openbmc_project/sensors/energy/";
400             sensorUnit = SensorUnit::Joules;
401             break;
402         case PLDM_SENSOR_UNIT_PERCENTAGE:
403             sensorNameSpace = "/xyz/openbmc_project/sensors/utilization/";
404             sensorUnit = SensorUnit::Percent;
405             break;
406         default:
407             lg2::error("Sensor {NAME} has Invalid baseUnit {UNIT}.", "NAME",
408                        sensorName, "UNIT", pdr->base_unit);
409             throw sdbusplus::xyz::openbmc_project::Common::Error::
410                 InvalidArgument();
411             break;
412     }
413 
414     path = sensorNameSpace + sensorName;
415     try
416     {
417         auto service = pldm::utils::DBusHandler().getService(
418             path.c_str(), "xyz.openbmc_project.Sensor.Value");
419         if (!service.empty())
420         {
421             throw sdbusplus::xyz::openbmc_project::Common::Error::
422                 TooManyResources();
423         }
424     }
425     catch (const std::exception&)
426     {
427         /* The sensor object path is not created */
428     }
429 
430     auto& bus = pldm::utils::DBusHandler::getBus();
431     try
432     {
433         associationDefinitionsIntf =
434             std::make_unique<AssociationDefinitionsInft>(bus, path.c_str());
435     }
436     catch (const sdbusplus::exception_t& e)
437     {
438         lg2::error(
439             "Failed to create Association interface for compact numeric sensor {PATH} error - {ERROR}",
440             "PATH", path, "ERROR", e);
441         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
442     }
443     associationDefinitionsIntf->associations(
444         {{"chassis", "all_sensors", associationPath.c_str()}});
445 
446     double maxValue = std::numeric_limits<double>::quiet_NaN();
447     double minValue = std::numeric_limits<double>::quiet_NaN();
448     bool hasWarningThresholds = false;
449     bool hasCriticalThresholds = false;
450     double criticalHigh = std::numeric_limits<double>::quiet_NaN();
451     double criticalLow = std::numeric_limits<double>::quiet_NaN();
452     double warningHigh = std::numeric_limits<double>::quiet_NaN();
453     double warningLow = std::numeric_limits<double>::quiet_NaN();
454 
455     if (pdr->range_field_support.bits.bit0)
456     {
457         hasWarningThresholds = true;
458         warningHigh = pdr->warning_high;
459     }
460     if (pdr->range_field_support.bits.bit1)
461     {
462         hasWarningThresholds = true;
463         warningLow = pdr->warning_low;
464     }
465 
466     if (pdr->range_field_support.bits.bit2)
467     {
468         hasCriticalThresholds = true;
469         criticalHigh = pdr->critical_high;
470     }
471 
472     if (pdr->range_field_support.bits.bit3)
473     {
474         hasCriticalThresholds = true;
475         criticalLow = pdr->critical_low;
476     }
477 
478     resolution = std::numeric_limits<double>::quiet_NaN();
479     offset = std::numeric_limits<double>::quiet_NaN();
480     baseUnitModifier = pdr->unit_modifier;
481     timeStamp = 0;
482 
483     /**
484      * DEFAULT_SENSOR_UPDATER_INTERVAL is in milliseconds
485      * updateTime is in microseconds
486      */
487     updateTime = static_cast<uint64_t>(DEFAULT_SENSOR_UPDATER_INTERVAL * 1000);
488     try
489     {
490         valueIntf = std::make_unique<ValueIntf>(bus, path.c_str());
491     }
492     catch (const sdbusplus::exception_t& e)
493     {
494         lg2::error(
495             "Failed to create Value interface for compact numeric sensor {PATH} error - {ERROR}",
496             "PATH", path, "ERROR", e);
497         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
498     }
499     valueIntf->maxValue(unitModifier(conversionFormula(maxValue)));
500     valueIntf->minValue(unitModifier(conversionFormula(minValue)));
501     hysteresis = unitModifier(conversionFormula(hysteresis));
502     valueIntf->unit(sensorUnit);
503 
504     try
505     {
506         availabilityIntf =
507             std::make_unique<AvailabilityIntf>(bus, path.c_str());
508     }
509     catch (const sdbusplus::exception_t& e)
510     {
511         lg2::error(
512             "Failed to create Availability interface for compact numeric sensor {PATH} error - {ERROR}",
513             "PATH", path, "ERROR", e);
514         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
515     }
516     availabilityIntf->available(true);
517 
518     try
519     {
520         operationalStatusIntf =
521             std::make_unique<OperationalStatusIntf>(bus, path.c_str());
522     }
523     catch (const sdbusplus::exception_t& e)
524     {
525         lg2::error(
526             "Failed to create Operational status interface for compact numeric sensor {PATH} error - {ERROR}",
527             "PATH", path, "ERROR", e);
528         throw sdbusplus::xyz::openbmc_project::Common::Error::InvalidArgument();
529     }
530     operationalStatusIntf->functional(!sensorDisabled);
531 
532     if (hasWarningThresholds)
533     {
534         try
535         {
536             thresholdWarningIntf =
537                 std::make_unique<ThresholdWarningIntf>(bus, path.c_str());
538         }
539         catch (const sdbusplus::exception_t& e)
540         {
541             lg2::error(
542                 "Failed to create Warning threshold interface for compact numeric sensor {PATH} error - {ERROR}",
543                 "PATH", path, "ERROR", e);
544             throw sdbusplus::xyz::openbmc_project::Common::Error::
545                 InvalidArgument();
546         }
547         thresholdWarningIntf->warningHigh(unitModifier(warningHigh));
548         thresholdWarningIntf->warningLow(unitModifier(warningLow));
549     }
550 
551     if (hasCriticalThresholds)
552     {
553         try
554         {
555             thresholdCriticalIntf =
556                 std::make_unique<ThresholdCriticalIntf>(bus, path.c_str());
557         }
558         catch (const sdbusplus::exception_t& e)
559         {
560             lg2::error(
561                 "Failed to create Critical threshold interface for compact numeric sensor {PATH} error - {ERROR}",
562                 "PATH", path, "ERROR", e);
563             throw sdbusplus::xyz::openbmc_project::Common::Error::
564                 InvalidArgument();
565         }
566         thresholdCriticalIntf->criticalHigh(unitModifier(criticalHigh));
567         thresholdCriticalIntf->criticalLow(unitModifier(criticalLow));
568     }
569 }
570 
571 double NumericSensor::conversionFormula(double value)
572 {
573     double convertedValue = value;
574     convertedValue *= std::isnan(resolution) ? 1 : resolution;
575     convertedValue += std::isnan(offset) ? 0 : offset;
576     return convertedValue;
577 }
578 
579 double NumericSensor::unitModifier(double value)
580 {
581     return std::isnan(value) ? value : value * std::pow(10, baseUnitModifier);
582 }
583 
584 void NumericSensor::updateReading(bool available, bool functional, double value)
585 {
586     if (!availabilityIntf || !operationalStatusIntf || !valueIntf)
587     {
588         lg2::error(
589             "Failed to update sensor {NAME} D-Bus interface don't exist.",
590             "NAME", sensorName);
591         return;
592     }
593     availabilityIntf->available(available);
594     operationalStatusIntf->functional(functional);
595     double curValue = valueIntf->value();
596     double newValue = std::numeric_limits<double>::quiet_NaN();
597     if (functional && available)
598     {
599         newValue = unitModifier(conversionFormula(value));
600         if (newValue != curValue &&
601             (!std::isnan(newValue) || !std::isnan(curValue)))
602         {
603             valueIntf->value(newValue);
604             updateThresholds();
605         }
606     }
607     else
608     {
609         if (newValue != curValue &&
610             (!std::isnan(newValue) || !std::isnan(curValue)))
611         {
612             valueIntf->value(std::numeric_limits<double>::quiet_NaN());
613         }
614     }
615 }
616 
617 void NumericSensor::handleErrGetSensorReading()
618 {
619     if (!operationalStatusIntf || !valueIntf)
620     {
621         lg2::error(
622             "Failed to update sensor {NAME} D-Bus interfaces don't exist.",
623             "NAME", sensorName);
624         return;
625     }
626     operationalStatusIntf->functional(false);
627     valueIntf->value(std::numeric_limits<double>::quiet_NaN());
628 }
629 
630 bool NumericSensor::checkThreshold(bool alarm, bool direction, double value,
631                                    double threshold, double hyst)
632 {
633     if (direction)
634     {
635         if (value >= threshold)
636         {
637             return true;
638         }
639         if (value < (threshold - hyst))
640         {
641             return false;
642         }
643     }
644     else
645     {
646         if (value <= threshold)
647         {
648             return true;
649         }
650         if (value > (threshold + hyst))
651         {
652             return false;
653         }
654     }
655     return alarm;
656 }
657 
658 void NumericSensor::updateThresholds()
659 {
660     if (!valueIntf)
661     {
662         lg2::error(
663             "Failed to update thresholds sensor {NAME} D-Bus interfaces don't exist.",
664             "NAME", sensorName);
665         return;
666     }
667 
668     auto value = valueIntf->value();
669 
670     if (thresholdWarningIntf &&
671         !std::isnan(thresholdWarningIntf->warningHigh()))
672     {
673         auto threshold = thresholdWarningIntf->warningHigh();
674         auto alarm = thresholdWarningIntf->warningAlarmHigh();
675         auto newAlarm =
676             checkThreshold(alarm, true, value, threshold, hysteresis);
677         if (alarm != newAlarm)
678         {
679             thresholdWarningIntf->warningAlarmHigh(newAlarm);
680             if (newAlarm)
681             {
682                 thresholdWarningIntf->warningHighAlarmAsserted(value);
683             }
684             else
685             {
686                 thresholdWarningIntf->warningHighAlarmDeasserted(value);
687             }
688         }
689     }
690 
691     if (thresholdWarningIntf && !std::isnan(thresholdWarningIntf->warningLow()))
692     {
693         auto threshold = thresholdWarningIntf->warningLow();
694         auto alarm = thresholdWarningIntf->warningAlarmLow();
695         auto newAlarm =
696             checkThreshold(alarm, false, value, threshold, hysteresis);
697         if (alarm != newAlarm)
698         {
699             thresholdWarningIntf->warningAlarmLow(newAlarm);
700             if (newAlarm)
701             {
702                 thresholdWarningIntf->warningLowAlarmAsserted(value);
703             }
704             else
705             {
706                 thresholdWarningIntf->warningLowAlarmDeasserted(value);
707             }
708         }
709     }
710 
711     if (thresholdCriticalIntf &&
712         !std::isnan(thresholdCriticalIntf->criticalHigh()))
713     {
714         auto threshold = thresholdCriticalIntf->criticalHigh();
715         auto alarm = thresholdCriticalIntf->criticalAlarmHigh();
716         auto newAlarm =
717             checkThreshold(alarm, true, value, threshold, hysteresis);
718         if (alarm != newAlarm)
719         {
720             thresholdCriticalIntf->criticalAlarmHigh(newAlarm);
721             if (newAlarm)
722             {
723                 thresholdCriticalIntf->criticalHighAlarmAsserted(value);
724             }
725             else
726             {
727                 thresholdCriticalIntf->criticalHighAlarmDeasserted(value);
728             }
729         }
730     }
731 
732     if (thresholdCriticalIntf &&
733         !std::isnan(thresholdCriticalIntf->criticalLow()))
734     {
735         auto threshold = thresholdCriticalIntf->criticalLow();
736         auto alarm = thresholdCriticalIntf->criticalAlarmLow();
737         auto newAlarm =
738             checkThreshold(alarm, false, value, threshold, hysteresis);
739         if (alarm != newAlarm)
740         {
741             thresholdCriticalIntf->criticalAlarmLow(newAlarm);
742             if (newAlarm)
743             {
744                 thresholdCriticalIntf->criticalLowAlarmAsserted(value);
745             }
746             else
747             {
748                 thresholdCriticalIntf->criticalLowAlarmDeasserted(value);
749             }
750         }
751     }
752 }
753 
754 int NumericSensor::triggerThresholdEvent(
755     pldm::utils::Level eventType, pldm::utils::Direction direction,
756     double rawValue, bool newAlarm, bool assert)
757 {
758     if (!valueIntf)
759     {
760         lg2::error(
761             "Failed to update thresholds sensor {NAME} D-Bus interfaces don't exist.",
762             "NAME", sensorName);
763         return PLDM_ERROR;
764     }
765 
766     auto value = unitModifier(conversionFormula(rawValue));
767     lg2::error(
768         "triggerThresholdEvent eventType {TID}, direction {SID} value {VAL} newAlarm {PSTATE} assert {ESTATE}",
769         "TID", eventType, "SID", direction, "VAL", value, "PSTATE", newAlarm,
770         "ESTATE", assert);
771 
772     switch (eventType)
773     {
774         case pldm::utils::Level::WARNING:
775         {
776             if (!thresholdWarningIntf)
777             {
778                 lg2::error(
779                     "Error:Trigger sensor warning event for non warning threshold sensors {NAME}",
780                     "NAME", sensorName);
781                 return PLDM_ERROR;
782             }
783             if (direction == pldm::utils::Direction::HIGH &&
784                 !std::isnan(thresholdWarningIntf->warningHigh()))
785             {
786                 auto alarm = thresholdWarningIntf->warningAlarmHigh();
787                 if (alarm == newAlarm)
788                 {
789                     return PLDM_SUCCESS;
790                 }
791                 thresholdWarningIntf->warningAlarmHigh(newAlarm);
792                 if (assert)
793                 {
794                     thresholdWarningIntf->warningHighAlarmAsserted(value);
795                 }
796                 else
797                 {
798                     thresholdWarningIntf->warningHighAlarmDeasserted(value);
799                 }
800             }
801             else if (direction == pldm::utils::Direction::LOW &&
802                      !std::isnan(thresholdWarningIntf->warningLow()))
803             {
804                 auto alarm = thresholdWarningIntf->warningAlarmLow();
805                 if (alarm == newAlarm)
806                 {
807                     return PLDM_SUCCESS;
808                 }
809                 thresholdWarningIntf->warningAlarmLow(newAlarm);
810                 if (assert)
811                 {
812                     thresholdWarningIntf->warningLowAlarmAsserted(value);
813                 }
814                 else
815                 {
816                     thresholdWarningIntf->warningLowAlarmDeasserted(value);
817                 }
818             }
819             break;
820         }
821         case pldm::utils::Level::CRITICAL:
822         {
823             if (!thresholdCriticalIntf)
824             {
825                 lg2::error(
826                     "Error:Trigger sensor Critical event for non warning threshold sensors {NAME}",
827                     "NAME", sensorName);
828                 return PLDM_ERROR;
829             }
830             if (direction == pldm::utils::Direction::HIGH &&
831                 !std::isnan(thresholdCriticalIntf->criticalHigh()))
832             {
833                 auto alarm = thresholdCriticalIntf->criticalAlarmHigh();
834                 if (alarm == newAlarm)
835                 {
836                     return PLDM_SUCCESS;
837                 }
838                 thresholdCriticalIntf->criticalAlarmHigh(newAlarm);
839                 if (assert)
840                 {
841                     thresholdCriticalIntf->criticalHighAlarmAsserted(value);
842                 }
843                 else
844                 {
845                     thresholdCriticalIntf->criticalHighAlarmDeasserted(value);
846                 }
847             }
848             else if (direction == pldm::utils::Direction::LOW &&
849                      !std::isnan(thresholdCriticalIntf->criticalLow()))
850             {
851                 auto alarm = thresholdCriticalIntf->criticalAlarmLow();
852                 if (alarm == newAlarm)
853                 {
854                     return PLDM_SUCCESS;
855                 }
856                 thresholdCriticalIntf->criticalAlarmLow(newAlarm);
857                 if (assert)
858                 {
859                     thresholdCriticalIntf->criticalLowAlarmAsserted(value);
860                 }
861                 else
862                 {
863                     thresholdCriticalIntf->criticalLowAlarmDeasserted(value);
864                 }
865             }
866             break;
867         }
868 
869         default:
870             break;
871     }
872 
873     return PLDM_SUCCESS;
874 }
875 } // namespace platform_mc
876 } // namespace pldm
877