【学习日记】【FreeRTOS】调度器函数实现详解

写在前面

本文主要是对于 FreeRTOS 中调度器函数实现的详细解释,代码大部分参考了野火 FreeRTOS 教程配套源码,作了一小部分修改。

一、MSP 和 PSP

Cortex-M有两种栈空间,主堆栈和进程堆栈。

  • MSP 用于系统级别和中断处理的堆栈
    • MSP 用于保存中断发生时的堆栈状态以及在特殊操作(例如任务切换)期间的堆栈状态。MSP 在启动时会被设置为合适的内存地址,并在系统级代码运行期间始终保持不变。
  • PSP 用于任务级别的堆栈
    • 用于保存任务执行期间的局部变量、函数调用、参数等。在任务切换时,任务的 PSP 被保存,并加载下一个任务的 PSP。每个任务有自己独立的堆栈空间,并且在任务切换时,PSP 的值会发生变化。

FreeRTOS中:中断用MSP,中断以外用PSP。

二、调度器函数逻辑

在这里插入图片描述

三、调度器函数详解

1.vTaskStartScheduler()

  • 本函数为调度器的启动函数
  • pxCurrentTCB 是一个在 task.c 定义的全局指针,用于指向当前正在运行或者即将要运行的任务的任务控制块
  • 目前没有使用优先级,所以手动指定第一个运行的任务
  • 调用 xPortStartScheduler() 启动调度器
void vTaskStartScheduler( void )
{/* 手动指定第一个运行的任务 */pxCurrentTCB = &Task1TCB;/* 启动调度器 */if( xPortStartScheduler() != pdFALSE ){/* 调度器启动成功,则不会返回,即不会来到这里 */}
}

2.xPortStartScheduler()

  1. 配置PendSV 和 SysTick 的中断优先级为最低
  2. 调用函数 prvStartFirstTask()启动第一个任务
BaseType_t xPortStartScheduler( void )
{/*PendSV是一个用于低优先级任务切换的软件中断。通过触发PendSV中断,可以请求处理器在合适的时间切换到更高优先级的任务。PendSV中断具有最低的中断优先级,因此可以在其他中断处理完成后立即执行。*//* 配置PendSV 和 SysTick 的中断优先级为最低 */portNVIC_SYSPRI2_REG |= portNVIC_PENDSV_PRI;portNVIC_SYSPRI2_REG |= portNVIC_SYSTICK_PRI;/* 启动第一个任务,不再返回 */prvStartFirstTask();/* 不应该运行到这里 */return 0;
}

3.prvStartFirstTask()

  • 用于初始化启动第一个任务的环境,主要是重新设置MSP指针,并使能全局中断

  • 调度器启动函数xPortStartScheduler( void )调用:

  • prvStartFirstTask函数:

  1. PRESERVE8 指令保留 8 字节栈对齐
  2. 取出向量表起始地址对应的内容
  3. 使用向量表起始地址对应的内容设置主堆栈指针msp的值
  4. 使能全局中断
  5. 使用 dsb 和 isb 指令确保数据和指令同步
  6. 调用SVC去启动第一个任务
