GM6000 Digital Heater Controller Branch: main
SDX-1330
List of all members | Static Public Member Functions | Protected Member Functions | Static Protected Member Functions | Protected Attributes | Friends
Cpl::System::SimTick Class Reference

This class define the interface to provide a simulates system tick (in milliseconds) to the application. More...

Detailed Description

This class define the interface to provide a simulates system tick (in milliseconds) to the application.

The goal is provide a mechanism for an application to run in 'simulate time' instead of real time. The simulated tick mechanism is for the most part transparent an application, i.e. the application programmer does NOT have to be aware or design his code differently to be able to run in simulate tick mode. The 'intrusions points' are detailed below. The typical uses cases for using simulated time are:


        o Testing.  A simulated system tick allows the test harness to have
          deterministic control of the application's time.  In addition, time
          can be singled stepped, paused, accelerated (i.e. run N days in
          M minutes).

        o Model in the Loop (MIL).  An external application such as Matlab/
          Simulink can be used to provide the simulated tick input as well
          as capture the application's output and provide the application
          inputs, i.e. connecting the application to a simulated physical
          world.  Note: 'connecting' a MIL application to a simulation such
          as Matlib/Simulink requires a communication layer that is NOT
          provided by Colony as well as specialization to Matlib/Simulink
          that mates up with the afore mentioned communication layer.  This
          is left as an exercise to the reader :).


    NOTES:
    ------
        o All timing sources, e.g. Cpl::Api::sleep(), provided by Colony.Core
          have support for simulated ticks build in.

        o All of the Colony.Core's ITC message queues/mailbox servers, etc. also
          have support for simulated ticks build in.

        o With respect to threads that are using simulated time, There needs to
          be at least one thread (at any given time) that is a 'timing' source,
          e.g. waiting on sleep(), timedWait(), etc.  If your application uses
          a Cpl::System::EventLoop (or a sub-class) instance - then this 
          requirement has been met. Worse case you a create a thread that 
          forever loops on sleep().

        o This interface is independent of the underlaying platform, i.e works
          on ALL platforms.

        o NOT all threads and/or parts of an application need or should use
          simulated time.  For example the test harness (or communication
          interface to a simulation tool) that provide the stimulus to
          advance the simulated tick need to run in real-time.  It is the
          Application responsibility to properly selected what is and is
          not using simulated time.

            - The platform's native/main thread ALWAYS runs in 'real-time', i.e.
              call to SimTick::wait() never blocks.  Also any Api::sleep()
              calls from the native/main thread are redirected to the
              Api::sleepInRealTime().

        o THREADING: This interface provides ONLY a functional simulation of
          system ticks. These means the actual runtime timing of threads and/or
          sequence of execution will NOT be the same as the 'live'
          non-simulated application.

            - When using simulated ticks with multi-threaded applications, the
              timing of the threads using the simulated ticks behavior is 'better'
              if the thread supplying the ticks has LOWER priority than the
              threads using the simulate ticks.

            - BEWARE of multi-core targets.  Because many threads can actually
              execute at the SAME time on multi-core boxes, the timing between
              the tick source thread and the simulated-tick threads and between
              simulated-tick threads is UNPREDICTABLE.  The tick sequencing
              is guaranteed, but the order within a tick OR timing relations to
              the tick source OR inter-tick timing is NOT guaranteed.

            - Not all use cases using a simulated tick and threading will work
              and/or are supported.  The simulated tick logic only works for
              event driven and/or frequency based where all actions for a time
              period as completed in on pass of the "main loop".  Other issues:
                
                o The assumption is that all calls to a mutex will cause the 
                  thread to block across simulated ticks, i.e. mutex can ALWAYS 
                  be successfully acquired and release within the context of 
                  single simulated tick

            - Be careful about deleting threads when using Simulated Ticks -
              brute force terminating threads in this scenario has unpredictable
              and/or improper behavior. Always have threads run to completion
              when using Simulate Ticks.

        o The DEFAULT implementation of this interface assumes that one
          system tick EQUALS one millisecond.


    HOW TO ADD SIMULATED TICK SUPPORT
    ---------------------------------
    o Enabling Simulated time is done at compile done.  The following steps
      are required:
        1) Compile with the preprocessor symbol USE_CPL_SYSTEM_SIM_TICK
           defined.

        2) Compile & Link the src/Cpl/System/_simtick directory.  Also do NOT
           compile/link the src/Cpl/System/xxx/_realtime directories

    o Typically thats all that needs to be done. The only additional work is if
      application creates thread(s) that have an application specific (i.e. not
      a Colony.Core Event Loop, mailbox server, etc.) 'forever' loop AND it
      is desired for the thread(s) to use 'simulated time'.  For this scenario
      the application must add a call to this interface at the top of the
      forever loop.  See the example below:

        void myThreadEntryFunction( void* arg )
            {
            ...

            while(1)
                {
                // Block the main loop until the next simulated system tick
                CPL_SYSTEM_SIM_TICK_TOP_LEVEL_WAIT();
                ...
                }
            }
         }

