blob: d5705ef892052b9cae2cd439fb0a21da796a628f [file] [log] [blame]
Matthew Barthfd05d642019-11-14 15:01:57 -06001/**
2 * Copyright © 2019 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 */
Jolie Ku1a568652020-08-24 16:32:15 +080016#include "json_parser.hpp"
Matthew Barth2d2caa32020-05-26 11:07:24 -050017
18#include "anyof.hpp"
19#include "fallback.hpp"
20#include "gpio.hpp"
Jolie Ku1a568652020-08-24 16:32:15 +080021#include "json_config.hpp"
Matthew Barth2d2caa32020-05-26 11:07:24 -050022#include "sdbusplus.hpp"
23#include "tach.hpp"
24
Matthew Barth0961fae2019-11-15 09:00:27 -060025#include <nlohmann/json.hpp>
26#include <phosphor-logging/log.hpp>
Matthew Barth5060b102019-12-16 10:46:35 -060027#include <sdbusplus/bus.hpp>
Mike Cappsa35a8902021-06-10 10:20:14 -040028#include <xyz/openbmc_project/Logging/Create/server.hpp>
29#include <xyz/openbmc_project/Logging/Entry/server.hpp>
Matthew Barthfd05d642019-11-14 15:01:57 -060030
Matthew Barth2d2caa32020-05-26 11:07:24 -050031#include <filesystem>
32#include <fstream>
33#include <string>
Matthew Barthfd05d642019-11-14 15:01:57 -060034
35namespace phosphor
36{
37namespace fan
38{
39namespace presence
40{
41
Matthew Barth0961fae2019-11-15 09:00:27 -060042using json = nlohmann::json;
43namespace fs = std::filesystem;
44using namespace phosphor::logging;
45
Matthew Barthfd05d642019-11-14 15:01:57 -060046policies JsonConfig::_policies;
Matthew Barth2d2caa32020-05-26 11:07:24 -050047const std::map<std::string, methodHandler> JsonConfig::_methods = {
48 {"tach", method::getTach}, {"gpio", method::getGpio}};
49const std::map<std::string, rpolicyHandler> JsonConfig::_rpolicies = {
50 {"anyof", rpolicy::getAnyof}, {"fallback", rpolicy::getFallback}};
Matthew Barthfd05d642019-11-14 15:01:57 -060051
Mike Cappsa35a8902021-06-10 10:20:14 -040052const auto loggingPath = "/xyz/openbmc_project/logging";
53const auto loggingCreateIface = "xyz.openbmc_project.Logging.Create";
54
Patrick Williams61b73292023-05-10 07:50:12 -050055JsonConfig::JsonConfig(sdbusplus::bus_t& bus) : _bus(bus) {}
Jolie Ku1a568652020-08-24 16:32:15 +080056
Matthew Barth5b839912021-06-21 14:46:34 -050057void JsonConfig::start()
Matt Spinler0daedd12021-01-25 14:46:03 -060058{
Matthew Barth5b839912021-06-21 14:46:34 -050059 using config = fan::JsonConfig;
60
Matt Spinlerdfc8c4d2022-06-22 16:52:27 -050061 if (!_loaded)
Matt Spinler0daedd12021-01-25 14:46:03 -060062 {
Mike Capps808d7fe2022-06-13 10:12:16 -040063 process(config::load(config::getConfFile(confAppName, confFileName)));
Matt Spinlerdfc8c4d2022-06-22 16:52:27 -050064
65 _loaded = true;
66
67 for (auto& p : _policies)
68 {
69 p->monitor();
70 }
Matt Spinler0daedd12021-01-25 14:46:03 -060071 }
Matthew Barthfd05d642019-11-14 15:01:57 -060072}
73
74const policies& JsonConfig::get()
75{
76 return _policies;
77}
78
Mike Capps808d7fe2022-06-13 10:12:16 -040079void JsonConfig::sighupHandler(sdeventplus::source::Signal& /*sigSrc*/,
80 const struct signalfd_siginfo* /*sigInfo*/)
Matthew Barthf3e70472019-12-03 13:33:20 -060081{
82 try
83 {
Jolie Ku1a568652020-08-24 16:32:15 +080084 using config = fan::JsonConfig;
85
Matt Spinler9e9f5992020-09-30 08:29:24 -050086 _reporter.reset();
87
Matthew Barthf3e70472019-12-03 13:33:20 -060088 // Load and process the json configuration
Mike Capps808d7fe2022-06-13 10:12:16 -040089 process(config::load(config::getConfFile(confAppName, confFileName)));
Jolie Ku1a568652020-08-24 16:32:15 +080090
Matthew Barth2d2caa32020-05-26 11:07:24 -050091 for (auto& p : _policies)
Matthew Barthf3e70472019-12-03 13:33:20 -060092 {
93 p->monitor();
94 }
95 log<level::INFO>("Configuration loaded successfully");
96 }
Patrick Williamsddb773b2021-10-06 11:24:49 -050097 catch (const std::runtime_error& re)
Matthew Barthf3e70472019-12-03 13:33:20 -060098 {
99 log<level::ERR>("Error loading config, no config changes made",
100 entry("LOAD_ERROR=%s", re.what()));
101 }
102}
103
Matthew Barth4a94dec2019-11-15 10:40:47 -0600104void JsonConfig::process(const json& jsonConf)
105{
Matthew Barthf3e70472019-12-03 13:33:20 -0600106 policies policies;
107 std::vector<fanPolicy> fans;
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600108 // Set the expected number of fan entries
109 // to be size of the list of fan json config entries
110 // (Must be done to eliminate vector reallocation of fan references)
Matt Spinler9e9f5992020-09-30 08:29:24 -0500111 fans.reserve(jsonConf.size());
112 for (auto& member : jsonConf)
Matthew Barth4a94dec2019-11-15 10:40:47 -0600113 {
Matthew Barthe7566632019-11-18 16:13:04 -0600114 if (!member.contains("name") || !member.contains("path") ||
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600115 !member.contains("methods") || !member.contains("rpolicy"))
Matthew Barth4a94dec2019-11-15 10:40:47 -0600116 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500117 log<level::ERR>("Missing required fan presence properties",
118 entry("REQUIRED_PROPERTIES=%s",
119 "{name, path, methods, rpolicy}"));
Matthew Barth4a94dec2019-11-15 10:40:47 -0600120 throw std::runtime_error(
121 "Missing required fan presence properties");
122 }
Matthew Barthe7566632019-11-18 16:13:04 -0600123
124 // Loop thru the configured methods of presence detection
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600125 std::vector<std::unique_ptr<PresenceSensor>> sensors;
Matthew Barthe7566632019-11-18 16:13:04 -0600126 for (auto& method : member["methods"].items())
127 {
128 if (!method.value().contains("type"))
129 {
130 log<level::ERR>(
131 "Missing required fan presence method type",
132 entry("FAN_NAME=%s",
Matthew Barth2d2caa32020-05-26 11:07:24 -0500133 member["name"].get<std::string>().c_str()));
Matthew Barthe7566632019-11-18 16:13:04 -0600134 throw std::runtime_error(
135 "Missing required fan presence method type");
136 }
137 // The method type of fan presence detection
138 // (Must have a supported function within the method namespace)
139 auto type = method.value()["type"].get<std::string>();
140 std::transform(type.begin(), type.end(), type.begin(), tolower);
141 auto func = _methods.find(type);
142 if (func != _methods.end())
143 {
144 // Call function for method type
Matthew Barthf3e70472019-12-03 13:33:20 -0600145 auto sensor = func->second(fans.size(), method.value());
Matthew Barthe7566632019-11-18 16:13:04 -0600146 if (sensor)
147 {
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600148 sensors.emplace_back(std::move(sensor));
Matthew Barthe7566632019-11-18 16:13:04 -0600149 }
150 }
151 else
152 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500153 log<level::ERR>(
154 "Invalid fan presence method type",
155 entry("FAN_NAME=%s",
156 member["name"].get<std::string>().c_str()),
157 entry("METHOD_TYPE=%s", type.c_str()));
Matthew Barthe7566632019-11-18 16:13:04 -0600158 throw std::runtime_error("Invalid fan presence method type");
159 }
160 }
Matt Spinler9e9f5992020-09-30 08:29:24 -0500161
162 // Get the amount of time a fan must be not present before
163 // creating an error.
164 std::optional<size_t> timeUntilError;
165 if (member.contains("fan_missing_error_time"))
166 {
167 timeUntilError = member["fan_missing_error_time"].get<size_t>();
168 }
169
Matt Spinlerbc4179e2022-10-04 15:15:06 -0500170 std::unique_ptr<EEPROMDevice> eepromDevice;
171 if (member.contains("eeprom"))
172 {
173 const auto& eeprom = member.at("eeprom");
174 if (!eeprom.contains("bus_address") ||
175 !eeprom.contains("driver_name") ||
176 !eeprom.contains("bind_delay_ms"))
177 {
178 log<level::ERR>(
179 "Missing address, driver_name, or bind_delay_ms in eeprom "
180 "section",
181 entry("FAN_NAME=%s",
182 member["name"].get<std::string>().c_str()));
183
184 throw std::runtime_error("Missing address, driver_name, or "
185 "bind_delay_ms in eeprom section");
186 }
187 eepromDevice = std::make_unique<EEPROMDevice>(
188 eeprom["bus_address"].get<std::string>(),
189 eeprom["driver_name"].get<std::string>(),
190 eeprom["bind_delay_ms"].get<size_t>());
191 }
192
Patrick Williamsdfddd642024-08-16 15:21:51 -0400193 auto fan =
194 std::make_tuple(member["name"], member["path"], timeUntilError);
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600195 // Create a fan object
Matthew Barthf3e70472019-12-03 13:33:20 -0600196 fans.emplace_back(std::make_tuple(fan, std::move(sensors)));
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600197
198 // Add fan presence policy
Patrick Williamsdfddd642024-08-16 15:21:51 -0400199 auto policy =
200 getPolicy(member["rpolicy"], fans.back(), std::move(eepromDevice));
Matthew Barthf3e70472019-12-03 13:33:20 -0600201 if (policy)
202 {
203 policies.emplace_back(std::move(policy));
204 }
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600205 }
Matthew Barthf3e70472019-12-03 13:33:20 -0600206
207 // Success, refresh fans and policies lists
208 _fans.clear();
209 _fans.swap(fans);
210
211 _policies.clear();
212 _policies.swap(policies);
Matt Spinlere8122392020-09-24 13:22:18 -0500213
214 // Create the error reporter class if necessary
Matt Spinler9e9f5992020-09-30 08:29:24 -0500215 if (std::any_of(_fans.begin(), _fans.end(), [](const auto& fan) {
Patrick Williamsdfddd642024-08-16 15:21:51 -0400216 return std::get<std::optional<size_t>>(std::get<Fan>(fan)) !=
217 std::nullopt;
218 }))
Matt Spinlere8122392020-09-24 13:22:18 -0500219 {
Matt Spinler9e9f5992020-09-30 08:29:24 -0500220 _reporter = std::make_unique<ErrorReporter>(_bus, _fans);
Matt Spinlere8122392020-09-24 13:22:18 -0500221 }
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600222}
223
Matthew Barth2d2caa32020-05-26 11:07:24 -0500224std::unique_ptr<RedundancyPolicy>
Matt Spinlerbc4179e2022-10-04 15:15:06 -0500225 JsonConfig::getPolicy(const json& rpolicy, const fanPolicy& fpolicy,
226 std::unique_ptr<EEPROMDevice> eepromDevice)
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600227{
228 if (!rpolicy.contains("type"))
229 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500230 log<level::ERR>(
231 "Missing required fan presence policy type",
232 entry("FAN_NAME=%s",
233 std::get<fanPolicyFanPos>(std::get<Fan>(fpolicy)).c_str()),
234 entry("REQUIRED_PROPERTIES=%s", "{type}"));
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600235 throw std::runtime_error("Missing required fan presence policy type");
236 }
237
238 // The redundancy policy type for fan presence detection
239 // (Must have a supported function within the rpolicy namespace)
240 auto type = rpolicy["type"].get<std::string>();
241 std::transform(type.begin(), type.end(), type.begin(), tolower);
242 auto func = _rpolicies.find(type);
243 if (func != _rpolicies.end())
244 {
Matthew Barthf3e70472019-12-03 13:33:20 -0600245 // Call function for redundancy policy type and return the policy
Matt Spinlerbc4179e2022-10-04 15:15:06 -0500246 return func->second(fpolicy, std::move(eepromDevice));
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600247 }
248 else
249 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500250 log<level::ERR>(
251 "Invalid fan presence policy type",
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600252 entry("FAN_NAME=%s",
Matthew Barth2d2caa32020-05-26 11:07:24 -0500253 std::get<fanPolicyFanPos>(std::get<Fan>(fpolicy)).c_str()),
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600254 entry("RPOLICY_TYPE=%s", type.c_str()));
255 throw std::runtime_error("Invalid fan presence methods policy type");
Matthew Barth4a94dec2019-11-15 10:40:47 -0600256 }
257}
258
Matthew Barthe7566632019-11-18 16:13:04 -0600259/**
260 * Methods of fan presence detection function definitions
261 */
262namespace method
263{
Matthew Barth2d2caa32020-05-26 11:07:24 -0500264// Get a constructed presence sensor for fan presence detection by tach
265std::unique_ptr<PresenceSensor> getTach(size_t fanIndex, const json& method)
266{
267 if (!method.contains("sensors") || method["sensors"].size() == 0)
Matthew Barthe7566632019-11-18 16:13:04 -0600268 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500269 log<level::ERR>("Missing required tach method properties",
270 entry("FAN_ENTRY=%d", fanIndex),
271 entry("REQUIRED_PROPERTIES=%s", "{sensors}"));
272 throw std::runtime_error("Missing required tach method properties");
Matthew Barthe7566632019-11-18 16:13:04 -0600273 }
274
Matthew Barth2d2caa32020-05-26 11:07:24 -0500275 std::vector<std::string> sensors;
276 for (auto& sensor : method["sensors"])
Matthew Barthe7566632019-11-18 16:13:04 -0600277 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500278 sensors.emplace_back(sensor.get<std::string>());
Matthew Barthe7566632019-11-18 16:13:04 -0600279 }
280
Patrick Williamsdfddd642024-08-16 15:21:51 -0400281 return std::make_unique<PolicyAccess<Tach, JsonConfig>>(
282 fanIndex, std::move(sensors));
Matthew Barth2d2caa32020-05-26 11:07:24 -0500283}
284
285// Get a constructed presence sensor for fan presence detection by gpio
286std::unique_ptr<PresenceSensor> getGpio(size_t fanIndex, const json& method)
287{
288 if (!method.contains("physpath") || !method.contains("devpath") ||
289 !method.contains("key"))
290 {
291 log<level::ERR>(
292 "Missing required gpio method properties",
293 entry("FAN_ENTRY=%d", fanIndex),
294 entry("REQUIRED_PROPERTIES=%s", "{physpath, devpath, key}"));
295 throw std::runtime_error("Missing required gpio method properties");
296 }
297
298 auto physpath = method["physpath"].get<std::string>();
299 auto devpath = method["devpath"].get<std::string>();
300 auto key = method["key"].get<unsigned int>();
301
Mike Cappsa35a8902021-06-10 10:20:14 -0400302 try
303 {
304 return std::make_unique<PolicyAccess<Gpio, JsonConfig>>(
305 fanIndex, physpath, devpath, key);
306 }
307 catch (const sdbusplus::exception_t& e)
308 {
309 namespace sdlogging = sdbusplus::xyz::openbmc_project::Logging::server;
310
311 log<level::ERR>(
Patrick Williamsfbf47032023-07-17 12:27:34 -0500312 std::format(
Mike Cappsa35a8902021-06-10 10:20:14 -0400313 "Error creating Gpio device bridge, hardware not detected: {}",
314 e.what())
315 .c_str());
316
317 auto severity =
318 sdlogging::convertForMessage(sdlogging::Entry::Level::Error);
319
320 std::map<std::string, std::string> additionalData{
321 {"PHYSPATH", physpath},
322 {"DEVPATH", devpath},
323 {"FANINDEX", std::to_string(fanIndex)}};
324
325 try
326 {
Mike Cappsa35a8902021-06-10 10:20:14 -0400327 util::SDBusPlus::lookupAndCallMethod(
328 loggingPath, loggingCreateIface, "Create",
329 "xyz.openbmc_project.Fan.Presence.Error.GPIODeviceUnavailable",
330 severity, additionalData);
331 }
332 catch (const util::DBusError& e)
333 {
Patrick Williamsfbf47032023-07-17 12:27:34 -0500334 log<level::ERR>(std::format("Call to create an error log for "
Mike Cappsa35a8902021-06-10 10:20:14 -0400335 "presence-sensor failure failed: {}",
336 e.what())
337 .c_str());
338 }
339
340 return std::make_unique<PolicyAccess<NullGpio, JsonConfig>>();
341 }
Matthew Barth2d2caa32020-05-26 11:07:24 -0500342}
343
Matthew Barthe7566632019-11-18 16:13:04 -0600344} // namespace method
345
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600346/**
347 * Redundancy policies for fan presence detection function definitions
348 */
349namespace rpolicy
350{
Matthew Barth2d2caa32020-05-26 11:07:24 -0500351// Get an `Anyof` redundancy policy for the fan
Matt Spinlerbc4179e2022-10-04 15:15:06 -0500352std::unique_ptr<RedundancyPolicy>
353 getAnyof(const fanPolicy& fan, std::unique_ptr<EEPROMDevice> eepromDevice)
Matthew Barth2d2caa32020-05-26 11:07:24 -0500354{
355 std::vector<std::reference_wrapper<PresenceSensor>> pSensors;
356 for (auto& fanSensor : std::get<fanPolicySensorListPos>(fan))
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600357 {
Matthew Barth2d2caa32020-05-26 11:07:24 -0500358 pSensors.emplace_back(*fanSensor);
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600359 }
360
Matt Spinlerbc4179e2022-10-04 15:15:06 -0500361 return std::make_unique<AnyOf>(std::get<fanPolicyFanPos>(fan), pSensors,
362 std::move(eepromDevice));
Matthew Barth2d2caa32020-05-26 11:07:24 -0500363}
Matthew Barth26ad44a2019-11-22 15:37:14 -0600364
Matthew Barth2d2caa32020-05-26 11:07:24 -0500365// Get a `Fallback` redundancy policy for the fan
Patrick Williamsdfddd642024-08-16 15:21:51 -0400366std::unique_ptr<RedundancyPolicy> getFallback(
367 const fanPolicy& fan, std::unique_ptr<EEPROMDevice> eepromDevice)
Matthew Barth2d2caa32020-05-26 11:07:24 -0500368{
369 std::vector<std::reference_wrapper<PresenceSensor>> pSensors;
370 for (auto& fanSensor : std::get<fanPolicySensorListPos>(fan))
371 {
372 // Place in the order given to fallback correctly
373 pSensors.emplace_back(*fanSensor);
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600374 }
375
Matt Spinlerbc4179e2022-10-04 15:15:06 -0500376 return std::make_unique<Fallback>(std::get<fanPolicyFanPos>(fan), pSensors,
377 std::move(eepromDevice));
Matthew Barth2d2caa32020-05-26 11:07:24 -0500378}
379
380} // namespace rpolicy
Matthew Barthaa8d81d2019-11-21 14:07:31 -0600381
Matthew Barthfd05d642019-11-14 15:01:57 -0600382} // namespace presence
383} // namespace fan
384} // namespace phosphor