/** 参考资料《STM32F10xxx Cortex-M3 programming manual》4.4.3,百度搜索“PM0056”即可找到这个文档* 在Cortex-M中,内核外设SCB的地址范围为:0xE000ED00-0xE000ED3F* 0xE000ED008为SCB外设中SCB_VTOR这个寄存器的地址,里面存放的是向量表的起始地址,即MSP的地址*/
__asm void prvStartFirstTask( void )
{/*使用 PRESERVE8 指令保留 8 字节栈对齐*/PRESERVE8/* 在Cortex-M中,0xE000ED08是SCB_VTOR这个寄存器的地址,里面存放的是向量表的起始地址,即MSP的地址 */
//	向量表通常是从内部 FLASH 的起始地址开
//	始存放,那么可知 memory:0x00000000 处存放的就是 MSP 的值。ldr r0, =0xE000ED08ldr r0, [r0]	//把 0xE000ED08 处向量表起始地址取出ldr r0, [r0]	//取出向量表起始地址对应的内容/* 设置主堆栈指针msp的值 */msr msp, r0/* 使能全局中断 */cpsie i	//开中断 PRIMASK=0cpsie f	//开异常 FAULTMASK=0/*使用 dsb 和 isb 指令确保数据和指令同步*/
//1. dsb 指令:dsb 指令用于确保数据的同步。它会强制在 dsb 指令之
//	前的所有数据访问和加载操作完成,然后再继续执行 dsb 指令后面
//	的指令。这样可以确保所有数据操作在 dsb 指令之前都已经完成,
//	避免数据争用和不一致性的问题。
//2. isb 指令:isb 指令用于确保指令的同步。它会刷新处理器流水线中
//的指令,并确保在 isb 指令之前的所有指令都已经执行完毕,然后再继
//续执行 isb 指令后面的指令。这样可以确保流水线中的指令执行顺序与
//程序中的顺序一致,避免指令重排和乱序执行带来的问题。dsbisb/* 调用SVC去启动第一个任务 */
//	"Supervisor Call"(超级用户调用),
//	用于从用户模式(通常是应用程序运行的模式)
//	切换到特权模式(通常是操作系统内核运行的模式)
//	执行一段特权代码,以执行一些需要特权级别权限的操作或服务svc 0  //服务号 0表示 SVC 中断,接下来将会执行 SVC 中断服务函数nopnop
}
  • 关于Cortex-M中三个中断屏蔽寄存器
    在这里插入图片描述

4.vPortSVCHandler()

  • 本函数为 SVC 的中断服务函数
  1. 加载 TCB 到 r0,以 r0 为基地址,将栈里面的内容加载到 r4~r11 寄存器
  2. 开启所有中断
  3. 设置 r14 寄存器,以使用 PSP 出栈,进入线程模式,返回 Thumb 状态
  4. 如果异常返回,则 bx r14 进入 Thumb 状态,并且栈中的剩下内容将会自动加载到CPU寄存器
//SVC中断函数
__asm void vPortSVCHandler( void )
{extern pxCurrentTCB;	//1. 加载要运行的 TCB 的指针PRESERVE8ldr	r3, =pxCurrentTCB	//2. 加载要运行的 TCB 的指针的地址到 r3ldr r1, [r3]			//3. 加载要运行的 TCB 的指针到 r1ldr r0, [r1]			//4. 加载 TCB 到 r0,目前 r0 的值等于第一个任务堆栈的栈顶ldmia r0!, {r4-r11}		//5. 以 r0 为基地址,将栈里面的内容加载到 r4~r11 寄存器,同时r0会递增msr psp, r0				//6. 将r0的值,即任务的栈指针更新到 pspisb						//7. 等待指令同步mov r0, #0 msr	basepri, r0         //8. 设置basepri寄存器的值为0,即所有的中断都没有被屏蔽orr r14, #0xd           //9. 当从SVC中断服务退出前,通过向r14寄存器最后4位按位或上0x0D,//   使得硬件在退出时使用进程堆栈指针PSP完成出栈操作并返回后进入线程模式、返回Thumb状态bx r14                  //10. 异常返回,这个时候栈中的剩下内容将会自动加载到CPU寄存器://    xPSR,PC(任务入口地址),R14,R12,R3,R2,R1,R0(任务的形参)//    同时PSP的值也将更新,即指向任务栈的栈顶
}

执行成功后,PSP 的指向(图片来自野火):
在这里插入图片描述

ARM 状态和 Thumb 状态详解

在 ARM 架构中,ARM 状态和 Thumb 状态是指处理器运行的不同工作模式。这些模式决定了处理器执行代码的指令集。

  • ARM 状态:
  1. 在 ARM 状态下,处理器执行 ARM 指令集。这些指令集是 32 位宽度的。
  2. ARM 状态提供了更高的代码密度和更强大的功能,可以执行更复杂的指令。
  3. ARM 状态下的指令集包括了更多的寄存器和更多的数据处理指令。
  4. ARM 指令使用的是 32 位的寄存器。
  5. 进入 ARM 状态可以使用跳转指令 bx。
  • Thumb 状态:
  1. 在 Thumb 状态下,处理器执行 Thumb 指令集。这些指令集是 16 位宽度的,它们可以通过压缩来提供更好的代码密度。
  2. Thumb 状态下的指令集相对于 ARM 状态来说更为紧凑,但功能上略有限制。
  3. Thumb 指令使用的是 16 位的寄存器,这些寄存器只能存放 16 位的数据。
  4. 进入 Thumb 状态可以使用跳转指令 bx。

