control: Add zone increase & decrease timers

Add the increase and decrease timers to the JSON based zone objects to
handle target increase and decrease on their configured intervals.

Change-Id: I55f777b5f88b6fc05937c7acf91428e31c138b90
Signed-off-by: Matthew Barth <msbarth@us.ibm.com>
diff --git a/control/json/zone.cpp b/control/json/zone.cpp
index 6249194..be827cf 100644
--- a/control/json/zone.cpp
+++ b/control/json/zone.cpp
@@ -27,6 +27,7 @@
 #include <sdeventplus/event.hpp>
 
 #include <algorithm>
+#include <chrono>
 #include <filesystem>
 #include <fstream>
 #include <iterator>
@@ -54,12 +55,15 @@
     ConfigBase(jsonObj),
     ThermalObject(bus, (fs::path{CONTROL_OBJPATH} /= getName()).c_str(), true),
     _manager(mgr), _incDelay(0), _floor(0), _target(0), _incDelta(0),
-    _decDelta(0), _requestTargetBase(0), _isActive(true)
+    _decDelta(0), _requestTargetBase(0), _isActive(true),
+    _incTimer(event, std::bind(&Zone::incTimerExpired, this)),
+    _decTimer(event, std::bind(&Zone::decTimerExpired, this))
 {
     // Increase delay is optional, defaults to 0
     if (jsonObj.contains("increase_delay"))
     {
-        _incDelay = jsonObj["increase_delay"].get<uint64_t>();
+        _incDelay =
+            std::chrono::seconds(jsonObj["increase_delay"].get<uint64_t>());
     }
     setDefaultCeiling(jsonObj);
     setDefaultFloor(jsonObj);
@@ -69,6 +73,9 @@
     {
         setInterfaces(jsonObj);
     }
+
+    // Start timer for fan target decreases
+    _decTimer.restart(_decInterval);
 }
 
 void Zone::addFan(std::unique_ptr<Fan> fan)
@@ -133,11 +140,18 @@
             requestTarget = _ceiling;
         }
         setTarget(requestTarget);
-        // TODO // Restart timer countdown for fan speed increase
-        // _incTimer.restartOnce(_incDelay);
+        // Restart timer countdown for fan target increase
+        _incTimer.restartOnce(_incDelay);
     }
 }
 
+void Zone::incTimerExpired()
+{
+    // Clear increase delta when timer expires allowing additional target
+    // increase requests or target decreases to occur
+    _incDelta = 0;
+}
+
 void Zone::requestDecrease(uint64_t targetDelta)
 {
     // Only decrease the lowest target delta requested
@@ -147,6 +161,40 @@
     }
 }
 
+void Zone::decTimerExpired()
+{
+    // Check all entries are set to allow a decrease
+    auto pred = [](auto const& entry) { return entry.second; };
+    auto decAllowed = std::all_of(_decAllowed.begin(), _decAllowed.end(), pred);
+
+    // Only decrease targets when allowed, a requested decrease target delta
+    // exists, where no requested increases exist and the increase timer is not
+    // running (i.e. not in the middle of increasing)
+    if (decAllowed && _decDelta != 0 && _incDelta == 0 &&
+        !_incTimer.isEnabled())
+    {
+        auto requestTarget = getRequestTargetBase();
+        // Request target should not start above ceiling
+        if (requestTarget > _ceiling)
+        {
+            requestTarget = _ceiling;
+        }
+        // Target can not go below the defined floor
+        if ((requestTarget < _decDelta) || (requestTarget - _decDelta < _floor))
+        {
+            requestTarget = _floor;
+        }
+        else
+        {
+            requestTarget = requestTarget - _decDelta;
+        }
+        setTarget(requestTarget);
+    }
+    // Clear decrease delta when timer expires
+    _decDelta = 0;
+    // Decrease timer is restarted since its repeating
+}
+
 void Zone::setPersisted(const std::string& intf, const std::string& prop)
 {
     if (std::find_if(_propsPersisted[intf].begin(), _propsPersisted[intf].end(),
@@ -226,7 +274,8 @@
                         entry("JSON=%s", jsonObj.dump().c_str()));
         throw std::runtime_error("Missing required zone's decrease interval");
     }
-    _decInterval = jsonObj["decrease_interval"].get<uint64_t>();
+    _decInterval =
+        std::chrono::seconds(jsonObj["decrease_interval"].get<uint64_t>());
 }
 
 void Zone::setInterfaces(const json& jsonObj)
diff --git a/control/json/zone.hpp b/control/json/zone.hpp
index 5256018..e39a735 100644
--- a/control/json/zone.hpp
+++ b/control/json/zone.hpp
@@ -22,8 +22,10 @@
 #include <nlohmann/json.hpp>
 #include <sdbusplus/bus.hpp>
 #include <sdeventplus/event.hpp>
+#include <sdeventplus/utility/timer.hpp>
 
 #include <any>
+#include <chrono>
 #include <functional>
 #include <map>
 #include <tuple>
@@ -39,6 +41,9 @@
 using ThermalObject = sdbusplus::server::object::object<
     sdbusplus::xyz::openbmc_project::Control::server::ThermalMode>;
 
+/* Dbus event timer */
+using Timer = sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic>;
+
 /**
  * @class Zone - Represents a configured fan control zone
  *
@@ -236,6 +241,12 @@
     void requestIncrease(uint64_t targetDelta);
 
     /**
+     * @brief Callback function for the increase timer that delays
+     * processing of requested target increases while fans are increasing
+     */
+    void incTimerExpired();
+
+    /**
      * @brief Calculate the lowest requested decrease target from the given
      * delta within a decrease interval.
      *
@@ -244,6 +255,12 @@
     void requestDecrease(uint64_t targetDelta);
 
     /**
+     * @brief Callback function for the decrease timer that processes any
+     * requested target decreases if allowed
+     */
+    void decTimerExpired();
+
+    /**
      * @brief Set the requested target base to be used as the target to base a
      * new requested target from
      *
@@ -334,10 +351,10 @@
     uint64_t _defaultFloor;
 
     /* Zone's increase delay(in seconds) (OPTIONAL) */
-    uint64_t _incDelay;
+    std::chrono::seconds _incDelay;
 
     /* Zone's decrease interval(in seconds) */
-    uint64_t _decInterval;
+    std::chrono::seconds _decInterval;
 
     /* The floor target to not go below */
     uint64_t _floor;
@@ -369,6 +386,12 @@
     /* Automatic fan control active state */
     bool _isActive;
 
+    /* The target increase timer object */
+    Timer _incTimer;
+
+    /* The target decrease timer object */
+    Timer _decTimer;
+
     /* Map of active fan control allowed by a string identifier */
     std::map<std::string, bool> _active;