1、ARM指令集讲解ARM指令和指令系统: 指令是指示计算机某种操作的命令,指令的集合称为指令系统。指令系统的功能强弱很大程度上决定了这类计算机智能的高低,它集中地反应了微处理器的硬件功能和属性。 ARM指令在机器中的表示格式是用32位的二进制数表示。如ARM中有一条指令为 ADDEQS R0,R1,#8; 其二进制代码形式为: 3128 | 2725 | 2421 | 20 | 1916 | 1512 | 110 0000 | 001 | 0100 | 1 | 0001 | 0000 | 0000 0000 1000 cond | opcode | Rn | Rd | Op2 ARM指令格式一般
2、如下: s, 格式中的内容是必不可少的, 中的内容可忽略 表示操作码。如ADD表示算术加法 表示指令执行的条件域。如EQ、NE等,缺省为AL。 S 决定指令的执行结果是否影响CPSR的值,使用该后缀则指令执行结果影响CPSR的值,否则不影响 表示目的寄存器 表示第一个操作数,为寄存器 表示第二个操作数,可以是立即数。寄存器和寄存器移位操作数ARM指令后缀:S、! S后缀:指令中使用S后缀时,指令执行后程序状态寄存器的条件标志位将被刷新,不使用S后缀时,指令执行后程序状态寄存器的条件标志将不会发生变化。S后缀常用于对条件进行测试,如是否有溢出,是否进位等,根据这些变化,就可以进行一些判断,如是否
3、大于,相等,从而影响指令执行的顺序。 !后缀:如果指令地址表达式中不含!后缀,则基址寄存器中的地址值不会发生变化。加上此后缀后,基址寄存器中的值(指令执行后) = 指令执行前的值 + 地址偏移量 (1)!后缀必须紧跟在地址表达式后面,而地址表达式要有明确的地址偏移量 (2)!后缀不能用于R15(PC)的后面 (3)当用在单个地址寄存器后面时,必须确信这个寄存器有隐性的偏移量,例如“STMDB R1!,R3,R5,R7”。此时地址基址寄存器R1的隐性偏移量为4(一条指令占32位,即4个字节)指令的条件码:31-28位4个字节存储,共16个条件码 条件码 助记符后缀 标志 含义 0000 EQ Z
4、置位 相等 0001 NE Z清零 不相等 0010 CS C置位 无符号数大于或等于 0011 CC C清零 无符号数小于 0100 MI N置位 负数 0101 PL N清零 正数或零 0110 VS V置位 溢出 0111 VC V清零 未溢出 1000 HI C置位 Z清零 无符号数大于 1001 LS C清零 Z置位 无符号数小于或等于 1010 GE N等于V 带符号数大于或等于 1011 LT N不等于V 带符号数小于 1100 GT Z清零且(N等于V) 带符号数大于 1101 LE Z置位或(N不等于V) 带符号数小于或等于 1110 AL 忽略 无条件执行ARM指令分类:六
5、大类 ARM指令集可以分为数据处理指令,数据加载指令和存储指令,分支指令,程序状态寄存器(PSR)处理指令,协处理器指令和异常产生指令六大类。ARM指令的寻址方式:8类 ARM指令的寻址方式一般可以分为8类:立即数寻址,寄存器寻址,寄存器间接寻址,寄存器移位寻址,基址变址寻址,多寄存器寻址,相对寻址,堆栈寻址等 举例: MOV R0,#15 ;立即数15放入寄存器R0中 ADD R0,R1,R2 ;R0 = R1+R2 LDR R0,R4 ;R0 = R4(R4中存放的是一个指针变量, 表示取改地址值指向的内容) ADD R0,R1,R2,LSL #1 ;R0 = R1+R2(R2左移一位后的
6、值) MOV R0,R1,LSL R3 ;R0=R1(R1左移R3位后) LDR R0,R1,#4 ;R0=R1+4 LDR R0,R1,#4! ;R0=R1+4,R1= R1+4。同时更新基址 LDR R0,R1,#4 ;R0=R1,R1= R1+4 LDR R0,R1,R2 ;R0=R1+R2 LDMIA R0!,R1 - R4 ;R1=R0、R1=R0+4、R1=R0+8、R110,则执行ADDGT指令,将R0加5 CMN R0,R1 ;R0 - (-R1),反值比较,影响CPSR标志位 CMN R0,#10 ;R0 - (-10),反值比较,影响CPSR标志位 TST R1,#3 ;检
7、查R1中第0位和第1位是否为1,根据结果更新条件标志位 TEQ R1,R2 ;将寄存器R1的值与寄存器R2的值进行按位异或, ;并根据结果设置CPSR的标志位 7、乘法指令:MUL、MLA、SMULL、SMLAL、UMULL、UMLAL MUL R0,R1,R2 ;R1和R2相乘的结果发送到R0 MULS R0,R1,R2 ;R1和R2相乘的结果发送到R0,同时设置CPSR的相关条件标志位 MLA R0,R1,R2,R3 ;R1和R2相乘的结果再加上R3后发送到R0 MLAS R0,R1,R2,R3 ;R1和R2相乘的结果再加上R3后发送到R0,更新CPSR标志位 SMULL R0,R1,R2
8、,R3 ;R2和R3相乘的结果的低32位放在R0,高32位放在R1 SMLAL R0,R1,R2,R3 ;R2和R3相乘的结果的低32位加上R0后放在R0, ;高32位加上R1后放在R1 UMULL R0,R1,R2,R3 ;无符号数相乘,结果与SMULL类似 UMLAL R0,R1,R2,R3 ;无符号数乘加,结果与SMLAL类似ARM数据加载和存储指令: 1、数据加载和存储的方向。寄存器到存储器方向:Store;从存储器到寄存器方向:Load 数据加载和存储指令共有三种类型:单寄存器加载和存储指令,多寄存器加载和存储指令 和 交换指令 2、数据加载与存储器指令寻址 LDR R5,R6,#0
9、x08 ;R6寄存器加0x08的和的地址值内的数据传送到R5 STR R6,R7,#-0x08 ;R6寄存器的数据传送到R7存储的地址值指向的存储空间, ;同时更新R7寄存器的内容为R7-0x08 LDR R5,R6,R3 ;R6寄存器加R3的和的地址值内的数据传送到R5 STR R6,R7,-R8 ;R6寄存器的数据传送到R7存储的地址值指向的存储空间, ;同时更新R7寄存器的内容为R7-R8 LDR R3,R2,R4,LSL #2 ;R3 = R2 + R4(R4左移两位) LDR R3,R2,-R4,LSR #3 ;R3 = R2 ,R2 = R2-R4(R4右移三位) LDR R4,S
10、TART ;将标号START标定的空间的数据加载到R4中 3、地址索引:前索引、自动索引、后索引 1】前索引:前索引也称为前变址,这种索引是在指令执行前把偏移量和基址相加减,得到的值作为变量的地址。如: LDR R5,R6,#0x04 STR R0,R5,-R8 2】自动索引:自动索引也称为自动变址,有时为了修改基址寄存器的内容,使之指向数据传送地址,可使用这种方法自动修改基址寄存器,如: LDR R5,R6,#0x04! 3】后索引:后索引也被称为后变址,后索引就是用基址寄存器的地址值寻址,找出操作数进行操作,操作完成后,再把地址偏移量和基址相加/减,结果送到基址寄存器,作为下一次寻址的基址
11、。如: LDR R5,R6,#0x04 STR R6,R7,#-0x08 4、单寄存器加载和存储指令:LDR/STR、LDRB/STRB、LDRH/STRH、LDRSB/LDRSH 1】字数据加载/存储指令格式: 3128 | 2726 |2524 23 22 2120| 1916 | 1512 | 110 cond | 01 | I P U B W L | Rn | Rd | Op2 cond:指令执行的条件编码 I、P、U、W:用于区别不同的地址模式(偏移量)。 I为0时,偏移量为12位立即数;I为1时,偏移量为移位寄存器移位 P表示前/后索引 U表示加/减 W表示回写 L:L为1时表示加
12、载,L为0时表示存储 B:B为1表示字节访问,B为0表示字访问 Rd:源/目标寄存器 Rn:基址寄存器 Op2:表示偏移量是一个12位的无符号二进制数,与Rn一起构成地址addr 2】存储器寄存器 LDR/STR LDR指令用于从存储器中间一个32位的字数据加载到目的寄存器Rd中。该指令通常用于从存储器中读取32位的字数据到通用寄存器,然后对数据进行处理。当程序计数器PC作为目的寄存器时,指令从存储器中读取的字数据被当做目的地址,从而实现程序流程的跳转。 LDR R4,START ;将存储地址为START的字数据读入R4 STR R5,DATA1 ;将R5存入存储地址为DATA1中 LDR R
13、0,R1 ;将存储器地址为R1的字数据读入寄存器R0 LDR R0,R1,R2 ;将存储器地址为R1+R2的字数据读入寄存器R0 LDR R0,R1,#8 ;将寄存器R1+8的内容读入寄存器R0 LDR R0,R1,R2,LSL #2 ;将R1+R2*4的字数据读入寄存器R0 STR R0,R1,R2! ;将R0字数据存入存储器地址为R1+R2的存储单元中, 并将新地址R1+R2写入R1 STR R0,R1,#8! ;将R0字数据存入存储器地址为R1+8的存储单元中, 并将新地址R1+8写入R1 STR R0,R1,R2,LSL #2! ;将R0字数据存入地址为R1+R2*4的存储单元中, 并
14、将新地址R1+R2*4写入R1 LDR R0,R1,#8 ;将存储器地址为R1的字数据读入寄存器R0, 并将新地址R1+8写入R1 LDR R0,R1,R2 ;将存储器地址为R1的字数据读入寄存器R0, 并将新地址R1+R2写入R1 LDR R0,R1,R2,LSL #2 ;将存储器地址为R1的字数据读入寄存器R0, 并将新地址R1+R2*4写入R1 【备注】注意事项: a、立即数绝对值不大于4095的数值,可使用带符号数,即在-4095 +4095之间。(4096D = 1000H) b、语句的标号不能指向程序存储器的程序存储区,而是指向程序存储器的数据存储区或数据存储器的数据存储区。另外指
15、向的区域是可修改的。例如,在用户模式下,有些存储区是不能访问的或是只读的。 c、字传送时,偏移量必须保证偏移的结果能够使地址对齐。 d、使用寄存器移位的方法计算偏移量时。移位的位数不能超过规定的数值,而且不能用寄存器表示移位的位数。各类移位指令的移位位数规定如下: ASR #n:算术右移(1n32) LSL #n:逻辑左移(0n31) LSR #n:逻辑右移(1n32) ROR #n:循环右移(1n31) e、R15作为基址寄存器Rn时,不可以使用回写功能,即使用后缀“!”,另外,R15不可作为偏移寄存器使用。 5、字节数据加载/存储指令:LDRB/STRB LDRB指令用于从存储器中将一个8
16、位字节的数据加载到目的寄存器,同时将寄存器的高24位清零。该指令通常用于从存储器中读取8位的字节数据到通用寄存器,然后对数据进行处理。当程序计数器PC作为目的寄存器时,指令从存储器读取的数据被当做目的地,从而可以实现程序流程的跳转 STRB指令用于从源寄存器中将一个8位的字节数据存储到存储器中,该字节数据为源寄存器的低8位,STRB指令和LDRB指令的区别在于数据的传送方向。 LDRB R0,R1 ;将存储器地址为R1的字节数据读入寄存器R0, 并将R0的高24位清零。 LDRB R0,R1,#8 ;将存储器地址为R1+8的字节数据读入寄存器R0, 并将R0的高24位清零。 STRB R0,R
17、1 ;将寄存器R0中的字节数据写入以R1为地址的存储器中。 STRB R0,R1,#8 ;将寄存器R0中的字节数据写入以R1+8为地址的存储器中。 6、LDRH/STRH 半字数据加载/存储指令 3128| 2725|24 23 22 2120| 1916|1512|118 | 7 6 5 4 | 30 cond | 000 | P U I W L| Rn | Rd |addr_H |1 S H 1 |addr_L cond:指令执行的条件编码 I、P、U、W:用于区别不同的地址模式(偏移量)。I为0时,偏移量为8位立即数,I为1时,偏移量为寄存器移位。P表示前/后变址,U表示加/减,W表示回
18、写。 L:L为1表示加载,L为0表示存储。 S:用于区别有符号访问(S为1)和无符号访问(S为0) H:用于区别半字访问(H为1)或字节访问(H为0) Rd:源/目标寄存器 Rn:基址寄存器 addr H / addr I:表示偏移量,I为0时,偏移量为8位立即数由addr H和addr I组成; I为1时,偏移量为寄存器移位addr H为0,addr L表示寄存器编号 LDR指令用于从寄存器中间一个16位的半字数据加载到目的寄存器Rd中,同时将寄存器的高16位清零,该指令通常用于从存储器中读取16位的半字数据到通用寄存器,然后对数据进行处理。当程序计数器PC作为目的寄存器时,指令从存储器中读
19、取的数据被当做目的地址,从而可以实现程序流程的跳转。 LDRH R0,R1 ;将存储器地址R1的半字数据读入寄存器R0, 并将R0的高16位清零 LDRH R0,R1,#8 ;将存储器地址为R1+8的半字数据读入寄存器R0, 并将R0的高16位清零 LDRH R0,R1,R2 ;将存储器地址为R1+R2的半字数据读入寄存器R0, 并将R0的高16位清零 STRH R0,R1 ;将寄存器R0中的半字数据写入以R1为地址的存储器中 使用半字加载/存储指令需要注意的事项: (1)必须半字地址对齐。 (2)对于R15的使用需要慎重,R15作为基址寄存器Rn时,不可以使用回写功能,不可使用R15作为目的
20、寄存器。 (3)立即数偏移使用的是8位无符号数。 (4)不能使用寄存器移位寻址 7、有符号数字节/半字加载指令:LDRSB / LDRSH LDRSB指令用于从存储器中间一个8位的字节数据加载到目的寄存器中,同时将寄存器的高24位设置为该字节数据的符号位的值,即将该8位字节数据进行符号位的扩展,生成32位数据;LDRSH指令用于从存储器中将一个16位的半字数据加载到目的寄存器Rd中,同时将寄存器的高16位设置为该字数据的符号位的值,即将该16位字数据进行符号位的扩展,生成32位数据。 LDRSB R0,R1,#4 ;将存储地址为R1+4的有符号字节数据读入R0, R0中的高24位设置为高字节数
21、据的符号位 LDRSH R6,R2,#2 ;将存储地址为R2+2的有符号半字数据读入R6, R6的高16位设置成该字节数据的符号位,R2=R2+2 8、多寄存器加载和存储指令:LDM / STM LDM指令用于从基址寄存器所指示的一片连续存储器中读取数据到寄存器列表所指示的多个寄存器中,内存单元的其实地址为基址寄存器Rn的值,各个寄存器有寄存器列表regs表示。该指令一般用于多个寄存器数据的出栈操作;STM指令用于将寄存器列表所指示的多个寄存器的值存入到由基址寄存器所指示的一片连续存储器中,内存单元的其实地址为基址寄存器Rn的值,各个寄存器由寄存器列表regs表示。指令的其它参数的用法和LDM
22、指令是相同的。该指令一般用于多个寄存器数据的进栈操作。 type类型。用于数据的存储和读取有一下几种情况: IA 每次传送后地址值加 IB 每次传送前地址值加 DA 每次传送后地址值减 DB 每次传送前地址值减 对于堆栈操作有如下几种情况: FD 满递减堆栈 ED 空递减堆栈 FA 满递增堆栈 EA 满递增堆栈 !为可选后缀,若选用该后缀,则当数据加载与存储完毕后,将最后的地址写入基址寄存器,否则基址寄存器的内容不改变。基址寄存器不允许为R15,寄存器列表可以为R0R15的任意组合。 为可选后缀,当治疗为LDM且寄存器列表中包含R15,选用该后缀时表示:除了正常数据加载和存储之外,还将SPSR复制到CPSR。同时,该后缀还表示传入或传出的是用户模式下的寄存器,而不是当前模式下的寄存器。 LDMIA R0!,R6-R8 ;R6 - R0,R7 - R0+4,R8 - R0+8,R0 - R0+12 LDMIB R0!,R6-R8 ;R6
copyright@ 2008-2022 冰豆网网站版权所有
经营许可证编号:鄂ICP备2022015515号-1