#include <SimTick.h>

Inheritance diagram for Cpl::System::SimTick:
[legend]
Collaboration diagram for Cpl::System::SimTick:
[legend]

Static Public Member Functions

static void topLevelWait (void) noexcept
 This method will cause the current thread to block UNTIL one simulated system tick has elapsed.
 
static bool usingSimTicks (void) noexcept
 This method returns true if the current thread is using 'simulated time'.
 
static void applicationWait (void) noexcept
 This method is used to support blocking calls (such as wait on semaphore, sleep(), etc.) within an individual iteration of thread's top level loop.
 
static size_t current (void) noexcept
 This method returns the current simulated tick count.
 
static bool advance (size_t numTicks) noexcept
 Calling this method advances the application's time by 'numTicks'.
 
static bool isWaitingOnNextTick (size_t threadID) noexcept
 This method returns true if the specified thread (by Thread ID) is blocked waiting on the next simulate tick.
 
static void threadInit_ (bool useSimTicks=true) noexcept
 This COMPONENT Scoped method is used during thread creation to insert the necessary hooks (per thread) for the simulate tick engine.
 
static void onThreadExit_ (void) noexcept
 This COMPONENT Scoped method is used during thread deletion to insert the necessary hooks (per thread) for the simulate tick engine.
 
- Static Public Member Functions inherited from Cpl::Container::Item
static void remove_ (Item *itemPtr) noexcept
 Helper method to do the proper 'clean-up' for the multiple-containers-error-trap when removing an item from a container.
 

Protected Member Functions

 SimTick ()
 Constructor.
 
- Protected Member Functions inherited from Cpl::Container::Item
 Item ()
 Constructor.
 
 Item (const char *)
 Constructor used ONLY with the child class MapItem: -->special constructor to allow a Map to be statically allocated.
 

Static Protected Member Functions

static bool testAndQueue (SimTick *simInfoPtr) noexcept
 Helper method. Returns true if the thread was queued for the next simulated tick.
 
static unsigned wakeUpWaiters (void) noexcept
 Helper method.
 
static unsigned getCurrentWaitersCount (void) noexcept
 Helper method.
 

Protected Attributes

size_t m_curTicks
 The thread's current simulated time in ticks.
 
Semaphore m_waiter
 Semaphore used to wait on a simulated tick.
 
bool m_ackPending
 Flag that keeps track if I need to signal/ack-back to the tick source for the current tick.
 
size_t m_threadId
 Thread ID of the thread using the simulate tick.
 

Friends

class ElapsedTime
 Friend(s)
 
class Api
 

Additional Inherited Members

- Public Member Functions inherited from Cpl::Container::Item
bool insert_ (void *newContainerPtr)
 Helper method to trap when inserting an item in multiple containers.
 
bool isInContainer_ (const void *containerPtr) const noexcept
 Returns 'true' if the instance is in the specified container.
 
- Public Attributes inherited from Cpl::Container::Item
voidm_nextPtr_
 The link field.
 
voidm_inListPtr_
 Debug field.
 

Constructor & Destructor Documentation

◆ SimTick()

Cpl::System::SimTick::SimTick ( )
protected

Constructor.

Member Function Documentation

◆ advance()

static bool Cpl::System::SimTick::advance ( size_t  numTicks)
staticnoexcept

Calling this method advances the application's time by 'numTicks'.

