GM6000 Digital Heater Controller Branch: main
SDX-1330
SimTick.h
Go to the documentation of this file.
1#ifndef Cpl_System_SimTick_h_
2#define Cpl_System_SimTick_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
15
16#include <stddef.h>
17#include "colony_config.h"
19#include "Cpl/Container/Item.h"
20
21
22/** Minimum number of ticks required to advance the simulated time. This
23 parameter is a work-around for a non-Real Time OS (such as Windoze) which
24 can not actually sleep/wait for 1msec. The parameter is defaulted to
25 10, i.e. after the simulation has been advance 10 tick -->the simulated
26 time will be advanced 10msec.
27
28 NOTE: The parameter also allows the simulation to be "speed-up" if the
29 timing requirements of the application under test do not require
30 millisecond resolution, e.g. setting this value to 1000 (aka 1 second)
31 the simulation runs MUCH faster than real time
32 */
33#ifndef OPTION_CPL_SYSTEM_SIM_TICK_MIN_TICKS_FOR_ADVANCE
34#define OPTION_CPL_SYSTEM_SIM_TICK_MIN_TICKS_FOR_ADVANCE 10
35#endif
36
37/** Value passed to the Cpl::System::sleep() call to effectively yield the CPU
38 do allow the threads running in simulated time to actually begin executing
39 */
40#ifndef OPTION_CPL_SYSTEM_SIM_TICK_YEILD_SLEEP_TIME
41#define OPTION_CPL_SYSTEM_SIM_TICK_YEILD_SLEEP_TIME 1
42#endif
43
44#ifdef USE_CPL_SYSTEM_SIM_TICK
45/** This macro is a wrapper for the topLevelWait() call. The macro allows the
46 call to be compiled-out when simulate time is NOT enabled.
47 */
48#define CPL_SYSTEM_SIM_TICK_TOP_LEVEL_WAIT() Cpl::System::SimTick::topLevelWait()
49
50/** This macro is a wrapper for the applicationWait() call. The macro allows the
51 call to be compiled-out when simulated time is NOT enabled
52 */
53#define CPL_SYSTEM_SIM_TICK_APPLICATION_WAIT() Cpl::System::SimTick::applicationWait()
54
55
56/** This macro is a wrapper for the usingSimTicks() call. The macro allows the call
57 to be compiled-out when simulate time is NOT enabled.
58 */
59#define CPL_SYSTEM_SIM_TICK_USING_SIM_TICKS() Cpl::System::SimTick::usingSimTicks()
60
61
62 /** This COMPONENT Scoped macro is a wrapper for threadInit_() call. The macro allows the
63 call to be compiled-out when simulate time is NOT enabled. NOTE: The
64 application SHOULD NEVER use this macro
65 */
66#define CPL_SYSTEM_SIM_TICK_THREAD_INIT_(f) Cpl::System::SimTick::threadInit_(f)
67
68
69/** This COMPONENT Scoped macro is a wrapper for onThreadExit_() call. The macro allows the
70 call to be compiled-out when simulate time is NOT enabled. NOTE: The
71 application SHOULD NEVER use this macro
72 */
73#define CPL_SYSTEM_SIM_TICK_ON_THREAD_EXIT_() Cpl::System::SimTick::onThreadExit_()
74
75
76
77#else
78/// Simulate Tick disabled
79#define CPL_SYSTEM_SIM_TICK_TOP_LEVEL_WAIT()
80
81/// Simulate Tick disabled
82#define CPL_SYSTEM_SIM_TICK_APPLICATION_WAIT()
83
84/// Simulate Tick disabled
85#define CPL_SYSTEM_SIM_TICK_USING_SIM_TICKS() false
86
87/// Simulate Tick disabled
88#define CPL_SYSTEM_SIM_TICK_THREAD_INIT_(f)
89
90/// Simulate Tick disabled
91#define CPL_SYSTEM_SIM_TICK_REAL_TIME_THREAD_INIT_()
92
93/// Simulate Tick disabled
94#define CPL_SYSTEM_SIM_TICK_ON_THREAD_EXIT_()
95
96#endif
97
98
99
100
101
102///
103namespace Cpl {
104///
105namespace System {
106
107
108/** This class define the interface to provide a simulates system tick (in
109 milliseconds) to the application. The goal is provide a mechanism for
110 an application to run in 'simulate time' instead of real time. The
111 simulated tick mechanism is for the most part transparent an application,
112 i.e. the application programmer does NOT have to be aware or design
113 his code differently to be able to run in simulate tick mode. The
114 'intrusions points' are detailed below. The typical uses cases for using
115 simulated time are:
116
117@htmlonly
118<pre>
119
120 o Testing. A simulated system tick allows the test harness to have
121 deterministic control of the application's time. In addition, time
122 can be singled stepped, paused, accelerated (i.e. run N days in
123 M minutes).
124
125 o Model in the Loop (MIL). An external application such as Matlab/
126 Simulink can be used to provide the simulated tick input as well
127 as capture the application's output and provide the application
128 inputs, i.e. connecting the application to a simulated physical
129 world. Note: 'connecting' a MIL application to a simulation such
130 as Matlib/Simulink requires a communication layer that is NOT
131 provided by Colony as well as specialization to Matlib/Simulink
132 that mates up with the afore mentioned communication layer. This
133 is left as an exercise to the reader :).
134
135
136 NOTES:
137 ------
138 o All timing sources, e.g. Cpl::Api::sleep(), provided by Colony.Core
139 have support for simulated ticks build in.
140
141 o All of the Colony.Core's ITC message queues/mailbox servers, etc. also
142 have support for simulated ticks build in.
143
144 o With respect to threads that are using simulated time, There needs to
145 be at least one thread (at any given time) that is a 'timing' source,
146 e.g. waiting on sleep(), timedWait(), etc. If your application uses
147 a Cpl::System::EventLoop (or a sub-class) instance - then this
148 requirement has been met. Worse case you a create a thread that
149 forever loops on sleep().
150
151 o This interface is independent of the underlaying platform, i.e works
152 on ALL platforms.
153
154 o NOT all threads and/or parts of an application need or should use
155 simulated time. For example the test harness (or communication
156 interface to a simulation tool) that provide the stimulus to
157 advance the simulated tick need to run in real-time. It is the
158 Application responsibility to properly selected what is and is
159 not using simulated time.
160
161 - The platform's native/main thread ALWAYS runs in 'real-time', i.e.
162 call to SimTick::wait() never blocks. Also any Api::sleep()
163 calls from the native/main thread are redirected to the
164 Api::sleepInRealTime().
165
166 o THREADING: This interface provides ONLY a functional simulation of
167 system ticks. These means the actual runtime timing of threads and/or
168 sequence of execution will NOT be the same as the 'live'
169 non-simulated application.
170
171 - When using simulated ticks with multi-threaded applications, the
172 timing of the threads using the simulated ticks behavior is 'better'
173 if the thread supplying the ticks has LOWER priority than the
174 threads using the simulate ticks.
175
176 - BEWARE of multi-core targets. Because many threads can actually
177 execute at the SAME time on multi-core boxes, the timing between
178 the tick source thread and the simulated-tick threads and between
179 simulated-tick threads is UNPREDICTABLE. The tick sequencing
180 is guaranteed, but the order within a tick OR timing relations to
181 the tick source OR inter-tick timing is NOT guaranteed.
182
183 - Not all use cases using a simulated tick and threading will work
184 and/or are supported. The simulated tick logic only works for
185 event driven and/or frequency based where all actions for a time
186 period as completed in on pass of the "main loop". Other issues:
187
188 o The assumption is that all calls to a mutex will cause the
189 thread to block across simulated ticks, i.e. mutex can ALWAYS
190 be successfully acquired and release within the context of
191 single simulated tick
192
193 - Be careful about deleting threads when using Simulated Ticks -
194 brute force terminating threads in this scenario has unpredictable
195 and/or improper behavior. Always have threads run to completion
196 when using Simulate Ticks.
197
198 o The DEFAULT implementation of this interface assumes that one
199 system tick EQUALS one millisecond.
200
201
202 HOW TO ADD SIMULATED TICK SUPPORT
203 ---------------------------------
204 o Enabling Simulated time is done at compile done. The following steps
205 are required:
206 1) Compile with the preprocessor symbol USE_CPL_SYSTEM_SIM_TICK
207 defined.
208
209 2) Compile & Link the src/Cpl/System/_simtick directory. Also do NOT
210 compile/link the src/Cpl/System/xxx/_realtime directories
211
212 o Typically thats all that needs to be done. The only additional work is if
213 application creates thread(s) that have an application specific (i.e. not
214 a Colony.Core Event Loop, mailbox server, etc.) 'forever' loop AND it
215 is desired for the thread(s) to use 'simulated time'. For this scenario
216 the application must add a call to this interface at the top of the
217 forever loop. See the example below:
218
219 void myThreadEntryFunction( void* arg )
220 {
221 ...
222
223 while(1)
224 {
225 // Block the main loop until the next simulated system tick
226 CPL_SYSTEM_SIM_TICK_TOP_LEVEL_WAIT();
227 ...
228 }
229 }
230 }
231
232</pre>
233@endhtmlonly
234 */
236{
237public:
238 /** This method will cause the current thread to block UNTIL one
239 simulated system tick has elapsed. Every call to this method
240 consumes one simulated system tick. This method SHOULD only be used
241 at the very most top level of a thread's 'forever' loop.
242
243 This method SHOULD NEVER be called directly, but via the preprocessor
244 macro CPL_SYSTEM_SIM_TICK_TOP_LEVEL_WAIT(). This allows the simulate
245 tick code to be compiled out of production versions of the application.
246 */
247 static void topLevelWait( void ) noexcept;
248
249 /** This method returns true if the current thread is using 'simulated time'
250
251 This method SHOULD NEVER called directly, but via the preprocessor
252 macro CPL_SYSTEM_SIM_TICK_USING_SIM_TICKS(). This allows the simulate tick
253 code to be compiled out of production versions of the application.
254 */
255 static bool usingSimTicks( void ) noexcept;
256
257 /** This method is used to support blocking calls (such as wait on semaphore,
258 sleep(), etc.) within an individual iteration of thread's top level
259 loop. The 'block' in this case IS NOT waiting on the simulated tick -->
260 it is waiting on the application logic.
261
262 This method SHOULD NEVER called directly, but via the preprocessor
263 macro CPL_SYSTEM_SIM_TICK_APPLICATION_WAIT(). This allows the simulate
264 tick code to be compiled out of production versions of the application.
265 */
266 static void applicationWait( void ) noexcept;
267
268 /** This method returns the current simulated tick count
269 */
270 static size_t current( void ) noexcept;
271
272
273public:
274 /** Calling this method advances the application's time by 'numTicks'. This
275 method should ONLY called by a test harness or some other module that
276 is typically not part of the application, i.e. the caller of this
277 method must be 'executing' in real time. Internally the simulated
278 time is advance at one tick at time.
279
280 The method returns true after completing advancing the internal tick
281 count by the amount specified. False is only returned when the
282 simulate tick engine has detected that there a NO simulated-tick-threads
283 executing OR that ALL simulated-tick-threads are dead locked.
284 */
285 static bool advance( size_t numTicks ) noexcept;
286
287
288 /** This method returns true if the specified thread (by Thread ID) is
289 blocked waiting on the next simulate tick. While this method is
290 thread safe -->it is intended (and only make sense) to be called
291 from the SAME thread that also calls the advance() method.
292 */
293 static bool isWaitingOnNextTick( size_t threadID ) noexcept;
294
295
296public:
297 /** This COMPONENT Scoped method is used during thread creation to insert the
298 necessary hooks (per thread) for the simulate tick engine. If
299 'useSimTicks' is false then thread does NOT use simulated time. This
300 method SHOULD NEVER be called by the application.
301 */
302 static void threadInit_( bool useSimTicks=true ) noexcept;
303
304 /** This COMPONENT Scoped method is used during thread deletion to insert the
305 necessary hooks (per thread) for the simulate tick engine. This
306 method SHOULD NEVER be called by the application.
307 */
308 static void onThreadExit_( void ) noexcept;
309
310
311protected:
312 /// The thread's current simulated time in ticks
314
315 /// Semaphore used to wait on a simulated tick
317
318 /// Flag that keeps track if I need to signal/ack-back to the tick source for the current tick
320
321 /// Thread ID of the thread using the simulate tick
323
324
325protected:
326 /// Constructor
328
329 /// Helper method. Returns true if the thread was queued for the next simulated tick
330 static bool testAndQueue( SimTick* simInfoPtr ) noexcept;
331
332 /// Helper method.
333 static unsigned wakeUpWaiters( void ) noexcept;
334
335 /// Helper method.
336 static unsigned getCurrentWaitersCount( void ) noexcept;
337
338 /// Friend(s)
339 friend class ElapsedTime;
340 friend class Api;
341};
342
343
344}; // end namespaces
345};
346#endif // end header latch
347
348
349
This class is used by the Container classes to implement a various types of singly linked containers.
Definition Item.h:33
This class defines methods for initializing the Colony.Core class library and other startup/init acti...
Definition Api.h:29
This class defines the interface for accessing the elapsed time since power up and/or reset of the pl...
Definition ElapsedTime.h:31
This semaphore class defines the interface for a Counting Semaphore.
Definition Semaphore.h:37
This class define the interface to provide a simulates system tick (in milliseconds) to the applicati...
Definition SimTick.h:236
static unsigned wakeUpWaiters(void) noexcept
Helper method.
static void applicationWait(void) noexcept
This method is used to support blocking calls (such as wait on semaphore, sleep(),...
bool m_ackPending
Flag that keeps track if I need to signal/ack-back to the tick source for the current tick.
Definition SimTick.h:319
Semaphore m_waiter
Semaphore used to wait on a simulated tick.
Definition SimTick.h:316
static void topLevelWait(void) noexcept
This method will cause the current thread to block UNTIL one simulated system tick has elapsed.
static size_t current(void) noexcept
This method returns the current simulated tick count.
static bool testAndQueue(SimTick *simInfoPtr) noexcept
Helper method. Returns true if the thread was queued for the next simulated tick.
static bool usingSimTicks(void) noexcept
This method returns true if the current thread is using 'simulated time'.
static void onThreadExit_(void) noexcept
This COMPONENT Scoped method is used during thread deletion to insert the necessary hooks (per thread...
size_t m_curTicks
The thread's current simulated time in ticks.
Definition SimTick.h:313
static bool advance(size_t numTicks) noexcept
Calling this method advances the application's time by 'numTicks'.
static unsigned getCurrentWaitersCount(void) noexcept
Helper method.
static void threadInit_(bool useSimTicks=true) noexcept
This COMPONENT Scoped method is used during thread creation to insert the necessary hooks (per thread...
size_t m_threadId
Thread ID of the thread using the simulate tick.
Definition SimTick.h:322
static bool isWaitingOnNextTick(size_t threadID) noexcept
This method returns true if the specified thread (by Thread ID) is blocked waiting on the next simula...
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20