GM6000 Digital Heater Controller Branch: main
SDX-1330
PeriodicScheduler.h
Go to the documentation of this file.
1#ifndef Cpl_System_PeriodicScheduler_h_
2#define Cpl_System_PeriodicScheduler_h_
3/*-----------------------------------------------------------------------------
4* This file is part of the Colony.Core Project. The Colony.Core Project is an
5* open source project with a BSD type of licensing agreement. See the license
6* agreement (license.txt) in the top/ directory or on the Internet at
7* http://integerfox.com/colony.core/license.txt
8*
9* Copyright (c) 2014-2022 John T. Taylor
10*
11* Redistributions of the source code must retain the above copyright notice.
12*----------------------------------------------------------------------------*/
13/** @file */
14
16
17/** Helper macro that is used to mark the end of an 'Interval Array'
18 */
19#define CPL_SYSTEM_PERIODIC_SCHEDULAR_END_INTERVALS {nullptr,{0,0},nullptr}
20
21
22 ///
23namespace Cpl {
24///
25namespace System {
26
27/** This concrete class is a 'policy' object that is used to add polled based,
28 cooperative monotonic scheduling to a Runnable object. Monotonic in this
29 content means that if a 'interval' method is scheduled to execute at a
30 periodic rate of 200Hz, then the method will be called on every 5ms boundary
31 of the system time.
32
33 The scheduler makes its best attempt at being monotonic and deterministic,
34 BUT because the scheduling is polled and cooperative (i.e. its is the
35 application's responsibility to not-overrun/over-allocate the processing
36 done during each interval) the actual the timing cannot be guaranteed.
37 That said, the scheduler will detect and report when the interval timing
38 slips.
39
40 The usage of this class is to implement (or extended) a Cpl::System::Runnable
41 object, i.e. invoked inside the 'forever' loop of the Runnable object's appRun()
42 method. The scheduler can be use within a 'bare' forever loop and it can be
43 used with existing event based Runnable objects to add periodic scheduling.
44 See Cpl::Dm::PeriodicScheduler for example extending an event based Runnable
45 object to include periodic scheduling.
46 */
48{
49public:
50 /** Definition for an interval method. This method is called when the
51 period time has expired for the interval.
52
53 Where:
54 currentTick - is current system when the interval method is called
55 currentInterval - is the deterministic interval boundary that is being
56 logically executed.
57 context - Application defined argument/value passed to the
58 interval method.
59
60 Example:
61 Given a interval method is scheduled to execute at 10Hz and the
62 current system time in the Runnable object's forever loop is 10.212
63 seconds when the interval method is called THEN:
64
65 currentTick:= 10212 ms
66 currentInterval:= 10200 ms
67 */
69 ElapsedTime::Precision_T currentInterval,
70 void* context);
71
72 /** Defines an interval. The application should treat this data struct as
73 an opaque structure.
74 */
76 {
77 IntervalCallbackFunc_T callbackFunc; //!< Callback function pointer
78 void* context; //!< Optional Context for the callback. The callback function is required to 'understand' the actual type of the context pointer being passed to it.
79 ElapsedTime::Precision_T duration; //!< The amount time in the Interval's period.
80 ElapsedTime::Precision_T timeMarker; //!< Internal Use ONLY: Marks the last time the interval executed
81
82 /// Constructor
84 ElapsedTime::Precision_T periodTime,
85 void* context = nullptr )
87 , context( context )
88 , duration( periodTime )
89 , timeMarker( { 0,0 } )
90 {
91 }
92
93 /// Data accessor
95
96 /// Data accessor
98
99 /// Data accessor
100 inline void* getContext() { return context; }
101 };
102
103
104 /** Defines the method that is used to report to the Application when an
105 Interval does not execute 'on time'
106
107 Where:
108 intervalThatSlipped - Reference to the interval that did not execute on time
109 currentTick - The system time when the scheduler executed the interval
110 missedInterval - The interval boundary that was missed
111 intervalContext - Application defined argument/value that was
112 passed to the interval method.
113
114 */
115 typedef void (*ReportSlippageFunc_T)(Interval_T& intervalThatSlipped,
116 ElapsedTime::Precision_T currentTick,
117 ElapsedTime::Precision_T missedInterval );
118
119 /** Defines the function that returns current system. This method has two
120 purposes:
121 1) It simplifies unit testing because it breaks the dependency
122 on 'real time
123 2) Makes unit testing easier
124 */
126
127 /** Defines the optional functions that are used to provide hooks during
128 startup/shutdown of the thread/loop[ to perform application specific
129 processing.
130 */
131 typedef void( *Hook_T )(ElapsedTime::Precision_T currentTick );
132
133public:
134 /** Constructor. The application provides a variable length array of interval
135 definitions that will be scheduled. The last entry in the
136 array must contain a 'null' Interval_T definition (i.e. all fields
137 set to zero). The Scheduler assumes that each Interval_T definition
138 has a unique period time.
139
140 The individual 'intervals' MUST be initialized (either statically or
141 by calling initializeInterval()) before creating the scheduler.
142
143 Notes:
144 - When extending a event based (i.e. inherits from EventLoop) runnable
145 object with scheduling, the Application should define all of the
146 scheduled Intervals to have period times that are multiples of
147 the EventLoop's 'timeOutPeriodInMsec' constructor value.
148 */
150 Hook_T beginThreadProcessing = nullptr,
151 Hook_T endThreadProcessing = nullptr,
152 ReportSlippageFunc_T slippageFunc = nullptr,
154
155 /// Virtual destructor
156 virtual ~PeriodicScheduler() {};
157
158
159public:
160 /** This method is used to invoke the scheduler. When called zero or more
161 Interval definitions will be executed. The method returns true if at
162 least one Interval was executed.
163
164 If a scheduled Interval does not execute 'on time', then the reportSlippage()
165 method will called. It is the Application's to decide (what if anything)
166 is done when there is slippage in the scheduling. The slippage is reported
167 AFTER the Interval's IntervalCallbackFunc_T is called.
168 */
169 virtual bool executeScheduler();
170
171
172 /** This method is expected to be called ONCE when the 'thread' is started and
173 prior to the thread entering its 'forever' loop
174 */
175 virtual void beginLoop();
176
177 /** This method is expected to be called ONCE when the 'thread' has exited
178 its 'forever' loop (but before the thread has actually terminated)
179 */
180 virtual void endLoop();
181
182protected:
183 /** Helper method to Round DOWN to the nearest 'interval' boundary.
184 A side effect the rounding-down is the FIRST execution of an interval
185 will NOT be accurate (i.e. will be something less than 'periodMs').
186 */
187 void setTimeMarker( Interval_T& interval, ElapsedTime::Precision_T currentTick ) noexcept;
188
189protected:
190 /// List of Intervals Pointers
192
193 /// Report slippage method
195
196 /// Current system callback
198
199 /// Application hook during thread start-up
201
202 /// Application hook during thread shutdown
204
205 /// Flag to managing the 'first' execution
207};
208
209}; // end namespaces
210};
211#endif // end header latch
static Precision_T precision() noexcept
This method returns the elapsed time, in seconds with milliseconds precision, since the system was po...
This concrete class is a 'policy' object that is used to add polled based, cooperative monotonic sche...
Definition PeriodicScheduler.h:48
void(* IntervalCallbackFunc_T)(ElapsedTime::Precision_T currentTick, ElapsedTime::Precision_T currentInterval, void *context)
Definition for an interval method.
Definition PeriodicScheduler.h:68
PeriodicScheduler(Interval_T intervals[], Hook_T beginThreadProcessing=nullptr, Hook_T endThreadProcessing=nullptr, ReportSlippageFunc_T slippageFunc=nullptr, NowFunc_T nowFunc=ElapsedTime::precision)
Constructor.
ReportSlippageFunc_T m_reportSlippage
Report slippage method.
Definition PeriodicScheduler.h:194
ElapsedTime::Precision_T(* NowFunc_T)()
Defines the function that returns current system.
Definition PeriodicScheduler.h:125
Hook_T m_endThreadFunc
Application hook during thread shutdown.
Definition PeriodicScheduler.h:203
void setTimeMarker(Interval_T &interval, ElapsedTime::Precision_T currentTick) noexcept
Helper method to Round DOWN to the nearest 'interval' boundary.
void(* ReportSlippageFunc_T)(Interval_T &intervalThatSlipped, ElapsedTime::Precision_T currentTick, ElapsedTime::Precision_T missedInterval)
Defines the method that is used to report to the Application when an Interval does not execute 'on ti...
Definition PeriodicScheduler.h:115
Interval_T * m_intervals
List of Intervals Pointers.
Definition PeriodicScheduler.h:191
void(* Hook_T)(ElapsedTime::Precision_T currentTick)
Defines the optional functions that are used to provide hooks during startup/shutdown of the thread/l...
Definition PeriodicScheduler.h:131
virtual void endLoop()
This method is expected to be called ONCE when the 'thread' has exited its 'forever' loop (but before...
virtual void beginLoop()
This method is expected to be called ONCE when the 'thread' is started and prior to the thread enteri...
virtual bool executeScheduler()
This method is used to invoke the scheduler.
virtual ~PeriodicScheduler()
Virtual destructor.
Definition PeriodicScheduler.h:156
Hook_T m_beginThreadFunc
Application hook during thread start-up.
Definition PeriodicScheduler.h:200
NowFunc_T m_nowFunc
Current system callback.
Definition PeriodicScheduler.h:197
bool m_firstExecution
Flag to managing the 'first' execution.
Definition PeriodicScheduler.h:206
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20
Data type for time in seconds with a 'fractional' millisecond precision.
Definition ElapsedTime.h:35
Defines an interval.
Definition PeriodicScheduler.h:76
IntervalCallbackFunc_T callbackFunc
Callback function pointer.
Definition PeriodicScheduler.h:77
IntervalCallbackFunc_T getCallbackFunction()
Data accessor.
Definition PeriodicScheduler.h:94
void * context
Optional Context for the callback.
Definition PeriodicScheduler.h:78
ElapsedTime::Precision_T getDuration()
Data accessor.
Definition PeriodicScheduler.h:97
ElapsedTime::Precision_T timeMarker
Internal Use ONLY: Marks the last time the interval executed
Definition PeriodicScheduler.h:80
ElapsedTime::Precision_T duration
The amount time in the Interval's period.
Definition PeriodicScheduler.h:79
Interval_T(IntervalCallbackFunc_T callbackFunc, ElapsedTime::Precision_T periodTime, void *context=nullptr)
Constructor.
Definition PeriodicScheduler.h:83
void * getContext()
Data accessor.
Definition PeriodicScheduler.h:100