xref: /openbmc/telemetry/src/sensor.cpp (revision fdb06a14)
1 #include "sensor.hpp"
2 
3 #include "utils/clock.hpp"
4 
5 #include <boost/container/flat_map.hpp>
6 #include <phosphor-logging/log.hpp>
7 #include <sdbusplus/asio/property.hpp>
8 
9 #include <functional>
10 
11 Sensor::Sensor(interfaces::Sensor::Id sensorId,
12                const std::string& sensorMetadata, boost::asio::io_context& ioc,
13                const std::shared_ptr<sdbusplus::asio::connection>& bus) :
14     sensorId(std::move(sensorId)),
15     sensorMetadata(sensorMetadata), ioc(ioc), bus(bus)
16 {}
17 
18 Sensor::Id Sensor::makeId(std::string_view service, std::string_view path)
19 {
20     return Id("Sensor", service, path);
21 }
22 
23 Sensor::Id Sensor::id() const
24 {
25     return sensorId;
26 }
27 
28 std::string Sensor::metadata() const
29 {
30     return sensorMetadata;
31 }
32 
33 std::string Sensor::getName() const
34 {
35     return sensorMetadata.empty() ? sensorId.path : sensorMetadata;
36 }
37 
38 void Sensor::async_read()
39 {
40     uniqueCall([this](auto lock) { async_read(std::move(lock)); });
41 }
42 
43 void Sensor::async_read(std::shared_ptr<utils::UniqueCall::Lock> lock)
44 {
45     makeSignalMonitor();
46 
47     sdbusplus::asio::getProperty<double>(
48         *bus, sensorId.service, sensorId.path,
49         "xyz.openbmc_project.Sensor.Value", "Value",
50         [lock, id = sensorId, weakSelf = weak_from_this()](
51             boost::system::error_code ec, double newValue) {
52             if (ec)
53             {
54                 phosphor::logging::log<phosphor::logging::level::WARNING>(
55                     "DBus 'GetProperty' call failed on Sensor Value",
56                     phosphor::logging::entry("SENSOR_PATH=%s", id.path.c_str()),
57                     phosphor::logging::entry("ERROR_CODE=%d", ec.value()));
58                 return;
59             }
60             if (auto self = weakSelf.lock())
61             {
62                 self->updateValue(newValue);
63             }
64         });
65 }
66 
67 void Sensor::registerForUpdates(
68     const std::weak_ptr<interfaces::SensorListener>& weakListener)
69 {
70     listeners.erase(
71         std::remove_if(listeners.begin(), listeners.end(),
72                        [](const auto& listener) { return listener.expired(); }),
73         listeners.end());
74 
75     if (auto listener = weakListener.lock())
76     {
77         listeners.emplace_back(weakListener);
78 
79         if (value)
80         {
81             listener->sensorUpdated(*this, timestamp, *value);
82         }
83         else
84         {
85             async_read();
86         }
87     }
88 }
89 
90 void Sensor::unregisterFromUpdates(
91     const std::weak_ptr<interfaces::SensorListener>& weakListener)
92 {
93     if (auto listener = weakListener.lock())
94     {
95         listeners.erase(
96             std::remove_if(
97                 listeners.begin(), listeners.end(),
98                 [listenerToUnregister = listener.get()](const auto& listener) {
99                     return (listener.expired() ||
100                             listener.lock().get() == listenerToUnregister);
101                 }),
102             listeners.end());
103     }
104 }
105 
106 void Sensor::updateValue(double newValue)
107 {
108     timestamp = Clock().steadyTimestamp();
109 
110     if (value == newValue)
111     {
112         for (const auto& weakListener : listeners)
113         {
114             if (auto listener = weakListener.lock())
115             {
116                 listener->sensorUpdated(*this, timestamp);
117             }
118         }
119     }
120     else
121     {
122         value = newValue;
123 
124         for (const auto& weakListener : listeners)
125         {
126             if (auto listener = weakListener.lock())
127             {
128                 listener->sensorUpdated(*this, timestamp, *value);
129             }
130         }
131     }
132 }
133 
134 void Sensor::makeSignalMonitor()
135 {
136     if (signalMonitor)
137     {
138         return;
139     }
140 
141     using namespace std::string_literals;
142 
143     const auto param = "type='signal',member='PropertiesChanged',path='"s +
144                        sensorId.path +
145                        "',arg0='xyz.openbmc_project.Sensor.Value'"s;
146 
147     signalMonitor = std::make_unique<sdbusplus::bus::match_t>(
148         *bus, param,
149         [weakSelf = weak_from_this()](sdbusplus::message::message& message) {
150             signalProc(weakSelf, message);
151         });
152 }
153 
154 void Sensor::signalProc(const std::weak_ptr<Sensor>& weakSelf,
155                         sdbusplus::message::message& message)
156 {
157     if (auto self = weakSelf.lock())
158     {
159         std::string iface;
160         boost::container::flat_map<std::string, ValueVariant>
161             changed_properties;
162         std::vector<std::string> invalidated_properties;
163 
164         message.read(iface, changed_properties, invalidated_properties);
165 
166         if (iface == "xyz.openbmc_project.Sensor.Value")
167         {
168             const auto it = changed_properties.find("Value");
169             if (it != changed_properties.end())
170             {
171                 if (auto val = std::get_if<double>(&it->second))
172                 {
173                     self->updateValue(*val);
174                 }
175                 else
176                 {
177                     phosphor::logging::log<phosphor::logging::level::ERR>(
178                         "Failed to receive Value from Sensor "
179                         "PropertiesChanged signal",
180                         phosphor::logging::entry("SENSOR_PATH=%s",
181                                                  self->sensorId.path.c_str()));
182                 }
183             }
184         }
185     }
186 }
187 
188 LabeledSensorInfo Sensor::getLabeledSensorInfo() const
189 {
190     return LabeledSensorInfo(sensorId.service, sensorId.path, sensorMetadata);
191 }
192