1 /** 2 * Copyright © 2021 IBM Corporation 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 #include "threshold_alarm_logger.hpp" 17 18 #include "sdbusplus.hpp" 19 20 #include <fmt/format.h> 21 22 #include <phosphor-logging/log.hpp> 23 #include <xyz/openbmc_project/Logging/Entry/server.hpp> 24 25 namespace sensor::monitor 26 { 27 28 using namespace sdbusplus::xyz::openbmc_project::Logging::server; 29 using namespace phosphor::logging; 30 using namespace phosphor::fan::util; 31 32 const std::string warningInterface = 33 "xyz.openbmc_project.Sensor.Threshold.Warning"; 34 const std::string criticalInterface = 35 "xyz.openbmc_project.Sensor.Threshold.Critical"; 36 const std::string perfLossInterface = 37 "xyz.openbmc_project.Sensor.Threshold.PerformanceLoss"; 38 constexpr auto loggingService = "xyz.openbmc_project.Logging"; 39 constexpr auto loggingPath = "/xyz/openbmc_project/logging"; 40 constexpr auto loggingCreateIface = "xyz.openbmc_project.Logging.Create"; 41 constexpr auto errorNameBase = "xyz.openbmc_project.Sensor.Threshold.Error."; 42 constexpr auto valueInterface = "xyz.openbmc_project.Sensor.Value"; 43 constexpr auto assocInterface = "xyz.openbmc_project.Association"; 44 45 using ErrorData = std::tuple<ErrorName, Entry::Level>; 46 47 /** 48 * Map of threshold interfaces and alarm properties and values to error data. 49 */ 50 const std::map<InterfaceName, std::map<PropertyName, std::map<bool, ErrorData>>> 51 thresholdData{ 52 53 {warningInterface, 54 {{"WarningAlarmHigh", 55 {{true, ErrorData{"WarningHigh", Entry::Level::Warning}}, 56 {false, 57 ErrorData{"WarningHighClear", Entry::Level::Informational}}}}, 58 {"WarningAlarmLow", 59 {{true, ErrorData{"WarningLow", Entry::Level::Warning}}, 60 {false, 61 ErrorData{"WarningLowClear", Entry::Level::Informational}}}}}}, 62 63 {criticalInterface, 64 {{"CriticalAlarmHigh", 65 {{true, ErrorData{"CriticalHigh", Entry::Level::Critical}}, 66 {false, 67 ErrorData{"CriticalHighClear", Entry::Level::Informational}}}}, 68 {"CriticalAlarmLow", 69 {{true, ErrorData{"CriticalLow", Entry::Level::Critical}}, 70 {false, 71 ErrorData{"CriticalLowClear", Entry::Level::Informational}}}}}}, 72 73 {perfLossInterface, 74 {{"PerfLossAlarmHigh", 75 {{true, ErrorData{"PerfLossHigh", Entry::Level::Warning}}, 76 {false, 77 ErrorData{"PerfLossHighClear", Entry::Level::Informational}}}}, 78 {"PerfLossAlarmLow", 79 {{true, ErrorData{"PerfLossLow", Entry::Level::Warning}}, 80 {false, 81 ErrorData{"PerfLossLowClear", Entry::Level::Informational}}}}}}}; 82 83 ThresholdAlarmLogger::ThresholdAlarmLogger(sdbusplus::bus::bus& bus, 84 sdeventplus::Event& event) : 85 bus(bus), 86 event(event), 87 warningMatch(bus, 88 "type='signal',member='PropertiesChanged'," 89 "path_namespace='/xyz/openbmc_project/sensors'," 90 "arg0='" + 91 warningInterface + "'", 92 std::bind(&ThresholdAlarmLogger::propertiesChanged, this, 93 std::placeholders::_1)), 94 criticalMatch(bus, 95 "type='signal',member='PropertiesChanged'," 96 "path_namespace='/xyz/openbmc_project/sensors'," 97 "arg0='" + 98 criticalInterface + "'", 99 std::bind(&ThresholdAlarmLogger::propertiesChanged, this, 100 std::placeholders::_1)), 101 perfLossMatch(bus, 102 "type='signal',member='PropertiesChanged'," 103 "path_namespace='/xyz/openbmc_project/sensors'," 104 "arg0='" + 105 perfLossInterface + "'", 106 std::bind(&ThresholdAlarmLogger::propertiesChanged, this, 107 std::placeholders::_1)) 108 { 109 // check for any currently asserted threshold alarms 110 std::for_each( 111 thresholdData.begin(), thresholdData.end(), 112 [this](const auto& thresholdInterface) { 113 const auto& interface = thresholdInterface.first; 114 auto objects = 115 SDBusPlus::getSubTreeRaw(this->bus, "/", interface, 0); 116 std::for_each(objects.begin(), objects.end(), 117 [interface, this](const auto& object) { 118 const auto& path = object.first; 119 const auto& service = 120 object.second.begin()->first; 121 checkThresholds(interface, path, service); 122 }); 123 }); 124 } 125 126 void ThresholdAlarmLogger::propertiesChanged(sdbusplus::message::message& msg) 127 { 128 std::map<std::string, std::variant<bool>> properties; 129 std::string sensorPath = msg.get_path(); 130 std::string interface; 131 132 msg.read(interface, properties); 133 134 auto alarmProperties = thresholdData.find(interface); 135 if (alarmProperties == thresholdData.end()) 136 { 137 return; 138 } 139 140 for (const auto& [propertyName, propertyValue] : properties) 141 { 142 if (alarmProperties->second.find(propertyName) != 143 alarmProperties->second.end()) 144 { 145 // If this is the first time we've seen this alarm, then 146 // assume it was off before so it doesn't create an event 147 // log for a value of false. 148 149 InterfaceKey key{sensorPath, interface}; 150 if (alarms.find(key) == alarms.end()) 151 { 152 alarms[key][propertyName] = false; 153 } 154 155 // Check if the value changed from what was there before. 156 auto alarmValue = std::get<bool>(propertyValue); 157 if (alarmValue != alarms[key][propertyName]) 158 { 159 alarms[key][propertyName] = alarmValue; 160 createEventLog(sensorPath, interface, propertyName, alarmValue); 161 } 162 } 163 } 164 } 165 166 void ThresholdAlarmLogger::checkThresholds(const std::string& interface, 167 const std::string& sensorPath, 168 const std::string& service) 169 { 170 auto properties = thresholdData.find(interface); 171 if (properties == thresholdData.end()) 172 { 173 return; 174 } 175 176 for (const auto& [property, unused] : properties->second) 177 { 178 try 179 { 180 auto alarmValue = SDBusPlus::getProperty<bool>( 181 bus, service, sensorPath, interface, property); 182 alarms[InterfaceKey(sensorPath, interface)][property] = alarmValue; 183 184 // This is just for checking alarms on startup, 185 // so only look for active alarms. 186 if (alarmValue) 187 { 188 createEventLog(sensorPath, interface, property, alarmValue); 189 } 190 } 191 catch (const DBusError& e) 192 { 193 log<level::ERR>( 194 fmt::format("Failed reading sensor threshold properties: {}", 195 e.what()) 196 .c_str()); 197 continue; 198 } 199 } 200 } 201 202 void ThresholdAlarmLogger::createEventLog(const std::string& sensorPath, 203 const std::string& interface, 204 const std::string& alarmProperty, 205 bool alarmValue) 206 { 207 std::map<std::string, std::string> ad; 208 209 auto type = getSensorType(sensorPath); 210 if (skipSensorType(type)) 211 { 212 return; 213 } 214 215 auto it = thresholdData.find(interface); 216 if (it == thresholdData.end()) 217 { 218 return; 219 } 220 221 auto properties = it->second.find(alarmProperty); 222 if (properties == it->second.end()) 223 { 224 log<level::INFO>( 225 fmt::format("Could not find {} in threshold alarms map", 226 alarmProperty) 227 .c_str()); 228 return; 229 } 230 231 ad.emplace("SENSOR_NAME", sensorPath); 232 233 try 234 { 235 auto sensorValue = SDBusPlus::getProperty<double>( 236 bus, sensorPath, valueInterface, "Value"); 237 238 ad.emplace("SENSOR_VALUE", std::to_string(sensorValue)); 239 240 log<level::INFO>( 241 fmt::format("Threshold Event {} {} = {} (sensor value {})", 242 sensorPath, alarmProperty, alarmValue, sensorValue) 243 .c_str()); 244 } 245 catch (const DBusServiceError& e) 246 { 247 // If the sensor was just added, the Value interface for it may 248 // not be in the mapper yet. This could only happen if the sensor 249 // application was started up after this one and the value exceeded the 250 // threshold immediately. 251 log<level::INFO>(fmt::format("Threshold Event {} {} = {}", sensorPath, 252 alarmProperty, alarmValue) 253 .c_str()); 254 } 255 256 auto callout = getCallout(sensorPath); 257 if (!callout.empty()) 258 { 259 ad.emplace("CALLOUT_INVENTORY_PATH", callout); 260 } 261 262 auto errorData = properties->second.find(alarmValue); 263 264 // Add the base error name and the sensor type (like Temperature) to the 265 // error name that's in the thresholdData name to get something like 266 // xyz.openbmc_project.Sensor.Threshold.Error.TemperatureWarningHigh 267 const auto& [name, severity] = errorData->second; 268 type.front() = toupper(type.front()); 269 std::string errorName = errorNameBase + type + name; 270 271 SDBusPlus::callMethod(loggingService, loggingPath, loggingCreateIface, 272 "Create", errorName, convertForMessage(severity), ad); 273 } 274 275 std::string ThresholdAlarmLogger::getSensorType(std::string sensorPath) 276 { 277 auto pos = sensorPath.find_last_of('/'); 278 if ((sensorPath.back() == '/') || (pos == std::string::npos)) 279 { 280 log<level::ERR>( 281 fmt::format("Cannot get sensor type from sensor path {}", 282 sensorPath) 283 .c_str()); 284 throw std::runtime_error("Invalid sensor path"); 285 } 286 287 sensorPath = sensorPath.substr(0, pos); 288 return sensorPath.substr(sensorPath.find_last_of('/') + 1); 289 } 290 291 bool ThresholdAlarmLogger::skipSensorType(const std::string& type) 292 { 293 return (type == "utilization"); 294 } 295 296 std::string ThresholdAlarmLogger::getCallout(const std::string& sensorPath) 297 { 298 const std::array<std::string, 2> assocTypes{"inventory", "chassis"}; 299 300 // Different implementations handle the association to the FRU 301 // differently: 302 // * phosphor-inventory-manager uses the 'inventory' association 303 // to point to the FRU. 304 // * dbus-sensors/entity-manager uses the 'chassis' association'. 305 // * For virtual sensors, no association. 306 307 for (const auto& assocType : assocTypes) 308 { 309 auto assocPath = sensorPath + "/" + assocType; 310 311 try 312 { 313 auto endpoints = SDBusPlus::getProperty<std::vector<std::string>>( 314 bus, assocPath, assocInterface, "endpoints"); 315 316 if (!endpoints.empty()) 317 { 318 return endpoints[0]; 319 } 320 } 321 catch (const DBusServiceError& e) 322 { 323 // The association doesn't exist 324 continue; 325 } 326 } 327 328 return std::string{}; 329 } 330 331 } // namespace sensor::monitor 332