后记

如果您觉得本文写得不错,可以点个赞激励一下作者!
如果您发现本文的问题,欢迎在评论区或者私信共同探讨!
共勉!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/31179.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

linux配置上网 linux adsl拨号上网设置

Linux里面配置ADSL上网是件很麻烦的事。但配置完成之后就能开机自动拨号上网,可谓十分的方便。支持的系统有Redhat,CentOS,SuSE,FreeBSD,Ubuntu等常见的Linux。 工具/原料 ADSL网络,电信,网通,移动等常见宽带。 Linux系统的安装光…

MinGW-W64 下载、安装与配置(支持最新版的GCC,目前 GCC 13.2.0)

文章目录 一、简介1. MinGW 和 MinGW-W64 区别和联系2. MSVCRT 和 UCRT 介绍 二、下载1. 从 sourceforge.net 下载2. 从 github 下载3. 从 镜像站点 下载4. 自己编译 三、安装与配置1. 在线安装2. 离线安装3. 环境配置 四、总结 一、简介 1. MinGW 和 MinGW-W64 区别和联系 M…

LinearAlgebraMIT_8_TheRankOfMatrix

这节课中主要讲解根据秩来判断方程组/矩阵的(solvability)解情况,即通过秩来判断(aumented matrix)增广矩阵的解。我们需要直接求解方程组的解就是求解矩阵的解。 x.1 判断(非齐次线性方程组)Axb是否有解 我们以下面这个方程组为例,它具有3个约束条件和…

MyBatis and or使用列表控制or条件

背景&#xff1a;最近项目需要&#xff0c;师傅可以查找订单&#xff0c;而师傅是指定可以服务2到3个区域&#xff0c;故需要使用到and, or条件的组合&#xff0c;以下记一下代码。 最重要的代码是&#xff1a; 1、构建List<Consumer<LambdaQueryWrapper<T>>&g…

【微信小程序】通过使用 wx.navigateTo方法进行页面跳转,跳转后的页面中通过一些方式回传值给原页面

以下是几种常见的回传值的方式&#xff1a; 使用 wx.navigateTo 方法传递参数&#xff1a; 在跳转时&#xff0c;可以在目标页面的 URL 中携带参数&#xff0c;然后在目标页面的 onLoad 方法中获取参数&#xff0c;并在目标页面中进行处理。例如&#xff1a; // 原页面跳转到目…

26.Netty源码之ThreadLocal

highlight: arduino-light JDK ThreadLocal 如果你需要变量在多线程之间隔离&#xff0c;或者在同线程内的类和方法中共享&#xff0c;那么 ThreadLocal 大显身手的时候就到了。ThreadLocal 可以理解为线程本地变量&#xff0c;它是 Java 并发编程中非常重要的一个类。 ThreadL…

【BASH】回顾与知识点梳理(二十)

【BASH】回顾与知识点梳理 二十 二十. 十六至十九章知识点总结及练习20.1 总结20.2 练习 该系列目录 --> 【BASH】回顾与知识点梳理&#xff08;目录&#xff09; 二十. 十六至十九章知识点总结及练习 20.1 总结 shell script 是利用 shell 的功能所写的一个『程序 (prog…

《OWASP代码审计》学习——跨站脚本注入(XSS)

一、跨站脚本概述 1.什么是跨站脚本 跨站点脚本(XSS)是一种编码注入漏洞。它通常出现在 web 应用程序中。XSS 使攻击者能够向其他用户浏览的网页中注入恶意内容。XSS 允许攻击者绕过访问控制&#xff0c;它是 OWASP Top10 最常见的漏洞之一。XSS 是网络服务器上的第二大漏洞。…

基于java快餐店线上点餐系统设计与实现

摘 要 随着计算机互联网的高速发展。餐饮业的发展也加入了电子商务团队。各种网上点餐系统纷纷涌现&#xff0c;不仅增加了商户的销售量和营业额&#xff0c;而且为买家提供了极大的方便&#xff0c;足不出户&#xff0c;就能订到喜欢的餐品。网上点餐已经是人们生活中所不可缺…

