c语言中简单的指令集,从最简单的例子,带你轻松学习ARM指令集

首先编辑一个最简单的函数,包含变量分配及初始化:test1.c

1.#include

2.

3.voidmain()

4.{

5.intd=4;

6.}

然后编译:arm-linux-gnueabihf-gcc test.c -o test1

b00be00072672c5874ff1a29df666697.png

然后看看汇编代码:arm-linux-gnueabihf-objdump -D test1;每一句的含义我已经给出详细注释。

首先对输出的文件格式说明,对于如下的输出,左边是程序地址(各种函数地址等等),第二列是指令码的十六进制表示也俗称机器码,剩下的就是给人类看的指令助记符号,举例举例:

835c: b480 push {r7}

这里,835c是main函数的地址,b480是机器码表示的指令,push {r7}就是给我们人类看的了。下面看test1的输出:

0000835c :

程序用到了r7寄存器,所以需要保护以免破坏之前的数据

1 835c: b480 push {r7}

堆栈向下增长栈用的不多,只需要12个字节就够用了: int d需要4个,多出来的8个没有使用

2 835e: b083 sub sp, #12

因为r0-r7是通用寄存器,可以使劲用,堆栈寄存器sp只有没办法的时候才使用。只好用r7 = sp + 0这种笨办法

3 8360: af00 add r7, sp, #0

参与int d = 4这条语句的是r3,这是通用寄存器,spec定义大家都可以用,不需要保护

4 8362: f04f 0304 mov.w r3, #4

把4存储到sp+4所指定的栈里,c语言描述:*(sp + 4) = 4;留给大家思考:为何不顶着sp放置--即*(sp+0)=4?

5 8366: 607b str r3, [r7, #4]

还记得第二条:sub sp, #12吗?此句和下一句是为从堆栈里恢复原来的r7--pop {r7},做准备;

6 8368: f107 070c add.w r7, r7, #12

r7已经是原来的sp了

7 836c: 46bd mov sp, r7

弹出sp指向的内存数据给r7,c语言:r7 = *sp;

8 836e: bc80 pop {r7}

没有调用子函数,即没有使用lr寄存器,所以不需要push lr。跳转到lr地址--进入main函数的下一条地址

9 8370: 4770 bx lr 10 8372: bf00 nop

我们再编辑一个稍微增加一个变量:test2.c

[cpp]view plaincopy

1.#include

2.

3.voidmain()

4.{

5.intd=4;

6.charb=2;

7.}

0000835c :

1 835c: b480 push {r7}2 835e: b083 sub sp, #123 8360: af00 add r7, sp, #0

4 8362: f04f 0304 mov.w r3, #4

把4存储到sp+0所指定的栈里,c语言描述:*(sp + 0) = 4,注意与上一个例子的区别,这里是顶着stack存放,为什么?

5 8366: 603b str r3, [r7, #0]

6 8368: f04f 0302 mov.w r3, #2

为何要这么任性的存放变量b?

7 836c: 71fb strb r3, [r7, #7]8 836e: f107 070c add.w r7, r7, #129 8372: 46bd mov sp, r710 8374: bc80 pop {r7}11 8376: 4770 bx lr

栈里的数据是这样的,b和d中间隔着好几条街呢:)

dbc690e11acbd6fe8e5dcf71a3add037.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值