当前位置: 首页 > news >正文

FreeRTOS菜鸟入门(二十)·ARM架构简介

目录

1. 前提

2. ARM架构

3. ARM 汇编指令

3.1 LDR(Load Register):读内存

3.2 STR(Store Register):写内存

3.3 ADD(加法)

3.4 SUB(减法)

3.5 比较指令 CMP

3.6 B(Branch):直接跳转

3.7 BL(Branch and Link):带链接跳转

4. 实例讲解

拓展:反汇编命令


1. 前提

该章节是对一些知识点扫盲,想要了解更多FreeRTOS相关可以查看:

FreeRTOS菜鸟入门系列_时光の尘的博客-CSDN博客

FreeRTOS实战系列_时光の尘的博客-CSDN博客

2. ARM架构

这里简单了解一下ARM架构,ARM 架构的全称是Advanced RISC Machine,天生就是为 RISC(Reduced Instruction Set Computing,精简指令集计算机)设计的,它的所有特性都围绕 RISC 的核心思想展开,是 RISC 架构最成功的商业化代表之一。

RISC 的核心是:用精简的硬件逻辑,实现高效、低功耗的指令执行,和传统 CISC(复杂指令集)的用复杂硬件兼容一切思路完全相反:

  • 优先实现高频使用的简单指令,单条指令只做一件事,执行周期固定,大多能在 1 个时钟周期内完成
  • 把复杂操作交给编译器,通过多条简单指令的组合实现,而非硬件内置复杂指令
  • 硬件设计简化,晶体管用量少,功耗更低,天生适配移动设备、嵌入式场景
对比维度RISC(ARM 架构)CISC(x86 架构)
指令集指令数量少(几十~上百条),功能单一指令数量多(数百~上千条),支持复杂操作
内存访问仅 Load/Store 指令可访问内存多数指令可直接操作内存数据
指令长度固定长度(如 ARM32 固定 4 字节)可变长度(1~15 字节不等)
硬件复杂度低,解码逻辑简单,易做低功耗设计高,解码单元复杂,硬件成本高
功耗表现低,适配移动、嵌入式场景较高,多用于 PC、服务器场景
代码密度较低,复杂操作需要多条指令组合较高,单条指令可实现复杂功能

我们平常使用的是Cortex-M3/M4/A7 系列 ARM 内核,这些内核都有 R0~R15 共 16 个 32 位通用寄存器,分为三类:

  • 低寄存器(Low Registers):R0~R7所有指令都可以无限制访问,是最常用的通用数据寄存器,用于暂存运算数据、函数参数等。
  • 高寄存器(High Registers):R8~R12在部分 Thumb 指令中访问受限,常用于保存临时变量或函数调用的额外数据。
  • 特殊功能寄存器:R13、R14、R15 + xPSR这三个寄存器有固定的系统级用途,是内核运行的核心控制寄存器。

对于特殊功能寄存器:

R13(SP,Stack Pointer 栈指针)

  • 指向当前栈顶的地址,用于管理栈空间(函数调用、中断处理时的现场保存 / 恢复都依赖栈)。
  • Cortex-M 内核支持双栈指针:SP_main(主栈,用于操作系统内核、异常处理)和 SP_process(进程栈,用于用户应用程序),可通过控制寄存器切换。
  • 栈操作(PUSH/POP)会自动修改 SP 的值,硬件自动维护栈平衡。

R14(LR,Link Register 链接寄存器)

  • 用于保存函数 / 子程序调用后的返回地址。
  • 当执行 BL(分支并链接)指令调用函数时,当前指令的下一条地址会自动存入 LR;函数执行结束后,通过BX LR指令即可跳回原位置。
  • 中断服务程序中,LR 会被自动压入栈,中断返回时恢复。

R15(PC,Program Counter 程序计数器)

  • 指向当前正在执行的指令的地址,CPU 每执行一条指令,PC 会自动递增。
  • 直接修改 PC 的值,就可以实现程序跳转(比如赋值跳转地址到 R15,或使用 B、BL 等跳转指令)。
  • 在 Thumb 指令集下,PC 的最低位必须为 1(表示 Thumb 状态),否则会触发硬件错误。

xPSR(Program Status Register 程序状态寄存器)

  • 图中最下方的寄存器,包含状态标志位(如 N/Z/C/V 条件码、中断屏蔽位、处理器模式位等)。
  • 条件指令(如 BEQ、BNE)的执行依赖 xPSR 中的标志位,中断响应也会修改其中的状态位。

对于一个芯片要想实现a+b需要怎么实现呢?如下图:

我们通过CPU获取内存数据存放到寄存器当中,根据FLASH当中存放的指令集进行相关操作。

