#pragma once #include "types.hpp" #include #include namespace phosphor::fan::monitor { /** * @class PowerOffCause * * This abstract class provides a satisfied() pure virtual method * that is called to know if the system should be powered off due * to fan health. Each type of class that is derived from this * one provides different behavior, for example one may count * missing fans, and another may count nonfunctional fans. */ class PowerOffCause { public: PowerOffCause() = delete; virtual ~PowerOffCause() = default; PowerOffCause(const PowerOffCause&) = delete; PowerOffCause& operator=(const PowerOffCause&) = delete; PowerOffCause(PowerOffCause&&) = delete; PowerOffCause& operator=(PowerOffCause&&) = delete; /** * @brief Constructor * * @param[in] count - The number of items that is compared * against in the derived class. * @param[in] name - The name of the cause. Used for tracing. */ PowerOffCause(size_t count, const std::string& name) : _count(count), _name(std::to_string(count) + " " + name) {} /** * @brief Pure virtual that says if the system should be powered * off based on the fan health. * * @param[in] fanHealth - The FanHealth map * * @return bool - If system should be powered off */ virtual bool satisfied(const FanHealth& fanHealth) = 0; /** * @brief Returns the name of the cause. * * For example: "3 Missing Fans" * * @return std::string - The name */ const std::string& name() const { return _name; } protected: /** * @brief The number of fan health items that the derived * class uses to compare to the fan health status. * For example, a 3 for 3 missing fans. */ const size_t _count; /** * @brief The cause name */ const std::string _name; }; /** * @class MissingFanFRUCause * * This class provides a satisfied() method that checks for * missing fans in the fan health map. * */ class MissingFanFRUCause : public PowerOffCause { public: MissingFanFRUCause() = delete; ~MissingFanFRUCause() = default; MissingFanFRUCause(const MissingFanFRUCause&) = delete; MissingFanFRUCause& operator=(const MissingFanFRUCause&) = delete; MissingFanFRUCause(MissingFanFRUCause&&) = delete; MissingFanFRUCause& operator=(MissingFanFRUCause&&) = delete; /** * @brief Constructor * * @param[in] count - The minimum number of fans that must be * missing to need a power off. */ explicit MissingFanFRUCause(size_t count) : PowerOffCause(count, "Missing Fan FRUs") {} /** * @brief Returns true if 'count' or more fans are missing * to require a power off. * * @param[in] fanHealth - The FanHealth map */ bool satisfied(const FanHealth& fanHealth) override { size_t count = std::count_if(fanHealth.begin(), fanHealth.end(), [](const auto& fan) { return !std::get(fan.second); }); return count >= _count; } }; /** * @class NonfuncFanRotorCause * * This class provides a satisfied() method that checks for * nonfunctional fan rotors in the fan health map. */ class NonfuncFanRotorCause : public PowerOffCause { public: NonfuncFanRotorCause() = delete; ~NonfuncFanRotorCause() = default; NonfuncFanRotorCause(const NonfuncFanRotorCause&) = delete; NonfuncFanRotorCause& operator=(const NonfuncFanRotorCause&) = delete; NonfuncFanRotorCause(NonfuncFanRotorCause&&) = delete; NonfuncFanRotorCause& operator=(NonfuncFanRotorCause&&) = delete; /** * @brief Constructor * * @param[in] count - The minimum number of rotors that must be * nonfunctional to need a power off. */ explicit NonfuncFanRotorCause(size_t count) : PowerOffCause(count, "Nonfunctional Fan Rotors") {} /** * @brief Returns true if 'count' or more rotors are nonfunctional * to require a power off. * * @param[in] fanHealth - The FanHealth map */ bool satisfied(const FanHealth& fanHealth) override { size_t count = std::accumulate(fanHealth.begin(), fanHealth.end(), 0, [](int sum, const auto& fan) { const auto& tachs = std::get(fan.second); auto nonFuncTachs = std::count_if(tachs.begin(), tachs.end(), [](bool tach) { return !tach; }); return sum + nonFuncTachs; }); return count >= _count; } }; /** * @class FanFRUsWithNonfuncRotorsCause * * This class provides a satisfied() method that checks for * fans with nonfunctional fan rotors in the fan health map. */ class FanFRUsWithNonfuncRotorsCause : public PowerOffCause { public: FanFRUsWithNonfuncRotorsCause() = delete; ~FanFRUsWithNonfuncRotorsCause() = default; FanFRUsWithNonfuncRotorsCause(const FanFRUsWithNonfuncRotorsCause&) = delete; FanFRUsWithNonfuncRotorsCause& operator=(const FanFRUsWithNonfuncRotorsCause&) = delete; FanFRUsWithNonfuncRotorsCause(FanFRUsWithNonfuncRotorsCause&&) = delete; FanFRUsWithNonfuncRotorsCause& operator=(FanFRUsWithNonfuncRotorsCause&&) = delete; /** * @brief Constructor * * @param[in] count - The minimum number of fan FRUs with * nonfunctional rotors to need a power off. */ explicit FanFRUsWithNonfuncRotorsCause(size_t count) : PowerOffCause(count, "Fans with Nonfunctional Rotors") {} /** * @brief Returns true if 'count' or more fan FRUs have * nonfunctional rotors. * * @param[in] fanHealth - The FanHealth map */ bool satisfied(const FanHealth& fanHealth) override { size_t count = std::count_if(fanHealth.begin(), fanHealth.end(), [](const auto& fan) { const auto& tachs = std::get(fan.second); return std::any_of(tachs.begin(), tachs.end(), [](bool func) { return !func; }); }); return count >= _count; } }; } // namespace phosphor::fan::monitor