blob: 7d23077ce84e5b61c6add7c9ef9c996b14f03e56 [file] [log] [blame]
Brad Bishopc1283ae2017-05-20 21:42:38 -04001#pragma once
2
3#include "data_types.hpp"
4
William A. Kennington III223c4092018-10-19 15:56:09 -07005#include <sdeventplus/clock.hpp>
6#include <sdeventplus/event.hpp>
7#include <sdeventplus/utility/timer.hpp>
Patrick Venture3d6d3182018-08-31 09:33:09 -07008
George Liu3fe976c2022-06-21 09:37:04 +08009#include <chrono>
10#include <cstddef>
11
Brad Bishopc1283ae2017-05-20 21:42:38 -040012namespace phosphor
13{
14namespace dbus
15{
16namespace monitoring
17{
18
19/** @class Callback
20 * @brief Callback interface.
21 *
22 * Callbacks of any type can be run.
23 */
24class Callback
25{
Brad Bishopd1eac882018-03-29 10:34:05 -040026 public:
27 Callback() = default;
28 Callback(const Callback&) = delete;
29 Callback(Callback&&) = default;
30 Callback& operator=(const Callback&) = delete;
31 Callback& operator=(Callback&&) = default;
32 virtual ~Callback() = default;
Brad Bishopc1283ae2017-05-20 21:42:38 -040033
Brad Bishopd1eac882018-03-29 10:34:05 -040034 /** @brief Run the callback.
35 * @param[in] ctx - caller context
36 * Context could be Startup or Signal
37 * Startup: Callback is called as part of process startup.
38 * Signal: Callback is called as part of watch condition has been met.
39 *
40 */
George Liu5e6b51d2022-06-21 16:59:39 +080041 virtual void operator()(Context /* ctx */) = 0;
Marri Devender Rao70aafbb2018-04-12 01:11:48 -050042
43 /** @brief Run the callback.
44 * @param[in] ctx - caller context
45 * Context could be Startup or Signal
46 * Startup: Callback is called as part of process startup.
47 * Signal: Callback is called as part of watch condition has been met.
48 * @param[in] msg - The sdbusplus signal message
49 */
George Liu5e6b51d2022-06-21 16:59:39 +080050 virtual void operator()(Context /* ctx */,
Patrick Williamseab4f8c2024-08-16 15:20:10 -040051 sdbusplus::message_t& /* msg */) {};
Brad Bishopc1283ae2017-05-20 21:42:38 -040052};
53
Brad Bishop4041d722017-05-21 10:06:07 -040054/** @class Conditional
55 * @brief Condition interface.
56 *
57 * Conditions of any type can be tested for true or false.
58 */
59class Conditional
60{
Brad Bishopd1eac882018-03-29 10:34:05 -040061 public:
62 Conditional() = default;
63 Conditional(const Conditional&) = delete;
64 Conditional(Conditional&&) = default;
65 Conditional& operator=(const Conditional&) = delete;
66 Conditional& operator=(Conditional&&) = default;
67 virtual ~Conditional() = default;
Brad Bishop4041d722017-05-21 10:06:07 -040068
Brad Bishopd1eac882018-03-29 10:34:05 -040069 /** @brief Test the condition. */
70 virtual bool operator()() = 0;
Brad Bishop4041d722017-05-21 10:06:07 -040071};
72
73/** @class IndexedConditional
74 * @brief Condition with an index.
75 */
76class IndexedConditional : public Conditional
77{
Brad Bishopd1eac882018-03-29 10:34:05 -040078 public:
79 IndexedConditional() = delete;
80 IndexedConditional(const IndexedConditional&) = delete;
81 IndexedConditional(IndexedConditional&&) = default;
82 IndexedConditional& operator=(const IndexedConditional&) = delete;
83 IndexedConditional& operator=(IndexedConditional&&) = default;
84 virtual ~IndexedConditional() = default;
Brad Bishop4041d722017-05-21 10:06:07 -040085
Brad Bishopd1eac882018-03-29 10:34:05 -040086 explicit IndexedConditional(const PropertyIndex& conditionIndex) :
87 Conditional(), index(conditionIndex)
George Liu3fe976c2022-06-21 09:37:04 +080088 {}
Brad Bishop4041d722017-05-21 10:06:07 -040089
Brad Bishopd1eac882018-03-29 10:34:05 -040090 /** @brief Test the condition. */
91 virtual bool operator()() override = 0;
Brad Bishop4041d722017-05-21 10:06:07 -040092
Brad Bishopd1eac882018-03-29 10:34:05 -040093 protected:
94 /** @brief Property names and their associated storage. */
95 const PropertyIndex& index;
Brad Bishop4041d722017-05-21 10:06:07 -040096};
97
Brad Bishopc1283ae2017-05-20 21:42:38 -040098/** @class IndexedCallback
99 * @brief Callback with an index.
100 */
101class IndexedCallback : public Callback
102{
Brad Bishopd1eac882018-03-29 10:34:05 -0400103 public:
104 IndexedCallback() = delete;
105 IndexedCallback(const IndexedCallback&) = delete;
106 IndexedCallback(IndexedCallback&&) = default;
107 IndexedCallback& operator=(const IndexedCallback&) = delete;
108 IndexedCallback& operator=(IndexedCallback&&) = default;
109 virtual ~IndexedCallback() = default;
110 explicit IndexedCallback(const PropertyIndex& callbackIndex) :
111 Callback(), index(callbackIndex)
George Liu3fe976c2022-06-21 09:37:04 +0800112 {}
Brad Bishopc1283ae2017-05-20 21:42:38 -0400113
Brad Bishopd1eac882018-03-29 10:34:05 -0400114 /** @brief Run the callback. */
115 virtual void operator()(Context ctx) override = 0;
Brad Bishopc1283ae2017-05-20 21:42:38 -0400116
Brad Bishopd1eac882018-03-29 10:34:05 -0400117 protected:
118 /** @brief Property names and their associated storage. */
119 const PropertyIndex& index;
Brad Bishopc1283ae2017-05-20 21:42:38 -0400120};
121
Brad Bishop49e66172017-05-23 19:16:21 -0400122/** @class GroupOfCallbacks
123 * @brief Invoke multiple callbacks.
124 *
Gunnar Mills78199b42017-10-25 16:30:18 -0500125 * A group of callbacks is implemented as a vector of array indices
Brad Bishop49e66172017-05-23 19:16:21 -0400126 * into an external array of callbacks. The group function call
Gunnar Mills78199b42017-10-25 16:30:18 -0500127 * operator traverses the vector of indices, invoking each
Brad Bishop49e66172017-05-23 19:16:21 -0400128 * callback.
129 *
130 * @tparam CallbackAccess - Access to the array of callbacks.
131 */
Patrick Venture3d6d3182018-08-31 09:33:09 -0700132template <typename CallbackAccess>
133class GroupOfCallbacks : public Callback
Brad Bishop49e66172017-05-23 19:16:21 -0400134{
Brad Bishopd1eac882018-03-29 10:34:05 -0400135 public:
136 GroupOfCallbacks() = delete;
137 GroupOfCallbacks(const GroupOfCallbacks&) = delete;
138 GroupOfCallbacks(GroupOfCallbacks&&) = default;
139 GroupOfCallbacks& operator=(const GroupOfCallbacks&) = delete;
140 GroupOfCallbacks& operator=(GroupOfCallbacks&&) = default;
141 ~GroupOfCallbacks() = default;
142 explicit GroupOfCallbacks(const std::vector<size_t>& graphEntry) :
143 graph(graphEntry)
George Liu3fe976c2022-06-21 09:37:04 +0800144 {}
Brad Bishop49e66172017-05-23 19:16:21 -0400145
Brad Bishopd1eac882018-03-29 10:34:05 -0400146 /** @brief Run the callbacks. */
147 void operator()(Context ctx) override
148 {
149 for (auto e : graph)
Brad Bishop49e66172017-05-23 19:16:21 -0400150 {
Brad Bishopd1eac882018-03-29 10:34:05 -0400151 (*CallbackAccess::get()[e])(ctx);
Brad Bishop49e66172017-05-23 19:16:21 -0400152 }
Brad Bishopd1eac882018-03-29 10:34:05 -0400153 }
Brad Bishop49e66172017-05-23 19:16:21 -0400154
Brad Bishopd1eac882018-03-29 10:34:05 -0400155 private:
156 /** @brief The offsets of the callbacks in the group. */
157 const std::vector<size_t>& graph;
Brad Bishop49e66172017-05-23 19:16:21 -0400158};
159
Brad Bishop4041d722017-05-21 10:06:07 -0400160/** @class ConditionalCallback
Manojkiran Eda0c1e0242024-06-17 12:02:59 +0530161 * @brief Callback adaptor that associates a condition with a callback.
Brad Bishop4041d722017-05-21 10:06:07 -0400162 */
Patrick Venture3d6d3182018-08-31 09:33:09 -0700163template <typename CallbackAccess>
164class ConditionalCallback : public Callback
Brad Bishop4041d722017-05-21 10:06:07 -0400165{
Brad Bishopd1eac882018-03-29 10:34:05 -0400166 public:
167 ConditionalCallback() = delete;
168 ConditionalCallback(const ConditionalCallback&) = delete;
169 ConditionalCallback(ConditionalCallback&&) = default;
170 ConditionalCallback& operator=(const ConditionalCallback&) = delete;
171 ConditionalCallback& operator=(ConditionalCallback&&) = default;
172 virtual ~ConditionalCallback() = default;
173 ConditionalCallback(const std::vector<size_t>& graphEntry,
Patrick Williamseab4f8c2024-08-16 15:20:10 -0400174 Conditional& cond) : graph(graphEntry), condition(cond)
George Liu3fe976c2022-06-21 09:37:04 +0800175 {}
Brad Bishop4041d722017-05-21 10:06:07 -0400176
Brad Bishopd1eac882018-03-29 10:34:05 -0400177 /** @brief Run the callback if the condition is satisfied. */
178 virtual void operator()(Context ctx) override
179 {
180 if (condition())
Brad Bishop4041d722017-05-21 10:06:07 -0400181 {
Brad Bishopd1eac882018-03-29 10:34:05 -0400182 (*CallbackAccess::get()[graph[0]])(ctx);
Brad Bishop4041d722017-05-21 10:06:07 -0400183 }
Brad Bishopd1eac882018-03-29 10:34:05 -0400184 }
Brad Bishop4041d722017-05-21 10:06:07 -0400185
Brad Bishopd1eac882018-03-29 10:34:05 -0400186 protected:
187 /** @brief The index of the callback to conditionally invoke. */
188 const std::vector<size_t>& graph;
Brad Bishop4041d722017-05-21 10:06:07 -0400189
Brad Bishopd1eac882018-03-29 10:34:05 -0400190 /** @brief The condition to test. */
191 Conditional& condition;
Brad Bishop4041d722017-05-21 10:06:07 -0400192};
193
Brad Bishop3539db62017-05-30 14:21:12 -0400194/** @class DeferrableCallback
195 *
196 * Deferrable callbacks wait a configurable period before
197 * invoking their associated callback.
198 *
Gunnar Mills78199b42017-10-25 16:30:18 -0500199 * When the callback condition is initially met, start a timer. If the
Brad Bishop3539db62017-05-30 14:21:12 -0400200 * condition is tested again before the timer expires and it is not
201 * met cancel the timer. If the timer expires invoke the associated
202 * callback.
203 *
204 * @tparam CallbackAccess - Provide access to callback group instances.
Brad Bishop3539db62017-05-30 14:21:12 -0400205 */
William A. Kennington III223c4092018-10-19 15:56:09 -0700206template <typename CallbackAccess>
Brad Bishop3539db62017-05-30 14:21:12 -0400207class DeferrableCallback : public ConditionalCallback<CallbackAccess>
208{
Brad Bishopd1eac882018-03-29 10:34:05 -0400209 public:
William A. Kennington III223c4092018-10-19 15:56:09 -0700210 using TimerType =
211 sdeventplus::utility::Timer<sdeventplus::ClockId::Monotonic>;
212
Brad Bishopd1eac882018-03-29 10:34:05 -0400213 DeferrableCallback() = delete;
214 DeferrableCallback(const DeferrableCallback&) = delete;
William A. Kennington III223c4092018-10-19 15:56:09 -0700215 DeferrableCallback(DeferrableCallback&&) = delete;
Brad Bishopd1eac882018-03-29 10:34:05 -0400216 DeferrableCallback& operator=(const DeferrableCallback&) = delete;
William A. Kennington III223c4092018-10-19 15:56:09 -0700217 DeferrableCallback& operator=(DeferrableCallback&&) = delete;
Brad Bishopd1eac882018-03-29 10:34:05 -0400218 ~DeferrableCallback() = default;
Brad Bishop3539db62017-05-30 14:21:12 -0400219
Brad Bishopd1eac882018-03-29 10:34:05 -0400220 DeferrableCallback(const std::vector<size_t>& graphEntry, Conditional& cond,
221 const std::chrono::microseconds& delay) :
222 ConditionalCallback<CallbackAccess>(graphEntry, cond),
George Liuecef1192022-07-07 08:59:12 +0800223 delayInterval(delay)
George Liu3fe976c2022-06-21 09:37:04 +0800224 {}
Brad Bishop3539db62017-05-30 14:21:12 -0400225
Alexander Soldatov78a5df92018-11-23 14:37:50 +0300226 /** @brief Start internal timer if the condition is satisfied.
227 *
228 * When the timer expires, it calls operator() for the
229 * ConditionalCallback with the context saved in
230 * DeferrableCallback instance.
231 */
Brad Bishopd1eac882018-03-29 10:34:05 -0400232 void operator()(Context ctx) override
233 {
234 if (!timer)
Brad Bishop3539db62017-05-30 14:21:12 -0400235 {
Patrick Williamseab4f8c2024-08-16 15:20:10 -0400236 timer = std::make_unique<TimerType>(
237 sdeventplus::Event::get_default(),
238 // **INDENT-OFF**
239 [this](auto& /* source */) {
240 // The timer uses the context saved on timer enable
241 this->ConditionalCallback<CallbackAccess>::operator()(
242 this->ctx);
243 });
Brad Bishopd1eac882018-03-29 10:34:05 -0400244 // **INDENT-ON**
Brad Bishop3539db62017-05-30 14:21:12 -0400245 }
246
Brad Bishopd1eac882018-03-29 10:34:05 -0400247 if (this->condition())
248 {
William A. Kennington III223c4092018-10-19 15:56:09 -0700249 if (!timer->isEnabled())
Brad Bishopd1eac882018-03-29 10:34:05 -0400250 {
251 // This is the first time the condition evaluated.
Alexander Soldatov78a5df92018-11-23 14:37:50 +0300252 // Save current context for timer use.
253 this->ctx = ctx;
Brad Bishopd1eac882018-03-29 10:34:05 -0400254 // Start the countdown.
William A. Kennington III223c4092018-10-19 15:56:09 -0700255 timer->restartOnce(delayInterval);
Brad Bishopd1eac882018-03-29 10:34:05 -0400256 }
257 }
258 else
259 {
260 // The condition did not evaluate. Stop the countdown.
William A. Kennington III223c4092018-10-19 15:56:09 -0700261 timer->setEnabled(false);
Brad Bishopd1eac882018-03-29 10:34:05 -0400262 }
263 }
Brad Bishop3539db62017-05-30 14:21:12 -0400264
Brad Bishopd1eac882018-03-29 10:34:05 -0400265 private:
266 /** @brief The length to wait for the condition to stop evaluating. */
267 std::chrono::microseconds delayInterval;
268
269 /** @brief Delegated timer functions. */
George Liuecef1192022-07-07 08:59:12 +0800270 std::unique_ptr<TimerType> timer = nullptr;
Alexander Soldatov78a5df92018-11-23 14:37:50 +0300271
272 /** @brief Current context for timer. */
George Liuecef1192022-07-07 08:59:12 +0800273 Context ctx = Context::START;
Brad Bishop3539db62017-05-30 14:21:12 -0400274};
275
Brad Bishopc1283ae2017-05-20 21:42:38 -0400276} // namespace monitoring
277} // namespace dbus
278} // namespace phosphor