3. ARM 汇编指令

3.1 LDR(Load Register):读内存

LDR R0, [R1, #4] ; 读地址 "R1+4" 处的4字节数据,存入R0寄存器

方括号 [] 表示内存地址,#4 是立即数偏移,支持多种寻址方式(如基址 + 偏移、寄存器间接寻址)。

3.2 STR(Store Register):写内存

STR R0, [R1, #4] ; 把R0寄存器中的4字节数据,写入地址 "R1+4" 处

STR 和 LDR 是成对的,前者是 “寄存器→内存”,后者是 “内存→寄存器”。

3.3 ADD(加法)

ADD R0, R1, R2 ; R0 = R1 + R2(寄存器+寄存器) ADD R0, R0, #1 ; R0 = R0 + 1(寄存器+立即数,自增)

3.4 SUB(减法)

SUB R0, R1, R2 ; R0 = R1 - R2(寄存器-寄存器) SUB R0, R0, #1 ; R0 = R0 - 1(寄存器-立即数,自减)

3.5 比较指令 CMP

CMP R0, R1 ; 计算 R0 - R1,结果不存入寄存器,只修改程序状态寄存器(PSR)的标志位

CMP 是 “条件指令” 的基础,它会设置 N/Z/C/V 等标志位,后续可以配合 BEQ(相等跳转)、BNE(不等跳转)等条件跳转指令使用。

3.6 B(Branch):直接跳转

B main ; 无条件跳转到标签 main 处执行,不保存返回地址

适用场景:死循环、无条件分支。

3.7 BL(Branch and Link):带链接跳转

BL main ; 先把下一条指令的地址保存到LR(R14)寄存器,再跳转到 main 处

适用场景:函数 / 子程序调用,后续可以通过 BX LR 指令返回原调用处。

4. 实例讲解

这里使用的代码是之前移植好的C8T6的代码:

基于STM32F103C8T6移植FreeRTOS资源-CSDN下载

对OLED_task.c进行如下修改:

#include "OLED_task.h" #include "OLED.h" #include "FreeRTOS.h" #include "task.h" int add(volatile int a, volatile int b) { volatile int sum; sum = a + b; return sum; } void oled_task_driver(void *pvParameters) { int cnt = 0; while(1) { // OLED_ShowChar(1, 1, 'A'); //1行1列显示字符A // OLED_ShowString(1, 3, "HelloWorld!"); //1行3列显示字符串HelloWorld! // OLED_ShowSignedNum(2, 7, -66, 2); //2行7列显示有符号十进制数字-66,长度为2 // OLED_ShowHexNum(3, 1, 0xAA55, 4); //3行1列显示十六进制数字0xA5A5,长度为4 // OLED_ShowBinNum(4, 1, 0xAA55, 16); //4行1列显示二进制数字0xA5A5,长度为16 OLED_ShowNum(2, 1, cnt, 5); cnt = add(cnt,1); vTaskDelay(50); } }


拓展:反汇编命令

反汇编作用是把 ARM 编译器生成的 .axf 可执行文件,转换成 人类能看懂的汇编代码文件 .dis,简单来说就是把机器码 → 反汇编成汇编代码,看一下下面这段代码:

fromelf --text -a -c --output=xxxx.dis xxxx.axf
  • fromelf:ARM 自带的格式转换 / 反汇编工具(Keil 安装目录里自带)
  • --text:输出文本格式的反汇编内容(不是二进制)
  • -a:输出地址信息(函数地址、变量地址)
  • -c:输出反汇编代码(assembly)
  • --output=xxxx.dis:把结果保存到 xxxx.dis 文件
  • xxxx.axf:输入文件(Keil 编译后生成的包含调试信息的可执行文件)

如何使用呢根据如下步骤:

对于.dis自己随便去个名字:

fromelf --text -a -c --output=test.dis .\Obj\Template.axf

对于.axf的路径可以在这里查找,改成自己的路径:

编译运行可以看到生成了.dis文件:


打开文件,这里打开方式我还是用Keil5打开,然后搜索找到i.add:

i.add add 0x08000fd4: b503 .. PUSH {r0,r1,lr} 0x08000fd6: b081 .. SUB sp,sp,#4 0x08000fd8: e9dd0101 .... LDRD r0,r1,[sp,#4] 0x08000fdc: 4408 .D ADD r0,r0,r1 0x08000fde: 9000 .. STR r0,[sp,#0] 0x08000fe0: 9800 .. LDR r0,[sp,#0] 0x08000fe2: bd0e .. POP {r1-r3,pc}

同样的方式找到 oled_task_driver:

i.oled_task_driver oled_task_driver 0x08001098: 2400 .$ MOVS r4,#0 0x0800109a: e00d .. B 0x80010b8 ; oled_task_driver + 32 0x0800109c: 2305 .# MOVS r3,#5 0x0800109e: 4622 "F MOV r2,r4 0x080010a0: 2101 .! MOVS r1,#1 0x080010a2: 2002 . MOVS r0,#2 0x080010a4: f7fffb6a ..j. BL OLED_ShowNum ; 0x800077c 0x080010a8: 2101 .! MOVS r1,#1 0x080010aa: 4620 F MOV r0,r4 0x080010ac: f7ffff92 .... BL add ; 0x8000fd4 0x080010b0: 4604 .F MOV r4,r0 0x080010b2: 2032 2 MOVS r0,#0x32 0x080010b4: f000ff9e .... BL vTaskDelay ; 0x8001ff4 0x080010b8: e7f0 .. B 0x800109c ; oled_task_driver + 4 0x080010ba: 0000 .. MOVS r0,r0

二者所对应的代码:

int add(volatile int a, volatile int b) { volatile int sum; sum = a + b; return sum; } void oled_task_driver(void *pvParameters) { int cnt = 0; while(1) { // OLED_ShowChar(1, 1, 'A'); //1行1列显示字符A // OLED_ShowString(1, 3, "HelloWorld!"); //1行3列显示字符串HelloWorld! // OLED_ShowSignedNum(2, 7, -66, 2); //2行7列显示有符号十进制数字-66,长度为2 // OLED_ShowHexNum(3, 1, 0xAA55, 4); //3行1列显示十六进制数字0xA5A5,长度为4 // OLED_ShowBinNum(4, 1, 0xAA55, 16); //4行1列显示二进制数字0xA5A5,长度为16 OLED_ShowNum(2, 1, cnt, 5); cnt = add(cnt,1); vTaskDelay(50); } }

FreeRTOS菜鸟入门系列_时光の尘的博客-CSDN博客

FreeRTOS实战系列_时光の尘的博客-CSDN博客

http://www.jsqmd.com/news/754678/

相关文章:

  • Flir Blackfly S多相机同步避坑指南:从SpinView配置到常见故障排查
  • RP2040 pHAT开发板:双模式微控制器与树莓派扩展板
  • YOLOv11户外徒步场景背包目标检测数据集-715张-backpack-1_6
  • 转载--AI Agent 架构设计:人和 Agent 的边界在哪里(OpenClaw、Claude Code、Hermes Agent 对比)
  • AI编程工具包深度解析:Cursor与Claude协同的工程化实践
  • 从概念到上线:在快马平台实战构建你的个人财务分析超级技能仪表盘
  • 手把手教你用MediaRecorder实现Android通话旁路录音(附完整代码与避坑清单)
  • 深入解析Auto-Code-Executor:声明式任务编排框架的设计与实战
  • 【多无人机动态避障路径规划】基于杜鹃鸟优化算法的多无人机三维协同路径规划方法(Matlab代码实现)
  • C语言(5)
  • Cursor编辑器资源宝库:主题插件与AI提示词全攻略
  • 初创公司如何借助 Taotoken 降低大模型 API 的接入与试用门槛
  • 基于Claude API的智能体服务器框架:工程化AI应用开发实践
  • 毕业季论文救星:百考通AI一站式解决查重与降重难题
  • Lemonade:开源本地AI服务器,打造私有化AI工作站
  • Java Spring Security 如何防止 JWT 密钥泄露导致签名伪造?
  • Rank-GRPO:强化学习优化对话推荐系统的新框架
  • 【LeetCode 刷题笔记】34. 在排序数组中查找元素的第一个和最后一个位置 | 二分查找经典刷题题解
  • RooMolt:基于最小描述长度与原子化MCP的AI自动化工作流实践
  • 通过动态规划优化插电式混合动力电动汽车 (PHEV) 能源管理(Matlab、Simulink代码实现)
  • 别再只调PWM了!STM32/CH32定时器的单脉冲模式,在电机刹车和精准开关上的妙用
  • Windows音频设备一键切换神器:voicemode命令行工具详解
  • ROCKET模型压缩技术:校准引导的动态剪枝与量化
  • 【RK3506实战-01】 BootLoader 全流程与实战优化
  • 3D场景理解与开放词汇检测技术解析
  • ARM汇编LDR指令详解:寄存器相对寻址与优化技巧
  • Kubernetes部署策略实战:从滚动更新到金丝雀发布的完整指南
  • Happy Island Designer终极指南:5步打造你的梦想岛屿规划
  • 4-bit/cell NAND技术:存储密度革命与工程实践
  • 开源AI模型部署与可解释性实践:CentminMod环境下的OpenClaw全栈指南