这里写自定义目录标题
1.前言
本文主要讲解arm汇编编程的基础知识,涉及到的例程都可以在如下链接进行在线调试:https://cpulator.01xz.net/?sys=arm
2.ARM汇编指令
2.1 跳转指令
跳转指令主要包含以下几条指令:B、BL、BX、BLX 。
(1) B指令
指令格式:B{Condition} address
B指令立即跳转到指定的地址
.global _start
_start:
mov r0, #100
mov r0, #100
mov r0, #100
B 0x18
mov r0, #200
mov r0, #300
mov r0, #400
mov r0, #500
mov r0, #600
mov r0, #700
在汇编调试窗口打几个断点,运行后会发现,执行B 0x18指令后,程序直接会跳到0x18地址运行,即运行 mov r0, #400 这一行
不过一般情况下,我们写代码的时候不会直接在B指令后面写地址,而是用一个label去代替,因为直接写地址的话,不好维护,一般我们会使用下面的方法:
B label
label代表一个标签,名称可以随便取,比如:
.global _start
_start:
mov r0, #100
mov r1, #200
B sum
mov r0, #200
mov r0, #300
sum:
add r2,r0,r1
(2) BL指令
指令格式 BL{Condition} address
BL指令与B指令的区别就在于在执行跳转之前,arm会将PC寄存器中的值(实际上是bl指令后面紧跟的下一条指令的地址)保存在链接寄存器lr(即R14)中(此处不考虑ARM的三级流水线,否则就是保存的PC-4);
和B指令一样我们也不会直接在BL指令后面直接写地址,而是使用标签:
BL label
.global _start
_start:
mov r0, #100
mov r1, #200
BL sum
mov r0, #200
mov r0, #300
sum:
add r2,r0,r1
mov pc,lr
分别在bl指令和sum标签打一个断点调试,会发现当程序执行完bl指令后,跳到sum标签执行的时候,lr的值变为了0x0c,我们可以很容易的看到bl指令所在的地址是0x08, 0x08+4 = 0x0c,即arm把bl指令后面的一条指令的地址保存在了lr寄存器当中;
在此例程中,在sum标签的最后,我们使用mov pc, lr 将lr保存的地址恢复到pc寄存器中,那么当sum标签里面的程序执行完后,程序会跳转到bl指令紧跟的后一条指令,即mov r0, #200, 这就相当于我们调用了一个函数,然后被调用函数执行完后,我们返回,继续执行后面的程序,这就相当于我们完成了一次函数调用.
(3) BLX指令