Advantages of Running RTOS on MCU Compared to Bare Metal

Follow+Star Public Account, don’t miss the wonderful content
Advantages of Running RTOS on MCU Compared to Bare Metal
Compiled by | strongerHuang
WeChat Official Account | Embedded Column
Readers often ask questions about RTOS, such as:Should I learn RTOS now?What are the benefits of learning RTOS?Should my project run on RTOS?
The root of these questions is actually a lack of understanding of RTOS and insufficient project development experience.For those friends, today I will share several related points:
In embedded systems, there are many ways to implement task scheduling.In small systems with limited functionality, an infinite loop is sufficient to achieve the functions of the small system. When the software design becomes large and complex, developers should consider using a Real-Time Operating System (RTOS).
Advantages of Running RTOS on MCU Compared to Bare Metal
Here are several advantages of RTOS compared to bare metal:
1. Hard Real-Time Response
RTOS based on priority preemption executes priority scheduling according to the real-time requirements of tasks.Tasks with strict timing constraints can be executed preferentially, improving the application’s response to time-critical events.
2. Maximizing System Performance
For large and complex embedded applications, using an event-driven RTOS instead of a polling-based super loop structure can create a more efficient design with smaller storage requirements, allowing applications to gain more processor time.
3. Reduced Complexity
RTOS allows applications to be divided into small, autonomously running tasks.Tasks execute in their own context, independent of other tasks or the scheduler.
4. Peak Load Management
RTOS provides an effective method for managing peak system activities.Higher priority is assigned to tasks executing peak load activities, ensuring they access the processor within critical time frames, while lower priority tasks are delayed.
5. Tightly Integrated Middleware
The modular design of RTOS makes it easy to add middleware.Middleware components are added in a task and driver manner.They communicate with other tasks using resources provided by RTOS.Scheduled by RTOS based on corresponding events.
6. Larger Development Teams
Each task can be considered a project.Define inputs and outputs using resources provided by RTOS (queues, semaphores, etc.).Defining the system as individual tasks makes it easier to deploy more developers to a project.
7. Easy Debugging and Verification
The system is divided into well-defined tasks that do not depend on other tasks.Each task can be easily debugged and verified before the entire system integration.
8. Code Reusability
The modular design of RTOS encourages the creation of software functions as independent, verified tasks.Their independence makes it easy to reuse these modules in other designs.
Now MCU resources are more abundant than before, making many scenarios suitable for running RTOS; of course, there are still some scenarios where bare metal is sufficient.
To help more friends understand the differences between bare metal and RTOS, let’s share the principles of bare metal systems and RTOS multi-threaded systems.

Logical Systems

Bare metal systems are usually divided intoPolling Systems and Foreground-Background Systems.

1. Polling System

A polling system initializes the relevant hardware first and then lets the main program run in an infinite loop, sequentially doing various tasks. The pseudo code is illustrated in the code listing:

int main(void)
{
 /* Hardware related initialization */
 HardWareInit();

 /* Infinite loop */
 for (;;) {
   /* Handle task 1 */
   DoSomething1();

   /* Handle task 2 */
   DoSomethingg2();

   /* Handle task 3 */
   DoSomethingg3();
 }
}

A polling system is a very simple software structure, usually only suitable for tasks that can be completed by sequential code execution without needing external events to drive them. In code listing 1-1, if only LED toggling, serial output, or LCD display operations are needed, then using a polling system would be perfect. However, if external signal events such as button operations are added to simulate emergency alarms, the real-time response capability of the entire system will be compromised.

Assuming DoSomethingg3 is button scanning, when an external button is pressed, it acts like an alarm, requiring an immediate response and urgent handling. However, at that moment, the program is executing DoSomethingg1, which takes a long time to complete. If it takes so long that the button is released before DoSomethingg1 finishes executing, the event will be missed when it finally executes DoSomethingg3. Therefore, polling systems are only suitable for sequentially executed functional code; when external events drive the system, real-time performance decreases.

2. Foreground-Background System

Compared to the polling system, the foreground-background system adds interrupts. The response to external events is completed in the interrupt, and the event handling is still completed in the polling system. Here, we refer to the interrupt as the foreground, and the infinite loop in the main function as the background. The pseudo code is illustrated in the code listing:

int flag1 = 0;
int flag2 = 0;
int flag3 = 0;

