| #pragma once |
| #include <chrono> |
| #include <vector> |
| #include <algorithm> |
| #include <sdbusplus/bus.hpp> |
| #include "fan.hpp" |
| #include "types.hpp" |
| #include "timer.hpp" |
| |
| namespace phosphor |
| { |
| namespace fan |
| { |
| namespace control |
| { |
| |
| /** |
| * The mode fan control will run in: |
| * - init - only do the initialization steps |
| * - control - run normal control algorithms |
| */ |
| enum class Mode |
| { |
| init, |
| control |
| }; |
| |
| /** |
| * @class Represents a fan control zone, which is a group of fans |
| * that behave the same. |
| */ |
| class Zone |
| { |
| public: |
| |
| Zone() = delete; |
| Zone(const Zone&) = delete; |
| Zone(Zone&&) = default; |
| Zone& operator=(const Zone&) = delete; |
| Zone& operator=(Zone&&) = delete; |
| ~Zone() = default; |
| |
| /** |
| * Constructor |
| * Creates the appropriate fan objects based on |
| * the zone definition data passed in. |
| * |
| * @param[in] mode - mode of fan control |
| * @param[in] bus - the dbus object |
| * @param[in] events - sd_event pointer |
| * @param[in] def - the fan zone definition data |
| */ |
| Zone(Mode mode, |
| sdbusplus::bus::bus& bus, |
| phosphor::fan::event::EventPtr& events, |
| const ZoneDefinition& def); |
| |
| /** |
| * Sets all fans in the zone to the speed |
| * passed in when the zone is active |
| * |
| * @param[in] speed - the fan speed |
| */ |
| void setSpeed(uint64_t speed); |
| |
| /** |
| * Sets the zone to full speed regardless of zone's active state |
| */ |
| void setFullSpeed(); |
| |
| /** |
| * @brief Sets the automatic fan control allowed active state |
| * |
| * @param[in] group - A group that affects the active state |
| * @param[in] isActiveAllow - Active state according to group |
| */ |
| void setActiveAllow(const Group* group, bool isActiveAllow); |
| |
| /** |
| * @brief Sets a given object's property value |
| * |
| * @param[in] object - Name of the object containing the property |
| * @param[in] interface - Interface name containing the property |
| * @param[in] property - Property name |
| * @param[in] value - Property value |
| */ |
| template <typename T> |
| void setPropertyValue(const char* object, |
| const char* interface, |
| const char* property, |
| T value) |
| { |
| _properties[object][interface][property] = value; |
| }; |
| |
| /** |
| * @brief Get the value of an object's property |
| * |
| * @param[in] object - Name of the object containing the property |
| * @param[in] interface - Interface name containing the property |
| * @param[in] property - Property name |
| * |
| * @return - The property value |
| */ |
| template <typename T> |
| inline auto getPropertyValue(const std::string& object, |
| const std::string& interface, |
| const std::string& property) |
| { |
| return sdbusplus::message::variant_ns::get<T>( |
| _properties.at(object).at(interface).at(property)); |
| }; |
| |
| /** |
| * @brief Get the object's property variant |
| * |
| * @param[in] object - Name of the object containing the property |
| * @param[in] interface - Interface name containing the property |
| * @param[in] property - Property name |
| * |
| * @return - The property variant |
| */ |
| inline auto getPropValueVariant(const std::string& object, |
| const std::string& interface, |
| const std::string& property) |
| { |
| return _properties.at(object).at(interface).at(property); |
| }; |
| |
| /** |
| * @brief Initialize a set speed event properties and actions |
| * |
| * @param[in] event - Set speed event |
| */ |
| void initEvent(const SetSpeedEvent& event); |
| |
| /** |
| * @brief Removes all the set speed event properties and actions |
| * |
| * @param[in] event - Set speed event |
| */ |
| void removeEvent(const SetSpeedEvent& event); |
| |
| /** |
| * @brief Get the default floor speed |
| * |
| * @return - The defined default floor speed |
| */ |
| inline auto getDefFloor() |
| { |
| return _defFloorSpeed; |
| }; |
| |
| /** |
| * @brief Get the ceiling speed |
| * |
| * @return - The current ceiling speed |
| */ |
| inline auto& getCeiling() const |
| { |
| return _ceilingSpeed; |
| }; |
| |
| /** |
| * @brief Set the ceiling speed to the given speed |
| * |
| * @param[in] speed - Speed to set the ceiling to |
| */ |
| inline void setCeiling(uint64_t speed) |
| { |
| _ceilingSpeed = speed; |
| }; |
| |
| /** |
| * @brief Swaps the ceiling key value with what's given and |
| * returns the value that was swapped. |
| * |
| * @param[in] keyValue - New ceiling key value |
| * |
| * @return - Ceiling key value prior to swapping |
| */ |
| inline auto swapCeilingKeyValue(int64_t keyValue) |
| { |
| std::swap(_ceilingKeyValue, keyValue); |
| return keyValue; |
| }; |
| |
| /** |
| * @brief Get the increase speed delta |
| * |
| * @return - The current increase speed delta |
| */ |
| inline auto& getIncSpeedDelta() const |
| { |
| return _incSpeedDelta; |
| }; |
| |
| /** |
| * @brief Get the decrease speed delta |
| * |
| * @return - The current decrease speed delta |
| */ |
| inline auto& getDecSpeedDelta() const |
| { |
| return _decSpeedDelta; |
| }; |
| |
| /** |
| * @brief Set the floor speed to the given speed and increase target |
| * speed to the floor when target is below floor. |
| * |
| * @param[in] speed - Speed to set the floor to |
| */ |
| void setFloor(uint64_t speed); |
| |
| /** |
| * @brief Calculate the requested target speed from the given delta |
| * and increase the fan speeds, not going above the ceiling. |
| * |
| * @param[in] targetDelta - The delta to increase the target speed by |
| */ |
| void requestSpeedIncrease(uint64_t targetDelta); |
| |
| /** |
| * @brief Calculate the requested target speed from the given delta |
| * and increase the fan speeds, not going above the ceiling. |
| * |
| * @param[in] targetDelta - The delta to increase the target speed by |
| */ |
| void requestSpeedDecrease(uint64_t targetDelta); |
| |
| /** |
| * @brief Callback function for the increase timer that delays |
| * processing of requested speed increases while fans are increasing |
| */ |
| void incTimerExpired(); |
| |
| /** |
| * @brief Callback function for the decrease timer that processes any |
| * requested speed decreases if allowed |
| */ |
| void decTimerExpired(); |
| |
| /** |
| * @brief Callback function for event timers that processes the given |
| * action for a group |
| * |
| * @param[in] eventGroup - Group to process action on |
| * @param[in] eventAction - Event action to run |
| */ |
| void timerExpired(Group eventGroup, Action eventAction); |
| |
| private: |
| |
| /** |
| * The dbus object |
| */ |
| sdbusplus::bus::bus& _bus; |
| |
| /** |
| * Full speed for the zone |
| */ |
| const uint64_t _fullSpeed; |
| |
| /** |
| * The zone number |
| */ |
| const size_t _zoneNum; |
| |
| /** |
| * The default floor speed for the zone |
| */ |
| const uint64_t _defFloorSpeed; |
| |
| /** |
| * The default ceiling speed for the zone |
| */ |
| const uint64_t _defCeilingSpeed; |
| |
| /** |
| * The floor speed to not go below |
| */ |
| uint64_t _floorSpeed = _defFloorSpeed; |
| |
| /** |
| * The ceiling speed to not go above |
| */ |
| uint64_t _ceilingSpeed = _defCeilingSpeed; |
| |
| /** |
| * The previous sensor value for calculating the ceiling |
| */ |
| int64_t _ceilingKeyValue = 0; |
| |
| /** |
| * Automatic fan control active state |
| */ |
| bool _isActive = true; |
| |
| /** |
| * Target speed for this zone |
| */ |
| uint64_t _targetSpeed = _fullSpeed; |
| |
| /** |
| * Speed increase delta |
| */ |
| uint64_t _incSpeedDelta = 0; |
| |
| /** |
| * Speed decrease delta |
| */ |
| uint64_t _decSpeedDelta = 0; |
| |
| /** |
| * Speed increase delay in seconds |
| */ |
| std::chrono::seconds _incDelay; |
| |
| /** |
| * Speed decrease interval in seconds |
| */ |
| std::chrono::seconds _decInterval; |
| |
| /** |
| * The increase timer object |
| */ |
| phosphor::fan::util::Timer _incTimer; |
| |
| /** |
| * The decrease timer object |
| */ |
| phosphor::fan::util::Timer _decTimer; |
| |
| /** |
| * Dbus event used on set speed event timers |
| */ |
| phosphor::fan::event::EventPtr& _sdEvents; |
| |
| /** |
| * The vector of fans in this zone |
| */ |
| std::vector<std::unique_ptr<Fan>> _fans; |
| |
| /** |
| * @brief Map of object property values |
| */ |
| std::map<std::string, |
| std::map<std::string, |
| std::map<std::string, |
| PropertyVariantType>>> _properties; |
| |
| /** |
| * @brief Map of active fan control allowed by groups |
| */ |
| std::map<const Group, bool> _active; |
| |
| /** |
| * @brief List of signal event arguments and Dbus matches for callbacks |
| */ |
| std::vector<SignalEvent> _signalEvents; |
| |
| /** |
| * @brief List of timers for events |
| */ |
| std::vector<std::unique_ptr<phosphor::fan::util::Timer>> _timerEvents; |
| |
| /** |
| * @brief Refresh the given property's cached value |
| * |
| * @param[in] bus - the bus to use |
| * @param[in] path - the dbus path name |
| * @param[in] iface - the dbus interface name |
| * @param[in] prop - the property name |
| */ |
| void refreshProperty(sdbusplus::bus::bus& bus, |
| const std::string& path, |
| const std::string& iface, |
| const std::string& prop); |
| |
| /** |
| * @brief Get a property value from the path/interface given |
| * |
| * @param[in] bus - the bus to use |
| * @param[in] path - the dbus path name |
| * @param[in] iface - the dbus interface name |
| * @param[in] prop - the property name |
| * @param[out] value - the value of the property |
| */ |
| static void getProperty(sdbusplus::bus::bus& bus, |
| const std::string& path, |
| const std::string& iface, |
| const std::string& prop, |
| PropertyVariantType& value); |
| |
| /** |
| * @brief Dbus signal change callback handler |
| * |
| * @param[in] msg - Expanded sdbusplus message data |
| * @param[in] eventData - The single event's data |
| */ |
| void handleEvent(sdbusplus::message::message& msg, |
| const EventData* eventData); |
| }; |
| |
| } |
| } |
| } |