CM3基础(持续更新)

2024-03-31 06:38
文章标签 基础 更新 持续 cm3

本文主要是介绍CM3基础(持续更新),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

CM3基础

操作模式和特权级别

参考Cortex-M3操作模式和特权级别

总结

系统有两种模式,handler模式和线程模式,前者始终位于特权级(用于处理异常),后者有特权和用户级两种。特权级别能够访问更多的数据范围。在系统复位后运行在线程模式的特权级下,位于特权级可以通过置为control[0]来进入用户级,而用户级别要进入特权级,只能被动等待异常出现然后进入handler模式来进入特权级。如下图
特权级和用户级

MSP&PSP

SP是堆栈指针,用于指向系统当前使用的堆栈位置,其实际位置取决于MSP或者PSP指向的地址,Cortex-M3的有两个堆栈指针寄存器MSP和PSP,任何时刻只能使用一个寄存器作为SP堆栈指针的索引值,可以在特权模式下通过修改control[1]来设置当前使用哪个寄存器,当系统位于handle模式时,只能使用MSP,即handle模式下虽然是特权模式但control[1]=0(即只能使用MSP),当处于线程模式时,则在特权级时可以设置为0(MSP)或者1(PSP)。
当然,我们不用担心何时去切换SP所对应寄存器地址,牢记一点:当处于中断中时,系统使用MSP,出中断前后使用PSP。更一般的,进入中断前,系统会自动使用PSP保存部分寄存器(R0-R3,R12,14,15,XPSR)到任务堆栈,在出中断前,系统会调用SP恢复寄存器,但出中断时仍处于MSP,所以在出中断前要先设置当前SP参考寄存器为PSPORR LR, LR, #0x04,这样出中断过程中会根据PSP恢复对应的软件堆栈寄存器(R0-R3,R12,14,15,XPSR)。


使用这种双指针是为了保护系统内核的堆栈数据,当系统位于软件层面时,系统使用的堆栈指针为PSP,而处理系统级别事务时则使用MSP,由于两个寄存器的储存值不同,系统堆栈和任务堆栈被人为分开了,这样能够确保软件层面的操作不会影响到系统层面。
MSP堆栈指针实际上就是整个SRAM的堆栈指针,而PSP指针只是作为任务堆栈指针来使用。
下图为MSP堆栈指针的初始化示例。因为代码中使用的数据结构可能不同(如使用了大量的数组时,位于最后的堆栈指针MSP所指的位置肯定要更靠后,即地址更高),导致实际的MSP的初始位置不同。
MSP和PSP堆栈1
在初始化SRAM时,诸如数组这样的全局变量占用SRAM低地址空间,这些数组在系统中就作为任务堆栈的储存空间,数组的末尾(数组的高地址段)就作为储存任务寄存器的地方,其中PSP就指向所压入堆栈的最后一个寄存器的地址。这样在空间上就把任务堆栈(PSP)和系统堆栈(MSP)分离开,分别标志不同的地址位,互不干扰。同时,不同的任务都有一个自己保存好的任务堆栈栈顶备份(位于TCB中),当切换回该任务时,会把自己的栈顶备份写入PSP寄存器,实现不同任务间的切换。
MSP和PSP指向不同的堆栈位置
MSP和PSP堆栈2
参考
Cortex-M3 双堆栈指针(MSP&PSP)
STM32双堆栈及其在uC/OS-II中的使用
PendSVC切换任务时,PSP和MSP的使用

这篇关于CM3基础(持续更新)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/863665

相关文章

poj3468(线段树成段更新模板题)

题意:包括两个操作:1、将[a.b]上的数字加上v;2、查询区间[a,b]上的和 下面的介绍是下解题思路: 首先介绍  lazy-tag思想:用一个变量记录每一个线段树节点的变化值,当这部分线段的一致性被破坏我们就将这个变化值传递给子区间,大大增加了线段树的效率。 比如现在需要对[a,b]区间值进行加c操作,那么就从根节点[1,n]开始调用update函数进行操作,如果刚好执行到一个子节点,

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

hdu1689(线段树成段更新)

两种操作:1、set区间[a,b]上数字为v;2、查询[ 1 , n ]上的sum 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdl

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

hdu 1754 I Hate It(线段树,单点更新,区间最值)

题意是求一个线段中的最大数。 线段树的模板题,试用了一下交大的模板。效率有点略低。 代码: #include <stdio.h>#include <string.h>#define TREE_SIZE (1 << (20))//const int TREE_SIZE = 200000 + 10;int max(int a, int b){return a > b ? a :

AI行业应用(不定期更新)

ChatPDF 可以让你上传一个 PDF 文件,然后针对这个 PDF 进行小结和提问。你可以把各种各样你要研究的分析报告交给它,快速获取到想要知道的信息。https://www.chatpdf.com/

GIS图形库更新2024.8.4-9.9

更多精彩内容请访问 dt.sim3d.cn ,关注公众号【sky的数孪技术】,技术交流、源码下载请添加微信:digital_twin123 Cesium 本期发布了1.121 版本。重大新闻,Cesium被Bentley收购。 ✨ 功能和改进 默认启用 MSAA,采样 4 次。若要关闭 MSAA,则可以设置scene.msaaSamples = 1。但是通过比较,发现并没有多大改善。

【Linux 从基础到进阶】Ansible自动化运维工具使用

Ansible自动化运维工具使用 Ansible 是一款开源的自动化运维工具,采用无代理架构(agentless),基于 SSH 连接进行管理,具有简单易用、灵活强大、可扩展性高等特点。它广泛用于服务器管理、应用部署、配置管理等任务。本文将介绍 Ansible 的安装、基本使用方法及一些实际运维场景中的应用,旨在帮助运维人员快速上手并熟练运用 Ansible。 1. Ansible的核心概念

AI基础 L9 Local Search II 局部搜索

Local Beam search 对于当前的所有k个状态,生成它们的所有可能后继状态。 检查生成的后继状态中是否有任何状态是解决方案。 如果所有后继状态都不是解决方案,则从所有后继状态中选择k个最佳状态。 当达到预设的迭代次数或满足某个终止条件时,算法停止。 — Choose k successors randomly, biased towards good ones — Close

JavaFX应用更新检测功能(在线自动更新方案)

JavaFX开发的桌面应用属于C端,一般来说需要版本检测和自动更新功能,这里记录一下一种版本检测和自动更新的方法。 1. 整体方案 JavaFX.应用版本检测、自动更新主要涉及一下步骤: 读取本地应用版本拉取远程版本并比较两个版本如果需要升级,那么拉取更新历史弹出升级控制窗口用户选择升级时,拉取升级包解压,重启应用用户选择忽略时,本地版本标志为忽略版本用户选择取消时,隐藏升级控制窗口 2.