Linux系统性能调优及调试课:Linux Kernel Printk

🚀返回专栏总目录 文章目录 0、printk 说明1、printk 日志等级设置2、屏蔽等级日志控制机制3、printk打印常用方式4、printk打印格式0、printk 说明 在开发Linux device Driver或者跟踪调试内核行为的时候经常要通过Log API来trace整个过程,Kernel API printk()是整个Kern…

amis百度前端框架,在js中使用amis写json转页面

amis百度前端框架&#xff0c;在js中使用用amis写的json页面 1.在项目中使用百度 amis 的sdk做开发库。 <script src"./sdk/sdk/sdk.js"></script> 2。加载sdk中的库&#xff1a; amis amisRequire(amis/embed);amisLib amisRequire(amis);const ma…

Flume拦截器

实现 Interceptor接口 方法1 是初始化: 方法2和3重载 拦截: 方法3 是关闭: 但是flume是通过内部类创建对象的

餐饮管理系统ssm酒店饭店仓库进销存jsp源代码mysql

本项目为前几天收费帮学妹做的一个项目&#xff0c;Java EE JSP项目&#xff0c;在工作环境中基本使用不到&#xff0c;但是很多学校把这个当作编程入门的项目来做&#xff0c;故分享出本项目供初学者参考。 一、项目描述 餐饮管理系统ssm 系统有1权限&#xff1a;管理员 二…

flask-sqlalchemy使用

# sqlalchemy 集成到flask中 # 第三方: flask-sqlalchemy 封装了用起来&#xff0c;更简洁 安装 pip install flask-sqlalchemy 使用 # 使用flask-sqlalchemy集成1 导入 from flask_sqlalchemy import SQLAlchemy2 实例化得到对象db SQLAlchemy()3 将db注册到app中db.in…

Vue中使用qrcode说明

1.安装 npm i qrcode1.5.3 2.导入 import QRCode from qrcode 3.转换 说明&#xff1a;拿到服务器传来的字符串&#xff0c;转换成base64&#xff0c;然后通过img标签展示。 // 字符串转成二维码 let result await this.$API.reqPayInfo(this.orderId); 总结&#xff1a;

Java一般用于postgis空间数据库通用的增删查改sql命令

目录 1 增加 2 删除 3 查询 4 更新 "public"."JGSQGW_Geo"为某模式下得表 一般postgrel有这样的设计模式 1 增加 #前端绘制出的数据插入 INSERT INTO "public"."JGSQGW_Geo" ( "geom","gridone","gridon…

集合Collection-List-ArrayList学习

一、集合 集合是数据容器。相较于数组集合具有以下几个特点&#xff1a; 数组一旦创建&#xff0c;长度不可改变。集合的长度会自动扩容。集合具有很多数组没有的功能函数API数组元素的存储特点单一&#xff0c;不同的集合有不同的存储特点。 1. Collection顶层接口 Collect…

Python3.x String内置函数大全

文章目录 总结一下Python3.x字符串的常用系统函数&#xff0c;总共分为8类1. 大小写字母转换类的函数str.capitalize()str.title()str.lower()str.upper()str.swapcase() 2. 统计类的函数str.count(str1, beg 0,endlen(string)) 3. 匹配类的函数str.endswith(suffix, beg0, end…

zustand:基于 Flux 模型实现的小型、快速和可扩展的状态管理

目录 ReactStep 1&#xff1a;安装Step 2&#xff1a;Store 初始化Step3&#xff1a;Store 绑定组件&#xff0c;就完成了!效果图 VueStep 1: 安装Step 2: Store 初始化Step 3: Store 绑定组件&#xff0c;就完成了!效果图 微前端为什么你需要 zustand-pub &#xff1f;安装ste…

Centos8上加速git clone

首先通过命令获取域名对应的IP地址 [rootggbond ~]# nslookup github.global.ssl.fastly.net [rootggbond ~]# nslookup github.com 之后如上获取到的IP地址 以IP-域名的格式加入到hosts文件中 [rootggbond ~]# vim /etc/hosts Centos8上更新DNS缓存 [rootggbond ~]# nscd -…