文档章节

arm汇编基础stmdb和ldmia(转)

For_MGP
 For_MGP
发布于 2014/10/08 11:21
字数 1178
阅读 263
收藏 0

先看个例子:

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 要么是零,要么指向回溯结构的最当前的部分。

本文转载自:http://general.blog.51cto.com/927298/657803

For_MGP
粉丝 2
博文 66
码字总数 20576
作品 0
深圳
程序员
私信 提问
嵌入式开发笔记(八) ARM汇编指令集基础4

ldm/stm与栈的处理 1.为什么需要多寄存器访问指令? ldr/str每周期只能访问4字节内存,如果需要批量读取、写入内存时太慢,解决方案是 stm/ldm ldm(load register mutiple) stm(store regis...

每天在改变
2016/12/28
71
0
嵌入式开发笔记(五) ARM汇编指令集基础1

1.指令与伪指令的概念: (1)(汇编)指令是CPU机器指令的助记符,经过编译后会得到一串10组成的机器 码,可以由CPU读取执行。 (2)(汇编)伪指令本质上不是指令(只是和指令一起写在代码...

每天在改变
2016/12/25
108
0
嵌入式开发笔记(六) ARM汇编指令集基础2

ARM汇编指令集基础2 基本内容回顾: 1. ldr, str, 2. mov r0,r1 mov r0, #0XFF00 mov r0 ,r1,lsl#3 ld r0, [r1] ld r0,[r1,#4] ldmia r0!,{r1-r7,r12} stmfd sp!,{r2-r7,r12} beq flag flag......

每天在改变
2016/12/25
114
0
05-ARM体系结构与常用汇编指令

一、ARM体系结构  ARM(Advanced RISC Machines) ,既可以认为是一个公司的名字,也可以认为是对一类微处理器的通称,还可以认为是一种技术的名字。ARM 处理器是一种低功耗高性能的 32 位R...

梦想成大牛
2018/01/05
0
0
编写高效简洁的C语言代码,是许多软件工程师追求的目标。各位有什么高招呢?请不吝赐教。。。

以下是我个人的一些体会和经验,不对的地方请各位指教。同时希望各位拿出自己的干货分享一下经验 第1招:以空间换时间   计算机程序中最大的矛盾是空间和时间的矛盾,那么,从这个角度出发...

刘学炜
2012/04/17
758
4

没有更多内容

加载失败,请刷新页面

加载更多

nginx学习笔记

中间件位于客户机/ 服务器的操作系统之上,管理计算机资源和网络通讯。 是连接两个独立应用程序或独立系统的软件。 web请求通过中间件可以直接调用操作系统,也可以经过中间件把请求分发到多...

码农实战
今天
5
0
Spring Security 实战干货:玩转自定义登录

1. 前言 前面的关于 Spring Security 相关的文章只是一个预热。为了接下来更好的实战,如果你错过了请从 Spring Security 实战系列 开始。安全访问的第一步就是认证(Authentication),认证...

码农小胖哥
今天
11
0
JAVA 实现雪花算法生成唯一订单号工具类

import lombok.SneakyThrows;import lombok.extern.slf4j.Slf4j;import java.util.Calendar;/** * Default distributed primary key generator. * * <p> * Use snowflake......

huangkejie
昨天
12
0
PhotoShop 色调:RGB/CMYK 颜色模式

一·、 RGB : 三原色:红绿蓝 1.通道:通道中的红绿蓝通道分别对应的是红绿蓝三种原色(RGB)的显示范围 1.差值模式能模拟三种原色叠加之后的效果 2.添加-颜色曲线:调整图像RGB颜色----R色增强...

东方墨天
昨天
11
1
将博客搬至CSDN

将博客搬至CSDN

算法与编程之美
昨天
13
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部