GM6000 Digital Heater Controller Branch: main
SDX-1330
Thread.h
Go to the documentation of this file.
1#ifndef Cpl_System_Thread_h_
2#define Cpl_System_Thread_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 "colony_map.h"
16#include "Cpl/System/Runnable.h"
17#include "Cpl/System/Signable.h"
18#include "Cpl/Type/Traverser.h"
19#include <stdlib.h>
20
21
22/// Defer the definition of the native thread handle to the application's 'platform'
23#define Cpl_System_Thread_NativeHdl_T Cpl_System_Thread_NativeHdl_T_MAP
24
25
26/// The highest possible thread priority
27#define CPL_SYSTEM_THREAD_PRIORITY_HIGHEST CPL_SYSTEM_THREAD_PRIORITY_HIGHEST_MAP
28
29/// The recommended/default priority for a thread
30#define CPL_SYSTEM_THREAD_PRIORITY_NORMAL CPL_SYSTEM_THREAD_PRIORITY_NORMAL_MAP
31
32/// The lowest possible thread priority
33#define CPL_SYSTEM_THREAD_PRIORITY_LOWEST CPL_SYSTEM_THREAD_PRIORITY_LOWEST_MAP
34
35/** This value can be 'added' to one of the above Priorities to increase (i.e.
36 higher) the priority by one 'level. A priority can be increased multiple
37 levels by adding this value multiple times.
38 */
39#define CPL_SYSTEM_THREAD_PRIORITY_RAISE CPL_SYSTEM_THREAD_PRIORITY_RAISE_MAP
40
41 /** This value can be 'added' to one of the above Priorities to decrease (i.e.
42 lower) the priority by one 'level. A priority can be decreased multiple
43 levels by adding this value multiple times.
44 */
45#define CPL_SYSTEM_THREAD_PRIORITY_LOWER CPL_SYSTEM_THREAD_PRIORITY_LOWER_MAP
46
47
48
49 ///
50namespace Cpl {
51///
52namespace System {
53
54
55/** This abstract class defines the operations that can be performed
56 on a thread. NOTE: The thread contains a built in counting semaphore
57 that is usable by the developer. This semaphore can be used to assist
58 in implementing inter-thread-communications and/or ISR-to-thread
59 communications.
60 */
61class Thread : public Signable
62{
63public:
64 /// Virtual destructor.
65 virtual ~Thread() {};
66
67 /// This method returns the name (null terminated string) of the current thread.
68 virtual const char* getName() noexcept = 0;
69
70 /// This method returns the task's unique identifier
71 virtual size_t getId() noexcept = 0;
72
73 /** This method returns the 'running' state of the thread. If the method
74 returns false, the underlying thread has terminated (i.e. the run()
75 method has completed) and then the Thread object/instance can be safely
76 deleted using the destroy() method below.
77 */
78 virtual bool isRunning( void ) noexcept = 0;
79
80 /** Returns the native thread handle. Note: This handle can NOT be used
81 a handle/reference/pointer to a Thread object instance.
82 */
83 virtual Cpl_System_Thread_NativeHdl_T getNativeHandle( void ) noexcept = 0;
84
85 /// This method returns a reference to the thread's Runnable object
86 virtual Runnable& getRunnable( void ) noexcept = 0;
87
88public:
89 /// This method returns a reference to the currently executing thread.
90 static Thread& getCurrent() noexcept;
91
92 /** Special version of getCurrent(). getCurrent() will trigger a FATAL
93 error when called from a NON CPL thread. tryGetCurrent() will
94 return a nullptr when it detects that the current thread is NOT a
95 CPL thread.
96
97 This method should ONLY be used when your application has mix of CPL and
98 non-CPL threads, i.e. the default should be to use getCurrent().
99 */
100 static Thread* tryGetCurrent() noexcept;
101
102 /** This method causes the CURRENT thread to wait until its 'thread
103 semaphore' is signal/set.
104 */
105 static void wait() noexcept;
106
107 /** This method is similar to Semaphore::tryWait(), except that it operates
108 on the thread semaphore
109 */
110 static bool tryWait() noexcept;
111
112 /** Same as wait(), except the call will return after 'timeoutInMsec'
113 has expired without the thread being signaled. The method return true
114 if the Thread was signaled; else false is returned if the timeout period
115 expired.
116 */
117 static bool timedWait( unsigned long timeoutInMsec ) noexcept;
118
119 /** Returns the name for the current thread (i.e. short-hand for
120 threadPtr->getCurrent().getName()
121 */
122 static const char* myName() noexcept;
123
124 /** Returns the ID for the current thread (i.e. short-hand for
125 threadPtr->getCurrent().getId()
126 */
127 static size_t myId() noexcept;
128
129 /// This method returns a reference to the current thread' runnable instance.
130 static inline Runnable& myRunnable() noexcept { return getCurrent().getRunnable(); }
131
132
133public:
134 /** This abstract class defines the client interface for walking
135 the list of threads, i.e. defines the callback method for
136 when walking/traversing the list of active threads.
137 */
139 {
140 public:
141 /// Virtual Destructor
142 virtual ~Traverser() {}
143
144 public:
145 /** This method is called once for ever item in the "traversee's"
146 list. The return code from the method is used by the traverser
147 to continue the traversal (eCONTINUE), or abort the traversal
148 (eABORT).
149 */
150 virtual Cpl::Type::Traverser::Status_T item( Thread& nextThread ) =0;
151 };
152
153
154 /** Internal Iterator that allows the Client to traverse the list
155 active threads.
156
157 NOTES:
158 o There is difference between an "active" thread and a
159 Thread object instance. The underlying platform/OS
160 determines if the a thread is "active", i.e. the thread
161 is created and running. The Thread object gets created
162 BEFORE the thread is active and exists AFTER thread
163 terminates/end. Why the explanation, this method only
164 includes "active" threads. Threads that have terminated,
165 but that their associated Thread object instance has not
166 been delete/destroyed will NOT show up in the active list.
167 o This call is mutex protected. It will prevent new
168 threads from being created/deleted until it has
169 completed.
170 */
171 static void traverse( Thread::Traverser& client ) noexcept;
172
173
174public:
175
176 /** This method creates a thread according to the specified parameters. The
177 thread/task begins execution immediately upon its creation.
178
179 NOTE: Not all platform support all options, e.g. The C++11 multi
180 threading support does not directly support thread priorities.
181
182
183 @returns A pointer to the Thread created, OR 0 if the was an error.
184
185 @param runnable a runnable object which is started when the task is created
186 @param name a null terminated string that represent that thread's name
187 @param priority a priority ranging from highest priority (CPL_SYSTEM_THREAD_PRIORITY_HIGHEST)
188 to lowest priority (CPL_SYSTEM_THREAD_PRIORITY_LOWEST).
189 Priorities are defined in relative 'levels' to the max/normal/min
190 settings because there is no standardized priority settings
191 across Operating Systems. If the priority value exceeds the
192 upper/lower bounds the platform's priority range, the thread's
193 priority will be clamped to the upper/lower bound of the priority
194 range.
195 @param stackSize Requested size, in bytes, of the stack for the thread. A value of
196 of zero will size the thread's stack per the platform's default
197 stack size setting.
198 @param stackPtr Pointer to the stack memory to be used for the thread's stack. If
199 zero is passed, then the underlying platform/OS will allocate the
200 stack memory.
201 @param allowSimTicks With this argument is true the thread will be created to
202 use/support simulated time when 'Simulate ticks' are enabled.
203 A value false will force the thread to always be a 'real time'
204 thread. See Cpl::System::SimTicks for more details about
205 simulated time.
206 */
207 static Thread* create( Runnable& runnable,
208 const char* name,
210 int stackSize = 0,
211 void* stackPtr = 0,
212 bool allowSimTicks = true
213 );
214
215
216 /** This method is used to destroy a thread that was created by the Thread
217 Factory. If you are going to delete a thread that was created by the
218 Thread Factory, you MUST use the Factory's destroy method to delete it
219 to properly insure that the memory used to create the task is released
220 correctly.
221
222 NOTES:
223
224 o The application should only delete/destroy a thread AFTER the
225 thread's Runnable object.run() method has ended/terminated. If
226 the thread is destroy/delete before run() has completed, there is
227 NO GUARANTEE with respect to whether or not the thread has
228 released all acquired resources!
229 o NOT ALL Platforms support brute-force termination of threads,
230 e.g. in C++11 there is no method/interface to forcibly terminate
231 a thread. Once again -->it is a HIGHLY DISCOURAGED to design
232 your application where you need to forcibly terminate threads.
233 o This method only deletes/destroys the Thread instance -- it does
234 NOT delete/destroy the associated Runnable instance.
235 */
236 static void destroy( Thread& threadToDestroy );
237
238};
239
240
241}; // end namespaces
242};
243#endif // end header latch
#define Cpl_System_Thread_NativeHdl_T
Defer the definition of the native thread handle to the application's 'platform'.
Definition Thread.h:23
#define CPL_SYSTEM_THREAD_PRIORITY_NORMAL
The recommended/default priority for a thread.
Definition Thread.h:30
This is an abstract class defines the interface for an object that is "executed" when a Thread object...
Definition Runnable.h:29
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 client interface for walking the list of threads, i....
Definition Thread.h:139
virtual Cpl::Type::Traverser::Status_T item(Thread &nextThread)=0
This method is called once for ever item in the "traversee's" list.
virtual ~Traverser()
Virtual Destructor.
Definition Thread.h:142
This abstract class defines the operations that can be performed on a thread.
Definition Thread.h:62
static const char * myName() noexcept
Returns the name for the current thread (i.e.
static void destroy(Thread &threadToDestroy)
This method is used to destroy a thread that was created by the Thread Factory.
static bool timedWait(unsigned long timeoutInMsec) noexcept
Same as wait(), except the call will return after 'timeoutInMsec' has expired without the thread bein...
virtual Cpl_System_Thread_NativeHdl_T getNativeHandle(void) noexcept=0
Returns the native thread handle.
virtual size_t getId() noexcept=0
This method returns the task's unique identifier.
static Thread * create(Runnable &runnable, const char *name, int priority=CPL_SYSTEM_THREAD_PRIORITY_NORMAL, int stackSize=0, void *stackPtr=0, bool allowSimTicks=true)
This method creates a thread according to the specified parameters.
static Runnable & myRunnable() noexcept
This method returns a reference to the current thread' runnable instance.
Definition Thread.h:130
static Thread * tryGetCurrent() noexcept
Special version of getCurrent().
static bool tryWait() noexcept
This method is similar to Semaphore::tryWait(), except that it operates on the thread semaphore.
virtual Runnable & getRunnable(void) noexcept=0
This method returns a reference to the thread's Runnable object.
static void wait() noexcept
This method causes the CURRENT thread to wait until its 'thread semaphore' is signal/set.
static Thread & getCurrent() noexcept
This method returns a reference to the currently executing thread.
static size_t myId() noexcept
Returns the ID for the current thread (i.e.
virtual bool isRunning(void) noexcept=0
This method returns the 'running' state of the thread.
virtual const char * getName() noexcept=0
This method returns the name (null terminated string) of the current thread.
virtual ~Thread()
Virtual destructor.
Definition Thread.h:65
static void traverse(Thread::Traverser &client) noexcept
Internal Iterator that allows the Client to traverse the list active threads.
Status_T
Return codes (for the callback method) that determine if the traversal should continue.
Definition Traverser.h:32
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20