FreeRTOS深入教程(任务的引入及栈的作用)

简介: FreeRTOS深入教程(任务的引入及栈的作用)

前言

本篇文章开始带大家深入学习FreeRTOS,带大家学习什么是任务,并且深入学习栈的作用。

一、任务的引入

在 FreeRTOS 中,任务(Task)是一个基本的执行单元,它代表了一个并行执行的工作单元。FreeRTOS 是一个实时操作系统,允许你创建多个任务,每个任务都有自己的代码、堆栈和优先级。这些任务可以独立运行,以实现多任务并发。

以下是关于 FreeRTOS 任务的一些关键概念:

任务的特点:

每个任务都有自己的独立堆栈,用于保存任务的局部变量和上下文信息

任务可以具有不同的优先级,操作系统根据任务的优先级来调度它们的执行。

任务可以是无限循环的,也可以在完成其工作后自行删除。

每个任务都有一个唯一的任务句柄,可以用于管理和控制任务。

二、深入理解C语言函数的调用

1.ARM架构

在我们看来执行a+b是一件非常简单的事情,但是在CPU的内部却做了非常多的操作。

CPU运行时,先去Flash上取得指令,再执行指令:

  • 把内存a的值读入CPU寄存器R0
  • 把内存b的值读入CPU寄存器R1
  • 把R0、R1累加,存入R0
  • 把R0的值写入内存a

2.基础汇编指令

LDR(Load Register):

LDR 指令用于将数据从内存加载到寄存器中。

