GM6000 Digital Heater Controller Branch: main
SDX-1330
EventLoop.h
Go to the documentation of this file.
1#ifndef Cpl_System_EventLoop_h_
2#define Cpl_System_EventLoop_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#include "Cpl/System/Runnable.h"
17#include "Cpl/System/Signable.h"
21
22
23/** Specifies the default timeout period for waiting on a event.
24 */
25#ifndef OPTION_CPL_SYSTEM_EVENT_LOOP_TIMEOUT_PERIOD
26#define OPTION_CPL_SYSTEM_EVENT_LOOP_TIMEOUT_PERIOD 1 //!< 1 msec timeout, aka 1 msec timer resolution for Software Timers
27#endif
28
29 ///
30namespace Cpl {
31///
32namespace System {
33
34/** This concrete class is a Runnable object that provides a event driven
35 execution model for a thread. The thread will remaining blocked until
36 an "event" occurs. The following "events" are supported:
37
38 1) The EventLoop Semaphore was signaled
39 2) A Event Flag was set( Cpl::System::EventFlag)
40 3) A Software Timer expiring (Cpl::System::Timer)
41
42 To receive a callbacks when Event Flags are set, the application must create
43 a child class and provide its own implementation of the processEventFlag()
44 method.
45
46 The class also provides a mechanism for registering 'other' events to
47 be processed.
48
49 Note: The EventLoop does NOT use/consume the Thread Semaphore.
50 */
51class EventLoop : public Runnable, public EventFlag, public Signable, public TimerManager
52{
53public:
54 /** Constructor. The 'timeOutPeriodInMsec' parameter specifies how
55 long the EventLoop will wait for an event before timing out and
56 processing the software timers. The value of 'timeOutPeriodInMsec' is
57 the resolution of the timing source for the Software Timers, i.e. the
58 event loop will 'wake-up' at least every 'timeOutPeriodInMsec'
59 milliseconds.
60
61 A fatal error is generated if 'timeOutPeriodInMsec' is set to zero.
62 */
63 EventLoop( unsigned long timeOutPeriodInMsec = OPTION_CPL_SYSTEM_EVENT_LOOP_TIMEOUT_PERIOD,
64 SharedEventHandlerApi* eventHandler = 0 );
65
66 /// Virtual destructor
67 virtual ~EventLoop() {};
68
69
70protected:
71 /** This method is used to initialize the Event Loop's thread has started
72 to executed.
73
74 This method is intended to be used by child classes that are extending
75 the Event Loop. For this use case - this method MUST be called once on
76 the beginning of the appRun() method and BEFORE the "main loop" for the
77 appRun() method is entered.
78 */
79 virtual void startEventLoop() noexcept;
80
81 /** This method is used to wait (and process) the next event(s). This
82 method returns after being woken up and processing all timer expired
83 and pending Event Flag events. This method should always be wrapped
84 in a loop (typically a forever loop).
85
86 If the 'skipWait' argument is set to true then the method does not
87 wait for event - it simply processing the current event flags and
88 timers. The use scenario for skipping the wait is for child classes
89 to execute the event loop multiple times to drain their message/notifications
90 queues one message/notification at a time (i.e. give equal time/priority
91 to all types of events)
92
93 The method typically returns true. The method only returns false if
94 the pleaseStop() method was called on the Event Loop instance.
95
96 This method is intended to be used by child classes that are extending
97 the Event Loop. For this use case - this method MUST be called inside
98 the child class's "main loop" in its the appRun() method.
99
100 Example appRun() for a child class extending the Event Loop:
101 @code
102
103 void ChildEventLoop::appRun( void )
104 {
105 startEventLoop();
106 <child specific intialization>
107
108 bool run = true;
109 while( run )
110 {
111 run = waitAndProcessEvents();
112 if ( run )
113 {
114 <child specific event processing>
115 }
116 }
117 stopEventLoop()
118 }
119
120 @endcode
121 */
122 virtual bool waitAndProcessEvents( bool skipWait = false ) noexcept;
123
124 /** This method is used to clean-up the Event Loop's when the thread is
125 being stopped.
126
127 This method is intended to be used by child classes that are extending
128 the Event Loop. For this use case - this method MUST be called once
129 AFTER the event-processing loop has exited.
130 */
131 virtual void stopEventLoop() noexcept;
132
133protected:
134 /** This method is used (by the concrete child class(es)) to process one
135 or more Event Flags. This method is called when at least one Event
136 Flag was set. The method is called N consecutive times - one call for
137 each Event Flag that is set. The 'eventNumber' (which is zero based)
138 identifies which Event Flag is/was set.
139
140 If no ShareEventHandler was provided when the event loop was created,
141 the default implementation of this method does NOTHING; else the
142 ShareEventHandler instance is used to process the event flag/number.
143 */
144 virtual void processEventFlag( uint8_t eventNumber ) noexcept;
145
146public:
147 /// See Cpl::System::Signable
148 int signal( void ) noexcept;
149
150 /// See Cpl::System::Signable
151 int su_signal( void ) noexcept;
152
153
154public:
155 /// See Cpl::System::Runnable
157
158
159protected:
160 /// See Cpl::System::Runnable
161 void appRun();
162
163public:
164 /// See Cpl::System::EventFlag
165 void notifyEvents( Cpl_System_EventFlag_T events ) noexcept;
166
167 /// See Cpl::System::EventFlag
168 void notify( uint8_t eventNumber ) noexcept;
169
170 /// See Cpl::System::EventFlag
172
173 /// See Cpl::System::EventFlag
174 void su_notify( uint8_t eventNumber ) noexcept;
175
176public:
177 /// See Cpl::System::Runnable
178 void setThreadOfExecution_( Thread* myThreadPtr );
179
180protected:
181 /// A pointer to the thread the Event Loop executes in
183
184 /// My shared event handler (if I have one)
186
187 /// Semaphore associated with the mailbox (note: the Thread semaphore is NOT used)
189
190 /// Timeout period for waiting on the next event
191 unsigned long m_timeout;
192
193 /// Timestamp, in milliseconds, of start of event/wait loop
194 unsigned long m_timeStartOfLoop;
195
196 /// The variable holds the current state of all Event Flags
198
199 /// Flag used to help with the pleaseStop() request
200 bool m_run;
201
202};
203
204}; // end namespaces
205};
206#endif // end header latch
#define Cpl_System_EventFlag_T
Default the Event Flags data type to 32 Events flags.
Definition EventFlag.h:20
#define OPTION_CPL_SYSTEM_EVENT_LOOP_TIMEOUT_PERIOD
Specifies the default timeout period for waiting on a event.
Definition EventLoop.h:26
This abstract class defines the interface to generated a 'Event Flag'.
Definition EventFlag.h:44
This concrete class is a Runnable object that provides a event driven execution model for a thread.
Definition EventLoop.h:52
unsigned long m_timeout
Timeout period for waiting on the next event.
Definition EventLoop.h:191
bool m_run
Flag used to help with the pleaseStop() request.
Definition EventLoop.h:200
EventLoop(unsigned long timeOutPeriodInMsec=OPTION_CPL_SYSTEM_EVENT_LOOP_TIMEOUT_PERIOD, SharedEventHandlerApi *eventHandler=0)
Constructor.
Cpl::System::Semaphore m_sema
Semaphore associated with the mailbox (note: the Thread semaphore is NOT used)
Definition EventLoop.h:188
int su_signal(void) noexcept
See Cpl::System::Signable.
void su_notifyEvents(Cpl_System_EventFlag_T events) noexcept
See Cpl::System::EventFlag.
unsigned long m_timeStartOfLoop
Timestamp, in milliseconds, of start of event/wait loop.
Definition EventLoop.h:194
void notify(uint8_t eventNumber) noexcept
See Cpl::System::EventFlag.
void notifyEvents(Cpl_System_EventFlag_T events) noexcept
See Cpl::System::EventFlag.
void setThreadOfExecution_(Thread *myThreadPtr)
See Cpl::System::Runnable.
virtual void startEventLoop() noexcept
This method is used to initialize the Event Loop's thread has started to executed.
Cpl_System_EventFlag_T m_events
The variable holds the current state of all Event Flags.
Definition EventLoop.h:197
void pleaseStop()
See Cpl::System::Runnable.
virtual bool waitAndProcessEvents(bool skipWait=false) noexcept
This method is used to wait (and process) the next event(s).
Thread * m_myThreadPtr
A pointer to the thread the Event Loop executes in.
Definition EventLoop.h:182
int signal(void) noexcept
See Cpl::System::Signable.
virtual ~EventLoop()
Virtual destructor.
Definition EventLoop.h:67
virtual void stopEventLoop() noexcept
This method is used to clean-up the Event Loop's when the thread is being stopped.
void appRun()
See Cpl::System::Runnable.
virtual void processEventFlag(uint8_t eventNumber) noexcept
This method is used (by the concrete child class(es)) to process one or more Event Flags.
void su_notify(uint8_t eventNumber) noexcept
See Cpl::System::EventFlag.
SharedEventHandlerApi * m_eventHandler
My shared event handler (if I have one)
Definition EventLoop.h:185
This is an abstract class defines the interface for an object that is "executed" when a Thread object...
Definition Runnable.h:29
This semaphore class defines the interface for a Counting Semaphore.
Definition Semaphore.h:37
This abstract class defines the interface for a Shared Event Handler.
Definition SharedEventHandler.h:30
This abstract class defines the interface by which a client can cause an object that is waiting-on-a-...
Definition Signable.h:28
This abstract class defines the operations that can be performed on a thread.
Definition Thread.h:62
This mostly concrete class implements manages a list of Software Timers.
Definition TimerManager.h:36
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20