mirror of
https://github.com/QuantumLeaps/qpc.git
synced 2025-01-14 06:43:19 +08:00
280 lines
17 KiB
Plaintext
280 lines
17 KiB
Plaintext
/*! @defgroup qep QEP
|
||
|
||
@brief
|
||
Hierarchical Event Processor
|
||
|
||
QEP is a universal, UML-compliant event processor that enables developers to code UML state machines in highly readable ANSI-C, in which every state machine element is mapped to code precisely, unambiguously, and exactly once (traceability). QEP fully supports hierarchical state nesting, which is the fundamental mechanism for reusing behavior across many states instead of repeating the same actions and transitions over and over again.
|
||
*/
|
||
|
||
/*##########################################################################*/
|
||
/*! @defgroup qf QF
|
||
|
||
@brief
|
||
Active Object (Actor) Framework
|
||
|
||
QF is a portable, event-driven, real-time framework for execution of active objects (concurrent state machines) specifically designed for real-time embedded (RTE) systems.
|
||
*/
|
||
|
||
/*##########################################################################*/
|
||
/*! @defgroup qs QS
|
||
|
||
@brief
|
||
Software Tracing Instrumentation
|
||
|
||
QS is software tracing system that enables developers to monitor live QP applications with minimal target system resources and without stopping or significantly slowing down the code. QS is an ideal tool for testing, troubleshooting, and optimizing QP applications. QS can even be used to support acceptance testing in product manufacturing. Please see <a href="https://state-machine.com/qtools/qs.html"><b>QS Manual</b></a> inside the <a href="https://state-machine.com/qtools" target="_blank" class="extern">QTools collection</a> for more information.
|
||
*/
|
||
|
||
/*##########################################################################*/
|
||
/*! @defgroup qv QV
|
||
|
||
@brief
|
||
Cooperative Kernel
|
||
|
||
@description
|
||
QV is a simple **cooperative** kernel (previously called "Vanilla" kernel). This kernel executes active objects one at a time, with priority-based scheduling performed before processing of each event. Due to naturally short duration of event processing in state machines, the simple QV kernel is often adequate for many real-time systems.
|
||
|
||
|
||
------------------------------------------------------------------------------
|
||
@section qv_overview QV Overview
|
||
The QV scheduler is engaged after every RTC step of any @termref{active object} to choose the next active object to execute. The QV scheduler always chooses the highest-priority active object that has any events in its event queue. The QV scheduler then extracts the next event from this queue and dispatches it to the state machine associated with the active object. The state machine runs to completion, after which the QV scheduler runs and the cycle repeats.
|
||
|
||
Please note that because the state machines always return to the QV scheduler after each RTC step, a single stack can be used to process all state machines (memory-friendly architecture).
|
||
|
||
The QV scheduler can also very easily detect when all event queues are empty, at which point it can call the idle callback to let the application put the CPU and peripherals to a low-power sleep mode (power-friendly architecture).
|
||
|
||
Given the simplicity, portability, and low-resource consumption, the QV scheduler is very attractive. It allows you to partition the problem into active objects and execute these active objects orderly. The task-level response of this scheduler is the longest RTC step in the whole system, but because event-driven active objects don’t block, the RTC steps tend to be very short (typically just a few microseconds). Also, often you can break up longer RTC steps into shorter pieces, by posting an event to self and returning (“Reminder” state pattern). The self-posted event then triggers the continuation of longer processing.
|
||
|
||
|
||
*/
|
||
|
||
/*##########################################################################*/
|
||
/*! @defgroup qk QK
|
||
|
||
@brief
|
||
Preemptive Run-to-Completion Kernel
|
||
|
||
QK is a tiny **preemptive**, priority-based, non-blocking kernel designed specifically for executing active objects. QK runs active objects in the same way as prioritized interrupt controller (such as NVIC in ARM Cortex-M) runs interrupts using the single stack. Active objects process their events in run-to-completion (RTC) fashion and remove themselves from the call stack, the same way as nested interrupts remove themselves from the stack upon completion. At the same time high-priority active objects can preempt lower-priority active objects, just like interrupts can preempt each other under a prioritized interrupt controller. QK meets all the requirement of the Rate Monotonic Scheduling (a.k.a. Rate Monotonic Analysis RMA) and can be used in hard real-time systems.
|
||
|
||
|
||
------------------------------------------------------------------------------
|
||
@section qk_overview QK Overview
|
||
|
||
Sometimes it is not practical to break up long RTC steps, and consequently the task-level response of the simple @ref "QV kernel" might be too slow. In this cases you need to use a *preemptive* kernel. The big advantage of preemptive kernel is that it effectively decouples high-priority task from low-priority tasks in the time domain. The timeliness of execution of high-priority task is almost independent on the low-priority tasks. But of course there is no such thing as a free lunch. Preemptive kernels open the whole new class of problems related to race conditions. So you need to be very careful about sharing any resources.
|
||
|
||
*/
|
||
|
||
/*##########################################################################*/
|
||
/*! @defgroup qxk QXK
|
||
|
||
@brief
|
||
Preemptive Dual-Mode (Run-to-Completion/Blocking) RTOS Kernel
|
||
|
||
QXK is a small, preemptive, priority-based, dual-mode **blocking** kernel that executes active objects like the @ref qk "QK kernel", but can also execute traditional __blocking__ threads (extended threads). In this respect, QXK behaves exactly as a conventional __RTOS__ (Real-Time Operating System). QXK has been designed specifically for mixing event-driven active objects with traditional blocking code, such as commercial middleware (TCP/IP stacks, UDP stacks, embedded file systems, etc.) or legacy software.
|
||
|
||
Currently, the QXK kernel has been ported to the following CPUs:
|
||
|
||
- @ref arm-cm_qxk "Cortex-M (M0/M0+/M1/M3/M4/M4F/M7)"@n
|
||
Supported toolchains include: ARM-KEIL MDK, IAR-ARM, GNU-ARM, TI-ARM.
|
||
|
||
Currently, the QXK kernel is illustrated by the following examples:
|
||
|
||
- @ref arm-cm_dpp_efm32-slstk3401a (ARM Cortex-M4F)@n
|
||
Example illustrates: 6 active objects plus two extended threads, QXK blocking delay, QXK semaphore, QXK mutex, QXK blocking message queue.
|
||
|
||
- @ref arm-cm_dpp_ek-tm4c123gxl (ARM Cortex-M4F)@n
|
||
Example illustrates: 6 active objects plus two extended threads, QXK blocking delay, QXK semaphore, QXK mutex, QXK blocking message queue.
|
||
|
||
- @ref arm-cm_dpp_nucleo-l053r8 (ARM Cortex-M0+)@n
|
||
Example illustrates: 6 active objects plus two extended threads, QXK blocking delay, QXK semaphore, QXK mutex, QXK blocking message queue.
|
||
|
||
|
||
------------------------------------------------------------------------------
|
||
@section qxk_overview QXK Overview
|
||
|
||
QXK distinguishes two kinds of threads: **basic**-threads (non-blocking, run-to-completion activations) and **extended**-threads (blocking, typically structrued as endless loops). The basic-threads all nest on the same stack (Main Stack Pointer in ARM Cortex-M). The extended-threads use private per-thread stacks, as in conventional RTOS kernels. Any switching from basic- to extended-thread or extended- to extended-thread requires full context switch. On the other hand, switching from basic-thread to another basic-thread requires only activation of the basic-thread, which is much simpler and faster.
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_classes Classes in QXK
|
||
The figure below shows the main classes introduced in the QXK kernel and their relation to the classes of the QP framework.
|
||
|
||
@image html qxk_classes.gif "Classes of the QXK dual-mode kernel"
|
||
|
||
<ul class="tag">
|
||
<li><span class="tag">0</span> The abstract ::QActive class represents active objects in QP. This class contains the @c thread object of the underlying kernel (QXK thread-control-block in this case) as well as the event queue and the unique priority of the active object.
|
||
</li>
|
||
|
||
<li><span class="tag">1</span> The ::QXThread class represents the extended (blocking) threads of the QXK kernel. It inherits ::QActive, so that extended-threads can be treated as active objects internally in the framework. However, the extended-threads do not implement state machines. Instead, the data fields used for storing the current state in active objects are re-used to store the private stack of the extended-thread. The ::QXThread class also contains the @c timeEvt object (see ::QTimeEvt) for generating timeouts when the extended-thread is blocked.
|
||
</li>
|
||
|
||
<li><span class="tag">2</span> The ::QXMutex class represents the @ref qxk_mutex "priority-ceiling mutex" of the QXK kernel. The mutex can be used by both the extended-threads and active object threads (in case they share resources that need to be protected). However, using any blocking mechanism inside active objects is not recommended, because it delays run-to-completion event processing.
|
||
</li>
|
||
|
||
<li><span class="tag">3</span> The ::QXSemaphore class represents the @ref qxk_sema "counting semaphore" of the QXK kernel. The semaphore can be waited on only in the extended-threads and the QXK kernel would assert if an active object thread would attempt to wait on a semaphore. On the other hand, a semaphore can be signaled from anywhere in the code, including active objects and ISRs.
|
||
</li>
|
||
</ul>
|
||
|
||
@note
|
||
The main takeaway from the QXK class diagram is QXK's **optimal, tight integration** with the QP framework. The QXK kernel reuses all mechanisms already provided in QP, thus avoiding any code duplication, inefficient layers of indirection, and additional licensing costs, which are inevitable when using @ref ports_rtos "3rd-party RTOS kernels" to run QP applications.
|
||
|
||
|
||
------------------------------------------------------------------------------
|
||
@section qxk_features QXK Features
|
||
As you can see in the list below, <span class="highlight">QXK provides most features you might expect of a traditional blocking **RTOS** kernel and is <strong>recommended</strong> as the preferred RTOS kernel for QP applications</span> that need to mix active objects with traditional blocking code.
|
||
|
||
<ul class="tag">
|
||
|
||
<li><span class="bullet">></span>Preemptive, priority-based scheduling of up to 64 threads. Each thread must be assigned its own unique priority (1 .. #QF_MAX_ACTIVE);
|
||
</li>
|
||
|
||
> NOTE: QXK always executes the highest-priority thread that is ready to run (is not blocked). The scheduling algorithm used in QXK meets all the requirement of the Rate Monotonic Scheduling (a.k.a. Rate Monotonic Analysis — RMA) and can be used in hard real-time systems.
|
||
|
||
<li><span class="bullet">></span>All threads in QXK are capable of **blocking**. However, QXK distinguishes between two types of threads:
|
||
</li>
|
||
|
||
- **basic threads** of active objects that are made ready-to-run by events posted to the active objects. Such basic threads are non-blocking, run-to-completion activations that cannot block in the middle of the RTC step. QXK asserts when a basic thread attempts to use a blocking mechanism, such as a time-delay or a semaphore-wait. All basic threads share the common stack.
|
||
|
||
- **extended threads** that can block anywhere in their thread-handler function, whereas QXK provides a typical assortments of blocking primitives for extended-threads, such as time-delay, blocking message queue, counting semaphore, or a mutex. Each extended thread must be provided with its own private stack.
|
||
|
||
<li><span class="bullet">></span>Tightly integrated mechanisms for communication between event-driven active objects and extended blocking threads:
|
||
</li>
|
||
|
||
- Basic threads (Active Objects) can signal semaphores and send messages to extended threads.
|
||
|
||
- Extended threads can post or publish events to active objects;
|
||
|
||
<li><span class="bullet">></span>Priority-Ceiling Mutexes with optional timeout;
|
||
</li>
|
||
|
||
> NOTE: Priority-ceiling protocol implemented in QXK is immune to priority-inversions.
|
||
|
||
<li><span class="bullet">></span>Counting Semaphores with optional timeout that can block multiple extended-threads;
|
||
</li>
|
||
|
||
<li><span class="bullet">></span>Blocking "zero-copy" message queue with optional timeout bound to each extended-thread;
|
||
</li>
|
||
|
||
<li><span class="bullet">></span>Deterministic fixed-size memory pools for dynamic memory management available both to extended-threads and active objects;
|
||
</li>
|
||
|
||
<li><span class="bullet">></span>Interrupt management, including "zero-latency", kernel-unaware interrupts that are never disabled;
|
||
</li>
|
||
|
||
> NOTE: This feature is only supported on CPUs that allow selective interrupt disabling, such as ARM Cortex-M3/M4 (but not ARM Cortex-M0);
|
||
|
||
<li><span class="bullet">></span> @ref qxk_tls "Thread-Local Storage" for all threads (basic threads and extended threads).
|
||
</li>
|
||
|
||
</ul>
|
||
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_kernel Kernel Initialization and Control
|
||
|
||
@sa
|
||
- QXK_init()
|
||
- QF_run()
|
||
- QXK_onIdle()
|
||
- QXK_getVersion()
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_thread Thread Management
|
||
|
||
@sa
|
||
- ::QXThread structure (Thread Control Block)
|
||
- QXThread_ctor()
|
||
- QXTHREAD_START()
|
||
- QXThread_delay()
|
||
- QXThread_delayCancel()
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_isr Interrupt Management
|
||
|
||
@sa
|
||
- QXK_ISR_ENTRY()
|
||
- QXK_ISR_EXIT()
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_mutex Mutexes
|
||
|
||
@sa
|
||
- ::QXMutex structure (Mutex Control Block)
|
||
- QXMutex_init()
|
||
- QXMutex_lock()
|
||
- QXMutex_unlock()
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_queue Message Queues
|
||
|
||
@sa
|
||
- QXTHREAD_POST_X() - posting messages to blocking threads
|
||
- QACTIVE_POST_X() - posting events to Active Objects
|
||
- QXThread_queueGet() - waiting (blocking) on message queue
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_sema Semaphores
|
||
|
||
@sa
|
||
- ::QXSemaphore structure (Semaphore Control Block)
|
||
- QXSemaphore_init()
|
||
- QXSemaphore_wait()
|
||
- QXSemaphore_signal()
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_mem Memory Pools
|
||
|
||
@sa
|
||
- ::QMPool structure
|
||
- QMPool_init()
|
||
- QMPool_get()
|
||
- QMPool_put()
|
||
|
||
<div class="separate"></div>
|
||
@subsection qxk_tls Thread Local Storage
|
||
<b>Thread-local storage (TLS)</b> is a programming method that uses static or global memory local to a thread. TLS is specifically useful for writing library-type code, which is used in a multithreaded environment and needs to access per-thread data in an independent way.
|
||
|
||
TLS is used in some places where ordinary, single-threaded programs would use static or global variables, but where this would be inappropriate in multithreaded cases. An example of such situations is where library-type functions use a global variable to set an error condition (for example the global variable errno used by many functions of the C library). If errno were simply a global variable, a call of a system function on one thread may overwrite the value previously set by a call of a system function on a different thread, possibly before following code on that different thread could check for the error condition. The solution is to have errno be a variable that looks like it is global, but in fact exists once per thread—i.e., it lives in *thread-local storage*. A second use case would be multiple threads accumulating information into a global variable. To avoid a race condition, every access to this global variable would have to be protected by a mutual-exclusion mechanism. Alternatively, each thread might accumulate into a thread-local variable (that, by definition, cannot be read from or written to from other threads, implying that there can be no race conditions). Threads then only have to synchronize a final accumulation from their own thread-local variable into a single, truly global variable.
|
||
|
||
The TLS implementations vary, but many systems, including QXK, implement TLS by providing a pointer-sized variable thread-local. This pointer can be set to arbitrarily sized memory blocks in a thread-local manner, by allocating such a memory block (statically or dynamically) and storing the memory address of that block in the thread-local variable.
|
||
|
||
Typical usage of TLS in QXK is illustrated in the example <span class="img folder">qpc/examples/arm-cm/dpp_efm32-slstk3401a/qxk/</span>, <span class="img file_c">test.c</span>, and consists:
|
||
|
||
- define the TLS structure
|
||
@code{c}
|
||
typedef struct {
|
||
uint32_t foo;
|
||
uint8_t bar[10];
|
||
} TLS_test;
|
||
@endcode
|
||
|
||
- allocate the TLS storage for all participating threads (extended or basic threads)
|
||
@code{c}
|
||
static TLS_test l_tls1;
|
||
static TLS_test l_tls2;
|
||
@endcode
|
||
|
||
- initialize the TLS per-thread pointer in each thread routine (for extended threads) or the top-most initial transition (for basic threads of active objects):
|
||
@code{c}
|
||
static void Thread1_run(QXThread * const me) {
|
||
me->super.thread = &l_tls1; /* initialize the TLS for Thread1 */
|
||
. . .
|
||
}
|
||
. . .
|
||
static void Thread2_run(QXThread * const me) {
|
||
me->super.thread = &l_tls2; /* initialize the TLS for Thread2 */
|
||
. . .
|
||
}
|
||
@endcode
|
||
|
||
- access the TLS from your code:
|
||
@code{c}
|
||
void lib_fun(uint32_t x) {
|
||
QXK_TLS(TLS_test *)->foo = x;
|
||
}
|
||
@endcode
|
||
|
||
@sa
|
||
- QXK_current()
|
||
- QXK_TLS()
|
||
|
||
*/
|