典型的 LDR 指令的语法:LDR Rd, [Rn, #Offset],其中 Rd 是目标寄存器,Rn 是基址寄存器,Offset 是偏移量。

举例:LDR R0, [R1, #4] 表示将存储在地址 (R1 + 4) 处的数据加载到 R0 寄存器中。

STR(Store Register):

STR 指令用于将寄存器中的数据存储到内存中。

典型的 STR 指令的语法:STR Rd, [Rn, #Offset],其中 Rd 是源寄存器,Rn 是基址寄存器,Offset 是偏移量。

举例:STR R0, [R1, #4] 表示将 R0 寄存器中的数据存储到地址 (R1 + 4) 处。

ADD(Add):

ADD 指令用于将两个操作数相加,并将结果存储在目标寄存器中。

典型的 ADD 指令的语法:ADD Rd, Rn, Operand2,其中 Rd 是目标寄存器,Rn 是第一个操作数寄存器,Operand2 是第二个操作数。

举例:ADD R0, R1, #10 表示将 R1 寄存器中的值与 10 相加,并将结果存储在 R0 中。

POP:

POP 指令用于从栈中弹出多个寄存器的值。

POP 指令的操作是根据栈指针(通常是 SP 寄存器)从栈中弹出值,同时更新栈指针。

通常用于函数返回时,以恢复之前保存的寄存器状态。

POP {R3, PC}

PUSH:

PUSH 指令用于将多个寄存器的值推送(压入)到栈上。

PUSH 指令的操作是根据栈指针(通常是 SP 寄存器)将值压入栈中,同时更新栈指针。

通常用于函数调用时,以保存当前寄存器状态。

PUSH {R3, LR}

3.函数运行流程分析

在keil5中编写一个加法函数:

/* a = a + b */
void add_val(int *pa, int *pb)
{
  //*pa = *pa + *pb;
  volatile int tmp;
  tmp = *pa;
  tmp = tmp + *pb;
  *pa = tmp;
}
int a = 1;
int b = 2;
add_val(&a, &b);  

汇编代码:

汇编代码分析:

首先执行PUSH  {r3,lr}命令
将r3寄存器的值和lr返回地址保存到了栈中。
r3就是局部变量tmp的值。

tmp = *pa;

LDR      r2,[r0,#0x00]   读取r0的值到r2中,也就是将*pa读取到r2中。
STR      r2,[sp,#0x00]   将r2的值保存到sp+0x00地址处也就是将*pa保存到tmp中。
      完成tmp = *pa;赋值语句

tmp = tmp + *pb;

LDR      r2,[r1, #0x00] 读取r1的值到r2中,也就是将*pb读取到r2中
LDR    r3,[sp,#0x00] 读取sp+0x00的值到r3中,也就是将tmp读取到r3中
ADD    r2,r2,r3 将r2+r3的和保存到r2中,也就是将*pb + tmp的值保存到r2中
STR    r2,[sp,#0x00] 将r2的值保存到sp中也就是tmp = *pb + tmp

*pa = tmp;

LDR    r2,[sp,#0x00] 读取sp+0x00的值保存到r2中,也就是将tmp保存到r2中
STR    r2,[r0,#0x00] 将r2的值保存到r0中,也就是*pa = tmp

POP {r3,lr}

出栈将lr的值返回给pc寄存器,执行下一条指令。

三.保存现场的几种情况

1.函数调用

当函数调用时R0,R1,R2通常被用来传递参数故不需要保存这几个寄存器的值。

void funA(void)
{
  .........
  funB(a, b);
}

2.中断处理

被中断打断时硬件会帮我们保存R0,R1,R2寄存器,其他寄存器需要通过软件来保存。

void funA(void)
{
  ->被中断打断
}

3.任务切换

当任务切换时需要保存所有的寄存器,因为任务并不知道会使用到哪些寄存器。

总结

本篇文章主要引入了任务及说明了栈在这里的作用。


相关文章
|
14天前
|
C语言
【数据结构】栈和队列(c语言实现)(附源码)
本文介绍了栈和队列两种数据结构。栈是一种只能在一端进行插入和删除操作的线性表,遵循“先进后出”原则;队列则在一端插入、另一端删除,遵循“先进先出”原则。文章详细讲解了栈和队列的结构定义、方法声明及实现,并提供了完整的代码示例。栈和队列在实际应用中非常广泛,如二叉树的层序遍历和快速排序的非递归实现等。
90 9
|
5天前
|
存储 算法
非递归实现后序遍历时,如何避免栈溢出?
后序遍历的递归实现和非递归实现各有优缺点,在实际应用中需要根据具体的问题需求、二叉树的特点以及性能和空间的限制等因素来选择合适的实现方式。
14 1
|
8天前
|
存储 算法 Java
数据结构的栈
栈作为一种简单而高效的数据结构,在计算机科学和软件开发中有着广泛的应用。通过合理地使用栈,可以有效地解决许多与数据存储和操作相关的问题。
|
11天前
|
存储 JavaScript 前端开发
执行上下文和执行栈
执行上下文是JavaScript运行代码时的环境,每个执行上下文都有自己的变量对象、作用域链和this值。执行栈用于管理函数调用,每当调用一个函数,就会在栈中添加一个新的执行上下文。
|
13天前
|
存储
系统调用处理程序在内核栈中保存了哪些上下文信息?
【10月更文挑战第29天】系统调用处理程序在内核栈中保存的这些上下文信息对于保证系统调用的正确执行和用户程序的正常恢复至关重要。通过准确地保存和恢复这些信息,操作系统能够实现用户模式和内核模式之间的无缝切换,为用户程序提供稳定、可靠的系统服务。
40 4
|
17天前
|
算法 安全 NoSQL
2024重生之回溯数据结构与算法系列学习之栈和队列精题汇总(10)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第3章之IKUN和I原达人之数据结构与算法系列学习栈与队列精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
30天前
数据结构(栈与列队)
数据结构(栈与列队)
17 1
|
1月前
|
存储 JavaScript 前端开发
为什么基础数据类型存放在栈中,而引用数据类型存放在堆中?
为什么基础数据类型存放在栈中,而引用数据类型存放在堆中?
67 1
|
1月前
【数据结构】-- 栈和队列
【数据结构】-- 栈和队列
16 0
|
1月前
探索顺序结构:栈的实现方式
探索顺序结构:栈的实现方式