| #include "watchdog_service.hpp" |
| |
| #include <exception> |
| #include <ipmid/api.hpp> |
| #include <phosphor-logging/elog-errors.hpp> |
| #include <phosphor-logging/elog.hpp> |
| #include <phosphor-logging/log.hpp> |
| #include <sdbusplus/bus.hpp> |
| #include <sdbusplus/message.hpp> |
| #include <stdexcept> |
| #include <string> |
| #include <xyz/openbmc_project/Common/error.hpp> |
| #include <xyz/openbmc_project/State/Watchdog/server.hpp> |
| |
| using phosphor::logging::elog; |
| using phosphor::logging::entry; |
| using phosphor::logging::level; |
| using phosphor::logging::log; |
| using sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; |
| using sdbusplus::xyz::openbmc_project::State::server::convertForMessage; |
| using sdbusplus::xyz::openbmc_project::State::server::Watchdog; |
| |
| static constexpr char wd_path[] = "/xyz/openbmc_project/watchdog/host0"; |
| static constexpr char wd_intf[] = "xyz.openbmc_project.State.Watchdog"; |
| static constexpr char prop_intf[] = "org.freedesktop.DBus.Properties"; |
| |
| ipmi::ServiceCache WatchdogService::wd_service(wd_intf, wd_path); |
| |
| WatchdogService::WatchdogService() : bus(ipmid_get_sd_bus_connection()) |
| { |
| } |
| |
| void WatchdogService::resetTimeRemaining(bool enableWatchdog) |
| { |
| bool wasValid = wd_service.isValid(bus); |
| auto request = wd_service.newMethodCall(bus, wd_intf, "ResetTimeRemaining"); |
| request.append(enableWatchdog); |
| auto response = bus.call(request); |
| if (response.is_method_error()) |
| { |
| wd_service.invalidate(); |
| if (wasValid) |
| { |
| // Retry the request once in case the cached service was stale |
| return resetTimeRemaining(enableWatchdog); |
| } |
| log<level::ERR>( |
| "WatchdogService: Method error resetting time remaining", |
| entry("ENABLE_WATCHDOG=%d", !!enableWatchdog)); |
| elog<InternalFailure>(); |
| } |
| } |
| |
| WatchdogService::Properties WatchdogService::getProperties() |
| { |
| bool wasValid = wd_service.isValid(bus); |
| auto request = wd_service.newMethodCall(bus, prop_intf, "GetAll"); |
| request.append(wd_intf); |
| auto response = bus.call(request); |
| if (response.is_method_error()) |
| { |
| wd_service.invalidate(); |
| if (wasValid) |
| { |
| // Retry the request once in case the cached service was stale |
| return getProperties(); |
| } |
| log<level::ERR>("WatchdogService: Method error getting properties"); |
| elog<InternalFailure>(); |
| } |
| try |
| { |
| std::map<std::string, std::variant<bool, uint64_t, std::string>> |
| properties; |
| response.read(properties); |
| Properties wd_prop; |
| wd_prop.initialized = std::get<bool>(properties.at("Initialized")); |
| wd_prop.enabled = std::get<bool>(properties.at("Enabled")); |
| wd_prop.expireAction = Watchdog::convertActionFromString( |
| std::get<std::string>(properties.at("ExpireAction"))); |
| wd_prop.timerUse = Watchdog::convertTimerUseFromString( |
| std::get<std::string>(properties.at("CurrentTimerUse"))); |
| wd_prop.expiredTimerUse = Watchdog::convertTimerUseFromString( |
| std::get<std::string>(properties.at("ExpiredTimerUse"))); |
| |
| wd_prop.interval = std::get<uint64_t>(properties.at("Interval")); |
| wd_prop.timeRemaining = |
| std::get<uint64_t>(properties.at("TimeRemaining")); |
| return wd_prop; |
| } |
| catch (const std::exception& e) |
| { |
| log<level::ERR>("WatchdogService: Decode error in get properties", |
| entry("ERROR=%s", e.what()), |
| entry("REPLY_SIG=%s", response.get_signature())); |
| elog<InternalFailure>(); |
| } |
| |
| // Needed instead of elog<InternalFailure>() since the compiler can't |
| // deduce the that elog<>() always throws |
| throw std::runtime_error( |
| "WatchdogService: Should not reach end of getProperties"); |
| } |
| |
| template <typename T> |
| T WatchdogService::getProperty(const std::string& key) |
| { |
| bool wasValid = wd_service.isValid(bus); |
| auto request = wd_service.newMethodCall(bus, prop_intf, "Get"); |
| request.append(wd_intf, key); |
| auto response = bus.call(request); |
| if (response.is_method_error()) |
| { |
| wd_service.invalidate(); |
| if (wasValid) |
| { |
| // Retry the request once in case the cached service was stale |
| return getProperty<T>(key); |
| } |
| log<level::ERR>("WatchdogService: Method error getting property", |
| entry("PROPERTY=%s", key.c_str())); |
| elog<InternalFailure>(); |
| } |
| try |
| { |
| std::variant<T> value; |
| response.read(value); |
| return std::get<T>(value); |
| } |
| catch (const std::exception& e) |
| { |
| log<level::ERR>("WatchdogService: Decode error in get property", |
| entry("PROPERTY=%s", key.c_str()), |
| entry("ERROR=%s", e.what()), |
| entry("REPLY_SIG=%s", response.get_signature())); |
| elog<InternalFailure>(); |
| } |
| |
| // Needed instead of elog<InternalFailure>() since the compiler can't |
| // deduce the that elog<>() always throws |
| throw std::runtime_error( |
| "WatchdogService: Should not reach end of getProperty"); |
| } |
| |
| template <typename T> |
| void WatchdogService::setProperty(const std::string& key, const T& val) |
| { |
| bool wasValid = wd_service.isValid(bus); |
| auto request = wd_service.newMethodCall(bus, prop_intf, "Set"); |
| request.append(wd_intf, key, std::variant<T>(val)); |
| auto response = bus.call(request); |
| if (response.is_method_error()) |
| { |
| wd_service.invalidate(); |
| if (wasValid) |
| { |
| // Retry the request once in case the cached service was stale |
| setProperty(key, val); |
| return; |
| } |
| log<level::ERR>("WatchdogService: Method error setting property", |
| entry("PROPERTY=%s", key.c_str())); |
| elog<InternalFailure>(); |
| } |
| } |
| |
| bool WatchdogService::getInitialized() |
| { |
| return getProperty<bool>("Initialized"); |
| } |
| |
| void WatchdogService::setInitialized(bool initialized) |
| { |
| setProperty("Initialized", initialized); |
| } |
| |
| void WatchdogService::setEnabled(bool enabled) |
| { |
| setProperty("Enabled", enabled); |
| } |
| |
| void WatchdogService::setExpireAction(Action expireAction) |
| { |
| setProperty("ExpireAction", convertForMessage(expireAction)); |
| } |
| |
| void WatchdogService::setTimerUse(TimerUse timerUse) |
| { |
| setProperty("CurrentTimerUse", convertForMessage(timerUse)); |
| } |
| |
| void WatchdogService::setExpiredTimerUse(TimerUse timerUse) |
| { |
| setProperty("ExpiredTimerUse", convertForMessage(timerUse)); |
| } |
| |
| void WatchdogService::setInterval(uint64_t interval) |
| { |
| setProperty("Interval", interval); |
| } |