int main(void)
{
 /* Hardware related initialization */
 HardWareInit();

 /* Infinite loop */
 for (;;) {
   if (flag1) {
     /* Handle task 1 */
     DoSomething1();
   }

   if (flag2) {
     /* Handle task 2 */
     DoSomethingg2();
   }

   if (flag3) {
     /* Handle task 3 */
     DoSomethingg3();
   }
 }
}

void ISR1(void)
{
 /* Set flag */
 flag1 = 1;
 /* If the event handling time is short, handle it in the interrupt
 If the event handling time is long, handle it in the background */
 DoSomething1();
}

void ISR2(void)
{
 /* Set flag */
 flag2 = 2;
 /* If the event handling time is short, handle it in the interrupt
 If the event handling time is long, handle it in the background */
 DoSomething2();
}

void ISR3(void)
{
 /* Set flag */
 flag3 = 1;
 /* If the event handling time is short, handle it in the interrupt
 If the event handling time is long, handle it in the background */
 DoSomething3();
}

While executing the background program sequentially, if an interrupt occurs, the interrupt will interrupt the normal execution flow of the background program and execute the interrupt service routine. In the interrupt service routine, events are flagged, and if the event handling is short, it can be handled in the interrupt service routine; if it is longer, it returns to the background program for handling.

Although the response and handling of events are separated, event handling still executes sequentially in the background. However, compared to the polling system, the foreground-background system ensures that events are not lost. Additionally, with the ability to nest interrupts, this greatly improves the program’s real-time response capability. In most small to medium-sized projects, a well-implemented foreground-background system can achieve an effect similar to that of an operating system.

Multi-Threaded Systems

Compared to the foreground-background system, the event response in a multi-threaded system is also completed in the interrupt, but the event handling is done within threads.In a multi-threaded system, threads, like interrupts, also have priorities, and higher priority threads are executed first.

When an urgent event is flagged in the interrupt, if the corresponding thread’s priority is high enough, it will respond immediately. Compared to the foreground-background system, the real-time performance of multi-threaded systems is further improved.

The pseudo code for a multi-threaded system is illustrated in the code listing:

int flag1 = 0;
int flag2 = 0;
int flag3 = 0;

int main(void)
{
 /* Hardware related initialization */
 HardWareInit();

 /* OS Initialization */
 RTOSInit();

 /* OS Start, begin multi-thread scheduling, do not return */
 RTOSStart();
}

void ISR1(void)
{
 /* Set flag */
 flag1 = 1;
}

void ISR2(void)
{
 /* Set flag */
 flag2 = 2;
}

void ISR3(void)
{
 /* Set flag */
 flag3 = 1;
}

void DoSomething1(void)
{
 /* Infinite loop, cannot return */
 for (;;) {
   /* Thread entity */
   if (flag1) {

   }
 }
}

void DoSomething2(void)
{
 /* Infinite loop, cannot return */
 for (;;) {
   /* Thread entity */
   if (flag2) {

   }
 }
}

void DoSomething3(void)
{
 /* Infinite loop, cannot return */
 for (;;) {
   /* Thread entity */
   if (flag3) {

   }
 }
}

Compared to the sequentially executing program body in the background of the foreground-background system, in the multi-threaded system, the program body is divided into independent, infinite loops that cannot return, which we call threads.

Each thread is independent, does not interfere with others, and has its own priority, managed by the operating system. With the addition of an operating system, programming becomes simpler as there is no need to carefully design the execution flow of the program and no concern about interference between functional modules.

With the addition of the operating system, our programming actually becomes simpler. The only additional overhead is the small amount of FLASH and RAM occupied by the operating system. Nowadays, the FLASH and RAM of microcontrollers are increasing, more than enough to cover the overhead of RTOS.

Differences between polling, foreground-background, and multi-threaded system software models:

Advantages of Running RTOS on MCU Compared to Bare Metal

———— END ————
Advantages of Running RTOS on MCU Compared to Bare Metal
● Column “Embedded Tools”
● Column “Embedded Development”
● Column “Keil Tutorial”
● Selected Tutorials from Embedded Column
Follow the public account Reply “Join Group” to join the technical exchange group according to the rules, reply “1024” to see more content.
Advantages of Running RTOS on MCU Compared to Bare Metal
Advantages of Running RTOS on MCU Compared to Bare Metal
Click “Read Original” to see more shares.

Leave a Comment