先看个例子:

void test2(int a,int b,int c)

int k=a,j=b,m=c;

}


GCC反汇编:
00000064 <test2>:
mov      ip, sp                   //IP=SP;保存SP
stmdb    sp!, {fp, ip, lr, pc}    //先对SP减4,再对fp,ip,lr,pc压栈。---------1
sub      fp, ip, #4       ; 0x4    //fp=ip-4;此时fp指向栈里面的“fp”
sub      sp, sp, #24      ; 0x18  //分配空间
str      r0, [fp, #-28]           //
str      r1, [fp, #-32]           //
str      r2, [fp, #-36]           //参数压栈
ldr      r3, [fp, #-28]           //
str      r3, [fp, #-24]           //
ldr      r3, [fp, #-32]           //
str      r3, [fp, #-20]           //
ldr      r3, [fp, #-36]           //
str      r3, [fp, #-16]           //
sub      sp, fp, #12      ; 0xc    //sp=fp-12;此时sp指向栈里面的lr
ldmia    sp, {fp, sp, pc}         //弹栈pc=lr,sp=ip,fp=fp。然后地址加4---------1


汇编基础:
stmdb    sp!, {fp, ip, lr, pc} //sp=sp-4,sp=pc;先压PC
                               //sp=sp-4,sp=lr;再压lr
                               //sp=sp-4,sp=ip;再压ip
                               //sp=sp-4,sp=fp;再压fp
ldmia    sp, {fp, sp, pc}       //和stmdb成对使用,
                               //fp=sp,sp=sp+4;先弹fp
                               //sp=sp,sp=sp+4;先弹sp,此处的弹出不会影响sp,因为ldmia是一个机器周期执行完的。
                               //pc=sp,sp=sp+4;先弹pc
LDRH           R0, [R13, #0xC] //加载无符号半字数据,即低16位
LDRB           R0, [R13, #0x4] //加载一字节数据,即低8位。

注意:R11=fp;R12=ip;R13=SP;R14=LR;R15=PC;R0,R1,R2用于传递参数和存放函数返回值。
注意;低地址的寄存器被压入低地址内存中,也就是说如果向下增长,高地址寄存器先压,向上增长测试低地址先压。 
注意:根据“ARM-thumb 过程调用标准”:
1,  r0-r3 用作传入函数参数,传出函数返回值。在子程序调用之间,可以将 r0-r3 用于任何用途。被调用函数在返回之前不必恢复 r0-r3。---如果调用函数需要再次使用 r0-r3 的内容,则它必须保留这些内容。
2, r4-r11 被用来存放函数的局部变量。如果被调用函数使用了这些寄存器,它在返回之前必须恢复这些寄存器的值。
3, r12 是内部调用暂时寄存器 ip。它在过程链接胶合代码(例如,交互操作胶合代码)中用于此角色。在过程调用之间,可以将它用于任何用途。被调用函数在返回之前不必恢复 r12。

4,寄存器 r13 是栈指针 sp。它不能用于任何其它用途。sp 中存放的值在退出被调用函数时必须与进入时的值相同。
5,寄存器 r14 是链接寄存器 lr。如果您保存了返回地址,则可以在调用之间将 r14 用于其它用途,程序返回时要恢复

6,寄存器 r15 是程序计数器 PC。它不能用于任何其它用途。

7,在中断程序中,所有的寄存器都必须保护,编译器会自动保护R4~R11,所以一般你自己只要在程序的开头
sub lr,lr,#4
stmfd sp!,{r0-r3,r12,lr};保护R0~R3,R12,LR就可以了,除非你用汇编人为的去改变R4~R11的值。(具体去看UCOS os_cpu_a.S中的IRQ中断的代码)


补充:

寄存器名字
Reg #  APCS   意义
R0 a1 工作寄存器
R1 a2 "
R2 a3 "
R3 a4 "
R4 v1 必须保护
R5 v2 "
R6 v3 "
R7 v4 "
R8 v5 "
R9 v6 "
R10 sl 栈限制
R11 fp 桢指针
R12 ip 
R13 sp 栈指针
R14 lr 连接寄存器
R15 pc 程序计数器

回溯结构

寄存器 fp (桢指针)应当是零或者是指向栈回溯结构的列表中的最后一个结构,提供了一种追溯程序的方式,来反向跟踪调用的函数。

回溯结构是:

地址高端

保存代码指针         [fp]          fp 指向这里

返回 lr 值           [fp, #-4]

返回 sp 值           [fp, #-8]

返回 fp 值           [fp, #-12]  指向下一个结构

[保存的 sl]

[保存的 v6]

[保存的 v5]

[保存的 v4]

[保存的 v3]

[保存的 v2]

[保存的 v1]

[保存的 a4]

[保存的 a3]

[保存的 a2]

[保存的 a1]

[保存的 f7]                           三个字

[保存的 f6]                           三个字

[保存的 f5]                           三个字

[保存的 f4]                           三个字

pc :总是包含下一个要被执行的指令的位置。 
lr   :(总是)包含着退出时要装载到 pc 中的值。在 26-bit 位代码中它还包含着 PSR。 
sp  :指向当前的栈块(chunk)限制,或它的上面。这是用于复制临时数据、寄存器和类似的东西到其中的地方。在 RISC OS 下,你有可选择的至少 256 字节来扩展它。 
fp 要么是零,要么指向回溯结构的最当前的部分。

寄存器r0-r15含义相关推荐

  1. Cortex-A7 MPCore 架构详细介绍(九种运行模式、内核寄存器组R0~R15,有特定的名字和功能)

    目录 0.ARM架构的历史简介 1.Cortex-A7 MPCore(即多核) 简介 2.Cortex-A 处理器九种运行模式 3.Cortex-A 寄存器组(内核寄存器) 3.1通用寄存器 3.1. ...

  2. Cortex-M3 R0~R15寄存器组

    [R0~R12通用寄存器] R0~R12都是32位通用寄存器,用于数据操作.其中: R0~R7为低组寄存器,所有的指令都可以访问. R8~R12为高组寄存器,只有32位Thumb2指令和很少的16位T ...

  3. 寄存器(R0~R16)以及从SysTick系统时钟理解RTOS移植初始化

    移植系统最重要的细节之一就是配置系统时钟 第一次玩RT-Thread,发现同样的程序逻辑,测试现象不一样,从现象很明显看出来是时钟频率配置不一样. 由于之前玩STM32几乎没有关注过系统时钟的初始化, ...

  4. [计组]寄存器的基本含义

    寄存器基本含义: 寄存器是CPU内部用来存放数据的一些小型存储区域,用来暂时存放参与运算的数据和运算结果.其实寄存器就是一种常用的时序逻辑电路,但这种时序逻辑电路只包含存储电路.寄存器的存储电路是由锁 ...

  5. 51单片机的工作寄存器R0~R7位于内部RAM什么位置

    转自:https://zhidao.baidu.com/question/1495509806096353459.html 51单片机的工作寄存器一共有32个,为RAM中的00H--1FH单元,分为4 ...

  6. 8086状态标志寄存器的英文含义(调试用

    标志位 全称 名称 =1 中文解释 =0 CF Carry Flag 进位标志 CY Carry/进位 NC PF Parity Flag 奇偶标志 PE Parity Event/偶 PO AF A ...

  7. ARM 寄存器 详解

    From( ARM 寄存器详解 ):https://blog.csdn.net/sandeldeng/article/details/52954781 ARM 汇编基础教程:2.数据类型和寄存器:ht ...

  8. 主线剧情01-ARM-IMX6ULL基础学习记录

    ARM & i.MX6ULL 基础学习记录 编辑整理 by Staok 本文大部分内容摘自"100ask imx6ull"开发板的配套资料(如<IMX6ULL裸机开发 ...

  9. ARM CMSIS DAP源码分析

    转ARM CMSIS DAP源码分析(一)_穿透灵魂的鼓点的博客-CSDN博客 结合ARM文档ADIv5,分析一下ARM提供的CMSIS DAP的开源代码,写点个人心得. 1.USB的整个传输有2个全 ...

  10. 计算机三级之嵌入式系统学习笔记1

    嵌入式系统的体系结构按指令集可以分为两大类:复杂指令集结构(CISC)和精简指令集结构(RISC) 嵌入式体系结构按存储机制分为冯诺依曼结构和哈佛结构 冯诺依曼结构中程序和指令并存,共用一条总线:而哈 ...

最新文章

  1. Java 数组中找最大值和最小值
  2. css 文字重叠_学习过CSS,那你知道BFC是什么吗?
  3. Educational Codeforces Round 40 (Rated for Div. 2)
  4. Heka:Go编写,来自Mozilla,高效、灵活的插件式数据挖掘工具(转)
  5. 链表(Linked List)之环形链表
  6. 1039. Course List for Student (25)
  7. r语言t检验输出检验统计量_如何进行统计分析
  8. 【运动学】基于matlab计步【含Matlab源码 524期】
  9. Win10系统高分辨率缩放时应用程序字体模糊
  10. 计算机表格列宽怎么设置,excel自动调整列宽在哪?excel中怎么自动调整各行宽度...
  11. win10下载c语言软件下载,Win tc win10
  12. CG CTF RE Py交易
  13. SAP 标准成本、计划成本、目标成本、实际成本计算公式
  14. LeCun:赋予机器 “常识” ,重新设计神经网络将是AI 研究重点
  15. 【高精浮点】关于long double的使用方法
  16. Java设计登录界面——GUI
  17. 【报告分享】中国消费者洞察报告-领航前所未有(附下载)
  18. 京东 API ,按图搜索京东商品(拍立淘)
  19. Hadamard积的介绍
  20. neuoj Blurred Pictures(小思维题

热门文章

  1. 【Liunx操作】一文解决Liunx命令使用
  2. “网银大盗”变种攻击六大网上银行
  3. android游戏和ios游戏哪个多,安卓手游与苹果手游哪个土豪多,为什么
  4. 安全合规--52--安全合规审计平台bombus-2.0部署实践
  5. 180220——折腾:U盘作启动盘安装win 7
  6. (三十六)Vue解决Ajax跨域问题
  7. 如何彻底删除微信聊天记录?这个方法简单管用
  8. C语言自我实现itoa函数
  9. linux查看端口占用
  10. DATEPART 使用方法