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