This method should ONLY called by a test harness or some other module that is typically not part of the application, i.e. the caller of this method must be 'executing' in real time. Internally the simulated time is advance at one tick at time.

The method returns true after completing advancing the internal tick count by the amount specified. False is only returned when the simulate tick engine has detected that there a NO simulated-tick-threads executing OR that ALL simulated-tick-threads are dead locked.

◆ applicationWait()

static void Cpl::System::SimTick::applicationWait ( void  )
staticnoexcept

This method is used to support blocking calls (such as wait on semaphore, sleep(), etc.) within an individual iteration of thread's top level loop.

The 'block' in this case IS NOT waiting on the simulated tick --> it is waiting on the application logic.

This method SHOULD NEVER called directly, but via the preprocessor macro CPL_SYSTEM_SIM_TICK_APPLICATION_WAIT(). This allows the simulate tick code to be compiled out of production versions of the application.

◆ current()

static size_t Cpl::System::SimTick::current ( void  )
staticnoexcept

This method returns the current simulated tick count.

◆ getCurrentWaitersCount()

static unsigned Cpl::System::SimTick::getCurrentWaitersCount ( void  )
staticprotectednoexcept

Helper method.

◆ isWaitingOnNextTick()

static bool Cpl::System::SimTick::isWaitingOnNextTick ( size_t  threadID)
staticnoexcept

This method returns true if the specified thread (by Thread ID) is blocked waiting on the next simulate tick.

While this method is thread safe -->it is intended (and only make sense) to be called from the SAME thread that also calls the advance() method.

◆ onThreadExit_()

static void Cpl::System::SimTick::onThreadExit_ ( void  )
staticnoexcept

This COMPONENT Scoped method is used during thread deletion to insert the necessary hooks (per thread) for the simulate tick engine.

This method SHOULD NEVER be called by the application.

◆ testAndQueue()

static bool Cpl::System::SimTick::testAndQueue ( SimTick simInfoPtr)
staticprotectednoexcept

Helper method. Returns true if the thread was queued for the next simulated tick.

◆ threadInit_()

static void Cpl::System::SimTick::threadInit_ ( bool  useSimTicks = true)
staticnoexcept

This COMPONENT Scoped method is used during thread creation to insert the necessary hooks (per thread) for the simulate tick engine.

If 'useSimTicks' is false then thread does NOT use simulated time. This method SHOULD NEVER be called by the application.

◆ topLevelWait()

static void Cpl::System::SimTick::topLevelWait ( void  )
staticnoexcept

This method will cause the current thread to block UNTIL one simulated system tick has elapsed.

Every call to this method consumes one simulated system tick. This method SHOULD only be used at the very most top level of a thread's 'forever' loop.

This method SHOULD NEVER be called directly, but via the preprocessor macro CPL_SYSTEM_SIM_TICK_TOP_LEVEL_WAIT(). This allows the simulate tick code to be compiled out of production versions of the application.

◆ usingSimTicks()

static bool Cpl::System::SimTick::usingSimTicks ( void  )
staticnoexcept

This method returns true if the current thread is using 'simulated time'.

This method SHOULD NEVER called directly, but via the preprocessor macro CPL_SYSTEM_SIM_TICK_USING_SIM_TICKS(). This allows the simulate tick code to be compiled out of production versions of the application.

◆ wakeUpWaiters()

static unsigned Cpl::System::SimTick::wakeUpWaiters ( void  )
staticprotectednoexcept

Helper method.

Friends And Related Symbol Documentation

◆ ElapsedTime

friend class ElapsedTime
friend

Friend(s)

Member Data Documentation

◆ m_ackPending

bool Cpl::System::SimTick::m_ackPending
protected

Flag that keeps track if I need to signal/ack-back to the tick source for the current tick.

◆ m_curTicks

size_t Cpl::System::SimTick::m_curTicks
protected

The thread's current simulated time in ticks.

◆ m_threadId

size_t Cpl::System::SimTick::m_threadId
protected

Thread ID of the thread using the simulate tick.

◆ m_waiter

Semaphore Cpl::System::SimTick::m_waiter
protected

Semaphore used to wait on a simulated tick.


The documentation for this class was generated from the following file: