当前位置:文档之家› ARM指令集详解

ARM指令集详解

ARM指令集详解
ARM指令集详解

好东西,说得非常详细的。要做底层的可以看看

ARM指令集详解

发布: 2010-3-19 11:42 | 作者: Saiu | 来源: MCU嵌入式领域

ARM可以用两套指令集:ARM指令集和Thumb指令集。本文介绍ARM指令集。在介绍ARM指令集之前,先介绍指令的格式。

1 指令格式

(1)基本格式

{}{S} ,{,}

其中,<>内的项是必须的,{}内的项是可选的,如是指令助记符,是必须的,而{}为指令执行条件,是可选的,如果不写则使用默认条件AL(无条件执行)。

opcode 指令助记符,如LDR,STR 等

cond 执行条件,如EQ,NE 等

S 是否影响CPSR 寄存器的值,书写时影响CPSR,否则不影响

Rd 目标寄存器

Rn 第一个操作数的寄存器

operand2 第二个操作数

指令格式举例如下:

LDR R0,[R1] ;读取R1 地址上的存储器单元内容,执行条件AL

BEQ DATAEVEN ;跳转指令,执行条件EQ,即相等跳转到DATAEVEN

ADDS R1,R1,#1 ;加法指令,R1+1=R1 影响CPSR 寄存器,带有S

SUBNES R1,R1,#0xD;条件执行减法运算(NE),R1-0xD=>R1,影响CPSR 寄存器,带有S (2)第2个操作数

在ARM 指令中,灵活的使用第2个操作数能提高代码效率,第2个操作数的形式如下:

#immed_8r

常数表达式,该常数必须对应8 位位图,即常数是由一个8 位的常数循环移位偶数位得到。

合法常量

0x3FC、0、0xF0000000、200、0xF0000001等都是合法常量。

非法常量

0x1FE、511、0xFFFF、0x1010、0xF0000010等都是非法常量。

常数表达式应用举例如下:

MOV R0,#1 ;R0=1

AND R1,R2,#0x0F ;R2 与0x0F,结果保存在R1

LDR R0,[R1],#-4 ;读取R1 地址上的存储器单元内容,且R1=R1-4

Rm

寄存器方式,在寄存器方式下操作数即为寄存器的数值。

寄存器方式应用举例:

SUB R1,R1,R2 ;R1-R2=>R1

MOV PC,R0 ;PC=R0,程序跳转到指定地址

LDR R0,[R1],-R2 ;读取R1 地址上的存储器单元内容并存入R0,且R1=R1-R2

Rm, shift

寄存器移位方式。将寄存器的移位结果作为操作数,但RM 值保存不变,移位方法如下:

ASR #n 算术右移n 位(1≤n≤32)

LSL #n 逻辑左移n 位(1≤n≤31)

LSR #n 逻辑左移n 位(1≤n≤32)

ROR #n 循环右移n 位(1≤n≤31)

RRX 带扩展的循环右移1位

type Rs 其中,type 为ASR,LSL,和ROR 中的一种;Rs 偏移量寄存器,低8位有效,若其值大于或等于32,则第2 个操作数的结果为0(ASR、ROR例外)。

寄存器偏移方式应用举例:

ADD R1,R1,R1,LSL #3 ;R1=R1*9

SUB R1,R1,R2,LSR#2 ;R1=R1-R2*4

R15 为处理器的程序计数器PC,一般不要对其进行操作,而且有些指令是不允许使用R15,如UMULL 指令。

(3)条件码

使用指令条件码,可实现高效的逻辑操作,提高代码效率。表A-1给出条件码表。

表A-1 条件码表

对于Thumb指令集,只有B 指令具有条件码执行功能,此指令条件码同表A-?,但如果为无条件执行时,条件码助记符“AL”不能在指令中书写。

条件码应用举例如下:

比较两个值大小,并进行相应加1 处理,C 代码为:

if(a>b)a++ ;

else b++ ;

对应的ARM 指令如下。其中R0为a,R1为b。

CMP R0,R1 ; R0 与R1 比较

ADDHI R0,R0,#1 ; 若R0>R1,则R0=R0+1

ADDLS R1,R1,#1 ;若R0<=R1,则R1=R1+1

若两个条件均成立,则将这两个数值相加,C代码为:

If((a!=10)&&(b!=20)) a=a+b;

对应的ARM 指令如下,其中R0 为a,R1 为b。

CMP R0,#10 ; 比较R0 是否为10

CMPNE R1,#20 ; 若R0 不为10,则比较R1 是否20

ADDNE R0,R0,R1 ; 若R0 不为10 且R1 不为20,指令执行,R0=R0+R1

2 ARM 存储器访问指令

ARM 处理是加载/存储体系结构的典型的RISC 处理器,对存储器的访问只能使用加载和存储指令实现。ARM 的加载/存储指令是可以实现字、半字、无符/有符字节操作;批量加载/存储指令可实现一条指令加载/存储多个寄存器的内容,大大提高效率;SWP指令是一条寄存器和存储器内容交换的指令,可用于信号量操作等。ARM 处理器是冯?诺依曼存储结构,程序空间、RAM 空间及IO 映射空间统一编址,除对对RAM 操作以外,对外围IO、程序数据的访问均要通过加载/存储指令进行。表A-2给出ARM 存储访问指令表。

表A-2 ARM 存储访问指令表

LDR 和STR

加载/存储字和无符号字节指令。使用单一数据传送指令(STR 和LDR)来装载和存储单一字节或字的数据从/到内存。LDR 指令用于从内存中读取数据放入寄存器中;STR 指令用于将寄存器中的数据保存到内存。指令格式如下:

LDR{cond}{T} Rd,<地址>;加载指定地址上的数据(字),放入Rd 中

STR{cond}{T} Rd,<地址>;存储数据(字)到指定地址的存储单元,要存储的数据在Rd中

LDR{cond}B{T} Rd,<地址>;加载字节数据,放入Rd中,即Rd最低字节有效,高24位清零

STR{cond}B{T} Rd,<地址>;存储字节数据,要存储的数据在Rd,最低字节有效

其中,T 为可选后缀,若指令有T,那么即使处理器是在特权模式下,存储系统也将访问看成是处理器是在用户模式下。T 在用户模式下无效,不能与前索引偏移一起使用T。

LDR/STR 指令寻址是非常灵活的,由两部分组成,一部分为一个基址寄存器,可以为任一个通用寄存器,另一部分为一个地址偏移量。地址偏移量有以下3 种格式:

(1) 立即数。立即数可以是一个无符号数值,这个数据可以加到基址寄存器,也可以从基址寄存器中减去这个数值。指令举例如下:

LDR R1,[R0,#0x12] ;将R0+0x12 地址处的数据读出,保存到R1 中(R0 的值不变)

LDR R1,[R0,#-0x12];将R0-0x12 地址处的数据读出,保存到R1 中(R0 的值不变)

LDR R1,[R0] ;将R0 地址处的数据读出,保存到R1 中(零偏移)

(2)寄存器。寄存器中的数值可以加到基址寄存器,也可以从基址寄存器中减去这个数值。指令举例值。指令举例如下:

LDR R1,[R0,R2] ;将R0+R2 地址的数据计读出,保存到R1 中(R0 的值不变)

LDR R1,[R0,-R2] ;将R0-R2 地址处的数据计读出,保存到R1 中(R0 的值不变)

(3)寄存器及移位常数。寄存器移位后的值可以加到基址寄存器,也可以从基址寄存器中减去这个数值。指令举例如下:

LDR R1,[R0,R2,LSL #2] ;将R0+R2*4地址处的数据读出,保存到R1中(R0,R2的值不变)

LDR R1,[R0,-R2,LSL #2];将R0-R2*4地址处的数据计读出,保存到R1中(R0,R2的值不变) 从寻址方式的地址计算方法分,加载/存储指令有以下4 种形式:

(1)零偏移。Rn 的值作为传送数据的地址,即地址偏移量为0。指令举例如下:

LDR Rd,[Rn]

(2)前索引偏移。在数据传送之前,将偏移量加到Rn 中,其结果作为传送数据的存储地址。若使用后缀“!”,则结果写回到Rn 中,且Rn 值不允许为R15。指令举例如下:

LDR Rd,[Rn,#0x04]!

LDR Rd,[Rn,#-0x04]

(3)程序相对偏移。程序相对偏移是索引形式的另一个版本。汇编器由PC 寄存器计算偏移量,并将PC 寄存器作为Rn 生成前索引指令。不能使用后缀“!”。指令举例如下:

LDR Rd,label ;label 为程序标号,label 必须是在当前指令的±4KB 范围内

(4) 后索引偏移。Rn 的值用做传送数据的存储地址。在数据传送后,将偏移量与Rn相加,结果写回到Rn 中。Rn 不允许是R15。指令举例如下:

LDR Rd,[Rn],#0x04

地址对准--大多数情况下,必须保证用于32 位传送的地址是32 位对准的。

加载/存储字和无符号字节指令举例如下:

LDR R2,[R5] ;加载R5 指定地址上的数据(字),放入R2 中

STR R1,[R0,#0x04] ;将R1 的数据存储到R0+0x04 存储单元,R0 值不变

LDRB R3,[R2],#1 ;读取R2 地址上的一字节数据,并保存到R3 中,R2=R3+1

STRB R6,[R7] ;读R6 的数据保存到R7 指定的地址中,只存储一字节数据

加载/存储半字和带符号字节。这类LDR/STR 指令可能加载带符字节\加载带符号半字、加载/存储无符号半字。偏移量格式、寻址方式与加载/存储字和无符号字节指令相同。指令格式如下:

LDR{cond}SB Rd,<地址> ;加载指定地址上的数据(带符号字节),放入Rd 中

LDR{cond}SH Rd,<地址> ;加载指定地址上的数据(带符号字节),放入Rd 中

LDR{cond}H Rd,<地址> ;加载半字数据,放入Rd中,即Rd最低16 位有效,高16位清零

STR{cond}H Rd,<地址> ;存储半字数据,要存储的数据在Rd,最低16 位有效

说明:带符号位半字/字节加载是指带符号位加载扩展到32 位;无符号位半字加载是指零扩展到32 位。

地址对准--对半字传送的地址必须为偶数。非半字对准的半字加载将使Rd 内容不可靠,非半字对准的半字存储将使指定地址的2 字节存储内容不可靠。

加载/存储半字和带符号字节指令举例如下:

LDRSB R1[R0,R3] ;将R0+R3 地址上的字节数据读出到R1,高24 位用符号位扩展

LDRSH R1,[R9] ;将R9 地址上的半字数据读出到R1,高16 位用符号位扩展

LDRH R6,[R2],#2 ;将R2 地址上的半字数据读出到R6,高16 位用零扩展,R2=R2+1

SHRH R1,[R0,#2]!;将R1 的数据保存到R2+2 地址中,只存储低2 字节数据,R0=R0+2

LDR/STR 指令用于对内存变量的访问,内存缓冲区数据的访问、查表、外设的控制操作等等,若使用LDR 指令加载数据到PC 寄存器,则实现程序跳转功能,这样也就实现了程序散转。

变量的访问

NumCount EQU 0x40003000 ;定义变量NumCount

LDR R0,=NumCount ;使用LDR 伪指令装载NumCount 的地址到R0

LDR R1,[R0] ;取出变量值

ADD R1,R1,#1 ;NumCount=NumCount+1

STR R1,[R0] ;保存变量值

GPIO 设置

GPIO-BASE EQU 0Xe0028000 ;定义GPIO 寄存器的基地址

LDR R0,=GPIO-BASE

LDR R1,=0x00FFFF00 ;装载32 位立即数,即设置值

STR R1,[R0,#0x0C] ;IODIR=0x00FFFF00, IODIR 的地址为0xE002800C

MOV R1,#0x00F00000

STR R1,[R0,#0x04] ;IOSET=0x00F00000,IOSET 的地址为0xE0028004

程序散转

MOV R2,R2,LSL #2 ;功能号乘上4,以便查表

LDR PC,[PC,R2] ;查表取得对应功能子程序地址,并跳转

NOP

FUN-TAB DCD FUN-SUB0

DCD FUN-SUB1

DCD FUN-SUB2

LDM 和STM

批量加载/存储指令可以实现在一组寄存器和一块连续的内存单元之间传输数据。LDM 为加载多个寄存器,STM 为存储多个寄存器。允许一条指令传送16 个寄存器的任何子集或所有寄存器。指令格式如下:

LDM{cond}<模式> Rn{!},reglist{^}

STM{cond}<模式> Rn{!},reglist{^}

LDM /STM 的主要用途是现场保护、数据复制、参数传送等。其模式有8 种,如下所列:(前面4 种用于数据块的传输,后面4 种是堆栈操作)。

(1) IA:每次传送后地址加4

(2) IB:每次传送前地址加4

(3) DA:每次传送后地址减4

(4) DB:每次传送前地址减4

(5) FD:满递减堆栈

(6) ED:空递增堆栈

(7) FA:满递增堆栈

(8) EA:空递增堆栈

其中,寄存器Rn 为基址寄存器,装有传送数据的初始地址,Rn 不允许为R15;后缀“!”表示最后的地址写回到Rn 中;寄存器列表reglist 可包含多于一个寄存器或寄存器范围,使用“,”分开,如{R1,R2,R6-R9},寄存器排列由小到大排列;“^”后缀不允许在用户模式呈系统模式下使用,若在LDM 指令用寄存器列表中包含有PC 时使用,那么除了正常的多寄存器传送外,将SPSR 拷贝到CPSR 中,这可用于异常处理返回;使用“^”后缀进行数据传送且寄存器列表不包含PC 时,加载/存储的是用户模式的寄存器,而不是当前模式的寄存器。

地址对准――这些指令忽略地址的位[1:0]。

批量加载/存储指令举例如下:

LDMIA R0!,{R3-R9} ;加载R0 指向的地址上的多字数据,保存到R3~R9 中,R0 值更新

STMIA R1!,{R3-R9} ;将R3~R9 的数据存储到R1 指向的地址上,R1 值更新

STMFD SP!,{R0-R7,LR} ;现场保存,将R0~R7、LR 入栈

LDMFD SP!,{R0-R7,PC}^;恢复现场,异常处理返回

在进行数据复制时,先设置好源数据指针,然后使用块拷贝寻址指令LDMIA/STMIA、LDMIB/STMIB、LDMDA/STMDA、LDMDB /STMDB 进行读取和存储。而进行堆栈操作时,则要先设置堆栈指针,一般使用SP 然后使用堆栈寻址指令STMFD/LDMFD、STMED。LDMED、STMFA/LDMFA、STMEA/LDMEA 实现堆栈操作。

多寄存器传送指令示意图如图A-1所示,其中R1为指令执行前的基址寄存器,R1’则为指令执行完后的基址寄存器。

(a)指令STMIA R1!,{R5-R7} (b)指令STMIB R1!,{R5-R7}

(c)指令STMDA R1!, {R5-R7} (d)指令STMDB R1!,{R5-R7}

图A-1 多寄存器传送指令示意图

数据是存储在基址寄存器的地址之上还是之下,地址是在存储第一个值之前还是之后增加还是减少。表A-3给出多寄存器传送指令映射示意表。

表A-3 多寄存器传送指令映射示意表

使用LDM/STM 进行数据复制例程如下:

LDR R0,=SrcData ;设置源数据地址

LDR R1,=DstData ;设置目标地址

LDMIA R0,{R2-R9} ;加载8 字数据到寄存器R2~R9

STMIA R1,{R2-R9} ;存储寄存器R2~R9 到目标地址

使用LDM/STM 进行现场寄存器保护,常在子程序中或异常处理使用:

SENDBYTE

STMFD SP!,{R0-R7,LR} ;寄存器入堆

BL DELAY ;调用DELAY 子程序

LDMFD SP!,{R0-R7,PC} ;恢复寄存器,并返回

SWP

寄存器和存储器交换指令。SWP指令用于将一个内存单元(该单元地址放在寄存器Rn中)的内容读取到一个寄存器Rd 中,同时将另一个寄存器Rm 的内容写入到该内存单元中。使用SWP 可实现信号量

操作。

指令格式如下:

SWP{cond}{B} Rd,Rm,[Rn]

其中,B 为可选后缀,若有B,则交换字节,否则交换32 位字:Rd 为数据从存储器加载到的寄存器;Rm 的数据用于存储到存储器中,若Rm 与Rn 相同,则为寄存器与存储器内容进行交换;Rn 为要进行数据交换的存储器地址,Rn 不能与Rd 和Rm 相同。

SWP 指令举例如下:

SWP R1,R1,[R0] ; 将R1 的内容与R0 指向的存储单元的内容进行交换

SWP R1,R2,,[R0] ; 将R0 指向的存储单元内容读取一字节数据到R1 中(高24 位清零)

; 并将R2 的内容写入到该内存单元中(最低字节有效) 使用SWP 指令可以方便地进行信号量的操作:

12C_SEM EQU 0x40003000

12C_SEM_WAIT

MOV R0,#0

LDR R0,=12C_SEM

SWP R1,R1,[R0] ;取出信号量,并设置其为0

CMP R1,#0 ;判断是否有信号

BEQ 12C_SEM_WAIT ;若没有信号,则等待

3 ARM 数据处理指令

数据处理指令大致可分为3 类;数据传送指令(如MOV、MVN),算术逻辑运算指令(如ADD,SUM,AND),比较指令(如CMP、TST)。数据处理指令只能对寄存器的内容进行操作。

所有ARM 数据处理指令均可选择使用S 后缀,以影响状态标志。比较指令CMP、CMN、TST和TEQ 不需要后缀S,它们会直接影响状态标志。ARM 数据处理指令列于表A-4中。

表A-4 ARM 数据处理指令

(1)数据传送指令

MOV

数据传送指令。将8 位图立即数或寄存器(operant2)传送到目标寄存器Rd,可用于移位运算等操作。指令格式如下:

MOV{cond}{S} Rd,operand2

MOV 指令举例如下:

MOV R1#0x10 ;R1=0x10

MOV R0,R1 ;R0=R1

MOVS R3,R1,LSL #2 ;R3=R1<<2,并影响标志位

MOV PC,LR ;PC=LR ,子程序返回

MVN

数据非传送指令。将8 位图立即数或寄存器(operand2)按位取反后传送到目标寄存器(Rd),因为其具有取反功能,所以可以装载范围更广的立即数。指令格式如下:

MVN{cond}{S} Rd,operand2

MVN 指令举例如下:

MVN R1,#0xFF ;R1=0xFFFFFF00

MVN R1,R2 ;将R2 取反,结果存到R1

(2)算术逻辑运算指令

ADD

加法运算指令。将operand2 数据与Rn 的值相加,结果保存到Rd 寄存器。指令格式如下:

ADD{cond}{S} Rd,Rn,operand2

ADD 指令举例如下:

ADDS R1,R1,#1 ;R1=R1+1

ADD R1,R1,R2 ;R1=R1+R2

ADDS R3,R1,R2,LSL #2 ;R3=R1+R2<<2

SUB

减法运算指令。用寄存器Rn 减去operand2。结果保存到Rd 中。指令格式如下:

SUB{cond}{S} Rd,Rn,operand2

SUB 指令举例如下:

SUBS R0,R0,#1 ;R0=R0-1

SUBS R2,R1,R2 ;R2=R1-R2

SUB R6,R7,#0x10 ;R6=R7-0x10

RSB

逆向减法指令。用寄存器operand2 减法Rn,结果保存到Rd 中。指令格式如下:

RSB{cond}{S} Rd,Rn,operand2

SUB 指令举例如下:

RSB R3,R1,#0xFF00 ;R3=0xFF00-R1

RSBS R1,R2,R2,LSL #2 ;R1=R2<<2-R2=R2×3

RSB R0,R1,#0 ;R0=-R1

ADC

带进位加法指令。将operand2 的数据与Rn 的值相加,再加上CPSR 中的C 条件标志位。结果保存到Rd 寄存器。指令格式如下:

ADC{cond}{S} Rd,Rn,operand2

ADDS R0,R0,R2

ADC R1,R1,R3 ;使用ADC 实现64 位加法,(R1、R0)=(R1、R0)+(R3、R2)

SBC

带进位减法指令。用寄存器Rn 减去operand2,再减去CPSR 中的C 条件标志位的非(即若C 标志清零,则结果减去1),结果保存到Rd 中。指令格式如下:

SCB{cond}{S}Rd,Rn,operand2

SBC 指令举例如下:

SUBS R0,R0,R2

SBC R1,R1,R3 ;使用SBC 实现64 位减法,(R1,R0)-(R3,R2)

RSC

带进位逆向减法指令。用寄存器operand2 减去Rn,再减去CPSR 中的C 条件标志位,结果保存到Rd 中。指令格式如下:

RSC{cond}{S} Rd,Rn,operand2

RSC 指令举例如下:

RSBS R2,R0,#0

RSC R3,R1,#0 ;使用RSC 指令实现求64 位数值的负数

AND

逻辑与操作指令。将operand2 值与寄存器Rn 的值按位作逻辑与操作,结果保存到Rd 中。指令格式如下:

AND{cond}{S} Rd,Rn,operand2

AND 指令举例如下:

ANDS R0,R0,#x01 ;R0=R0&0x01,取出最低位数据

AND R2,R1,R3 ;R2=R1&R3

ORR

逻辑或操作指令。将operand2 的值与寄存器Rn 的值按位作逻辑或操作,结果保存到Rd 中。指令格式如下:

ORR{cond}{S} Rd,Rn,operand2

ORR R0,R0,#x0F ;将R0 的低4 位置1

MOV R1,R2,LSR #4

ORR R3,R1,R3,LSL #8 ;使用ORR 指令将近R2 的高8 位数据移入到R3 低8 位中

EOR

逻辑异或操作指令。将operand2 的值与寄存器Rn 的值按位作逻辑异或操作,结果保存到Rd 中。指令格式如下:

EOR{cond}{S}Rd,Rn,operand2

EOR 指令举例如下:

EOR R1,R1,#0x0F ;将R1 的低4 位取反

EOR R2,R1,R0 ;R2=R1^R0

EORS R0,R5,#0x01 ;将R5 和0x01 进行逻辑异或,结果保存到R0,并影响标志位

BIC

位清除指令。将寄存器Rn 的值与operand2 的值的反码按位作逻辑与操作,结果保存到Rd 中。指令格式如下:

BIC{cond}{S}Rd,Rn,operand2

BIC 指令举例如下:

BIC R1,R1,#0x0F ;将R1 的低4 位清零,其它位不变

BIC R1,R2,R3 ;将拭的反码和R2 相逻辑与,结果保存到R1

(3)比较指令

CMP

比较指令。指令使用寄存器Rn 的值减去operand2 的值,根据操作的结果更新CPSR中的相应条件标志位,以便后面的指令根据相应的条件标志来判断是否执行。指令格式如下:

CMP{cond} Rn,operand2

CMP 指令举例如下:

CMP R1,#10 ;R1 与10 比较,设置相关标志位

CMP R1,R2 ;R1 与R2 比较,设置相关标志位

CMP 指令与SUBS 指令的区别在于CMP 指令不保存运算结果。在进行两个数据大小判断时,常用CMP 指令及相应的条件码来操作。

CMN

负数比较指令。指令使用寄存器Rn 与值加上operand2 的值,根据操作的结果更新CPSR 中的相应条件标志位,以便后面的指令根据相应的条件标志来判断是否执行,指令格式如下:

CMN{cond} Rn,operand2

CMN R0,#1 ;R0+1,判断R0 是否为1 的补码,若是Z 置位

CMN 指令与ADDS 指令的区别在于CMN 指令不保存运算结果。CMN 指令可用于负数比较,比如CMNR0,#1 指令则表示R0 与-1 比较,若R0 为-(即1 的补码),则Z 置位,否则Z复位。

TST

位测试指令。指令将寄存器Rn 的值与operand2 的值按位作逻辑与操作,根据操作的结果更新CPSR 中相应的条件标志位,以便后面指令根据相应的条件标志来判断是否执行。指令格式如下:

TST{cond} Rn,operand2

TST 指令举例如下:

TST R0,#0x01 ;判断R0 的最低位是否为0

TST R1,#0x0F ;判断R1 的低4 位是否为0

TST 指令与ANDS 指令的区别在于TST4 指令不保存运算结果。TST 指令通常于EQ、NE条件码配合使用,当所有测试位均为0 时,EQ 有效,而只要有一个测试为不为0,则NE 有效。

TEQ

相等测试指令。指令寄存器Rn 的值与operand2 的值按位作逻辑异或操作,根据操作的结果更新CPSR 中相应条件标志位,以便后面的指令根据相应的条件标志来判断是否执行。指令格式如下:

TEQ{cond} Rn,operand2

TEQ 指令举例如下:

TEQ R0,R1 ;比较R0 与R1 是否相等(不影响V 位和C 位)

TST 指令与EORS 指令的区别在于TST 指令不保存运算结果。使用TEQ 进行相等测试,常与EQNE 条件码配合使用,当两个数据相等时,EQ 有效,否则NE 有效。

(4)乘法指令

ARM7TDMI(-S)具有32×32 乘法指令、32×32 乘加指令、32×32 结果为64 位的乘法指令。表

A-5给出全部的ARM 乘法指令。

表A-5 全部的ARM 乘法指令

MUL

32 位乘法指令。指令将Rm 和Rs 中的值相乘,结果的低32 位保存到Rd 中。指令格式如下:

MUL{cond}{S} Rd,Rm,Rs

MUL 指令举例如下:

MUL R1,R2,R3 ;R1=R2×R3

MULS R0,R3,R7 ;R0=R3×R7,同时设置CPSR 中的N 位和Z 位

MLA

32 位乘加指令。指令将Rm 和Rs 中的值相乘,再将乘积加上第3 个操作数,结果的低32 位保存到Rd 中。指令格式如下:

MLA{cond}{S} Rd,Rm,Rs,Rn

MLA 指令举例如下:

MLA R1,R2,R3,R0 ;R1=R2×R3+10

UMULL

64 位无符号乘法指令。指令将Rm 和Rs 中的值作无符号数相乘,结果的低32 位保存到RsLo 中,而高32 位保存到RdHi 中。指令格式如下:

UMULL{cond}{S} RdLo,RdHi,Rm,Rs

UMULL 指令举例如下:

UMULL R0,R1,R5,R8 ;(R1、R0)=R5×R8

UMLAL

64 位无符号乘加指令。指令将Rm 和Rs 中的值作无符号数相乘,64 位乘积与RdHi、RdLo 相加,

结果的低32 位保存到RdLo 中,而高32 位保存到RdHi 中。指令格式如下:

UMLAL{cond}{S} RdLo,RdHi,Rm,Rs

UMLAL 指令举例如下:

UMLAL R0,R1,R5,R8 ;(R1,R0)=R5×R8+(R1,R0)

SMULL

64 位有符号乘法指令。指令将Rm 和Rs 中的值作有符号数相乘,结果的低32 位保存到RdLo 中,而高32 位保存到RdHi 中。指令格式如下:

SMULL{cond}{S} RdLo,RdHi,Rm,Rs

SMULL 指令举例如下:

SMULL R2,R3,R7,R6 ;(R3,R2)=R7×R6

SMLAL

64 位有符号乘加指令。指令将Rm 和Rs 中的值作有符号数相乘,64 位乘积与RdHi、RdLo,相加,结果的低32 位保存到RdLo 中,而高32 位保存到RdHi 中。指令格式如下:

SMLAL{cond}{S} RdLo,RdHi,Rm,Rs

SMLAL 指令举例如下:

SMLAL R2,R3,R7,R6 ;(R3,R2)=R7×R6+(R3,R2)

4 ARM 跳转指令

在ARM 中有两种方式可以实现程序的跳转,一种是使用跳转指令直接跳转,另一种则是直接向PC 寄存器赋值实现跳转。跳转指令有跳转指令B,带链接的跳转指令BL 带状态切换的跳转指令BX。表A-6

给出全部的ARM跳转指令。

表A-6 ARM跳转指令

B

跳转指令。跳转到指定的地址执行程序。指令格式如下:

B{cond} label

跳转指令B 举例如下:

B WAITA ;跳转到WAITA 标号处

B 0x1234 ;跳转到绝对地址0x1234 处

跳转到指令B 限制在当前指令的±32Mb 的范围内。

BL

带链接的跳转指令。指令将下一条指令的地址拷贝到R14(即LR)链接寄存器中,然后跳转到指定地址运行程序。指令格式如下:

BL{cond} label

带链接的跳转指令BL 举例如下:

BL DELAY

跳转指令B 限制在当前指令的±32MB 的范围内。BL 指令用于子程序调用。

BX

带状态切换的跳转指令。跳转到Rm 指定的地址执行程序,若Rm 的位[0]为1,则跳转时自动将CPSR 中的标志T 置位,即把目标地址的代码解释为Thumb 代码;若Rm 的位[0]为0,则跳转时自动将CPSR 中的标志T 复位,即把目标地址的代码解释为ARM 代码。指令格式如下:

BX{cond} Rm

带状态切换的跳转指令BX 举例如下:

ADRL R0,ThumbFun+1

BX R0 ;跳转到R0 指定的地址,并根据R0 的最低位来切换处理器状态

5 ARM 协处理器指令

ARM 支持协处理器操作,协处理器的控制要通过协处理器命令实现。表A-7给出全部的ARM协处理器指令。

表A-7 ARM 协处理器指令

CDP

协处理器数据操作指令。ARM 处理器通过CDP 指令通知ARM 协处理器执行特定的操作。该操作由协处理器完成,即对命令的参数的解释与协处理器有关,指令的使用取决于协处理器。若协处理器不能成功地执行该操作,将产生未定义指令异常中断。指令格式如下:

CDP{cond} coproc,opcodel,CRd,CRn,CRm{,opcode2}

其中: coproc 指令操作的协处理器名。标准名为pn,n 为0~15。

opcodel 协处理器的特定操作码。

CRd 作为目标寄存器的协处理器寄存器。

CRN 存放第1 个操作数的协处理器寄存器。

CRm 存放第2 个操作数的协处理器寄存器。

Opcode2 可选的协处理器特定操作码。

CDP 指令举例如下:

CDP p7,0,c0,c2,c3,0 ;协处理器7 操作,操作码为0,可选操作码为0

CDP p6,1,c3,c4,c5 ;协处理器操作,操作码为1

LDC

协处理器数据读取指令。LDC 指令从某一连续的内存单元将数据读取到协处理器的寄存器中。协处理器数据的数据的传送,由协处理器来控传送的字数。若协处理器不能成功地执行该操作,将产生未定义指令异常中断。指令格式如下:

LDC{cond}{L} coproc,CRd,<地址>

其中: L 可选后缀,指明是长整数传送。

coproc 指令操作的协处理器名。标准名为pn,n 为0~15

CRd 作为目标寄存的协处理器寄存器。

<地址> 指定的内存地址

LDC 指令举例如下:

LDC p5,c2,[R2,#4];读取R2+4指向的内存单元的数据,传送到协处理器p5的c2寄存器中

LDC p6,c2,[R1] ;读取是指向的内存单元的数据,传送到协处理器p6 的c2 寄存器中

STC

协处理器数据写入指令。STC 指令将协处理器的寄存器数据写入到某一连续的内存单元中。进行协处理器数据的数据传送,由协处理器来控制传送的字数。若协处理器不能成功地执行该操作,将产生未定义指令异常中断。指令格式如下:

STC{cond}{L} coproc,CRd,<地址>

其中: L 可选后缀,指明是长整数传送。

coproc 指令操作的协处理器名。标准名为pn,n 为0~15

CRd 作为目标寄存的协处理器寄存器。

<地址> 指定的内存地址

STC 指令举例如下:

STC p5,c1,[R0]

STC p5,c1,[Ro,#-0x04]

MCR

ARM 寄存器到协处理器寄存器的数据传送指令。MCR 指令将ARM 处理器的寄存器中的数据传送到协处理器的寄存器中。若协处理器不能成功地执行该操作,将产生未定义指令异常中断。指令格式如下: MCR{cond} coproc,opcodel,Rd,CRn,CRm{,opcode2}

其中:coproc 指令操作的协处理器名。标准名为pn,n 为0~15。

cpcodel 协处理器的特定操作码。

CRD 作为目标寄存器的协处理器寄存器。

CRn 存放第1 个操作数的协处理器寄存器

CRm 存放第2 个操作数的协处理器寄存器。

Opcode2 可选的协处理器特定操作码。

MCR 指令举例如下:

MCR p6,2,R7,c1,c2,

MCR P7,0,R1,c3,c2,1,

ARM指令 中文全称及功能应用详解

指令格式:指令{条件}{S} {目的Register},{OP1}, {OP2} "{ }"中的内容可选。即,可以不带条件只有目的寄存器,或 只有目的寄存器和操作数1,也可以同时包含所有选项。“S”决定指令的操作是否影响CPSR中条件标志位的值,当没有S时指令不更新CPSR中条件标志位的值 助记符英文全称示例、功能 跳转指令 B Branch 跳转指令 B Label;程序无条件跳转到标号Label处执行 BL Branch with Link 带返回的跳转指令 BL Label ;当程序无条件跳转到标号Label处执行时,同时将当前的PC值保存到 R14中 BLX Branch with Link and exchange 带返回和状态切换的跳转指令 BLX Label;从ARM指令集跳转到指令中所指定的目标地址,并将处理器的工作状 态有ARM状态切换到Thumb状态,该指令同时将PC的当前内容保存到寄存器R14中BX Branch and exchange 带状态切换的跳转指令 BX Label ;跳转到指令中所指定的目标地址,目标地址处的指令既可以是ARM指 令,也可以是Thumb指令 数据处理MOV Move 数据传送 MOV R1,R0,LSL#3;将寄存器R0的值左移3位后传送到R1 MVN Move NOT 数据非传送 MVN R0,#0 ;将立即数0取反传送到寄存器R0中,完成后R0=-1 CMP Compare 比较指令 CMP R1,R0;将寄存器R1的值与寄存器R0的值相减,并根据结果设置CPSR的 标志位 CMN Compare negative 负数比较指令 CMN R1,R0 ;将寄存器R1的值与寄存器R0的值相加,并根据结果设置CPSR的 标志位 TST Test 位测试指令 TST R1,#0xffe ;将寄存器R1的值与立即数0xffe按位与,并根据结果设置CPSR 的标志位 TEQ Test equivalence 相等测试指令 TEQ R1,R2;将寄存器R1的值与寄存器R2的值按位异或,并根据结果设置CPSR 的标志位 ADD Add 加法运算指令 ADD R0,R2,R3,LSL#1; R0 = R2 + (R3 << 1) ADC Add with carry 带进位加法 ADCS R2,R6,R10; R2 = R6+R10+!C,且更新CPSR的进位标志位 SUB Subtract 减法运算指令 SUB R0,R1,#256; R0 = R1 – 256 SBC Subtract with carry 带进位减法指令 SUBS R0,R1,R2 ; R0 = R1 - R2 - !C,并根据结果设置CPSR的进位标志位RSB Reverse subtract 逆向减法指令 RSB R0,R1,R2 ; R0 = R2 – R1 RSC Reverse subtract with carry 带进位逆向减法指令 RSC R0,R1,R2; R0 = R2 – R1 - !C AND And 逻辑与操作指令 AND R0,R0,#3;该指令保持R0的0、1位,其余位清零。 ORR OR 逻辑或操作指令 ORR R0,R0,#3;该指令设置R0的0、1位,其余位保持不变。 EOR Exclusive OR 逻辑异或操作指令 EOR R0,R0,#3 ;该指令反转R0的0、1位,其余位保持不变。 BIC Bit clear 位清除指令 BIC R0,R0,#0b1011;该指令清除 R0 中的位 0、1、和 3,其余的位保持不变。CLZ Count left zero 计算操作数最高端0的个数 乘加指令 MUL Multiply 32位乘法指令 MUL R0,R1,R2 ;R0 = R1 × R2 MLA Multiply and accumulate 32位乘加指令 MLAS R0,R1,R2,R3;R0 = R1 × R2 + R3,同时设置CPSR中的相关条件标志位SMULL Signed multiply long 64位有符号数乘法指令 SMULL R0,R1,R2,R3;R0 = (R2 × R3)的低32位 R1 = (R2 × R3)的 高32位

ARM的汇编指令讲解

汇编知识点的要求: 1、能看的懂 2、可以做修改 3、不需要用汇编直接编写程序 汇编代码的应用场合: 1、ARM的启动代码必须要汇编,如:uboot最开始初始化硬件的代码 2、内核在最开始初始化的位置。。。。 一、ARM汇编指令的编码格式 1、编码格式 ARM汇编指令编译成机器码以后,机器码的长度是32bits,这32bits的编码有一个固定的格式。不同ARM 汇编指令,编码格式不同。 2、举例 C: if(a==10) a++; else a--;

汇编1: CMP R0, #10; ADDEQ R0,R0,#1 SUBNE R0,R0,#1 汇编2 SUBS R1, R0, #10; //S ---运算的结果会影响条件码标志位:CPSR:NZCV ADDEQ R0,R0,#1 SUBNE R0,R0,#1 提示: 空指令NOP,实际上是占用CPU的时间,但是执行后,没有什么意义。 NOP ---- MOV R0,R0 3、条件码标识 10 -10 Z = 1 C = 0 N = 0 V = 0 ================================================================================= 二、ARM的寻址方式 1、立即数寻址 操作数,有立即数。 ADD R0,R0,#1

MOV R1,#10 ORR R1,R1,#0xf @ R1=R1 | 0xf BIC R1,R1,#0xf @R1 = R1&(~(0xf)) 错误: ADD R1,#1,#2 注意:立即数合法的条件 在ARM汇编指令中,并不是所有的立即数,立即数是有一定的限制的。 什么样的立即数是合法的??? 1、如果一个立即数是小于256的(即该立即数是8bits以内的,0~255),该立即数是合法的。 2、如果一个立即数是大于等于256,该立即数经过循环左移偶数位,可以得到一个小于256的数,则该立即数合法。 256 = 0x100 ------→左移20位0x10000000----→左移4 0x1 合法 0x111 非法 0x102 非法 0x104 合法 0xfff 0xff00 0x12000 0x450000 0xab 原因: 在数据处理指令编码的时候,立即数用12bits来表示: 高4bits:循环左移左移偶数位除以2 低8bits:循环左移后的结果。 重要问题: ADD R1,R0,#0xffff 非法 解决: LDR R2,=0xffff // R2=0xffff,将立即数0xffff的值传送给R2 ADD R1, R0, R2 2、寄存器寻址 所有的操作数都是寄存器,没有立即数 ADD R0,R0,R1 MOV R1,R0 ORR R1,R1,R0 @ R1=R1 | 0xf BIC R1,R1,R0 @R1 = R1&(~(0xf))

Thumb指令集与ARM指令集的区别

标题:Thumb指令集与ARM指令集的区别 2010-06-21 21:43:58 Thumb指令集 Thumb指令可以看做是ARM指令压缩形式的子集,是针对代码密度【1】的问题而提出的,它具有16为的代码密度。Thumb不是一个完整的体系结构,不能指望处理程序只执行Thumb指令而不支持ARM指令集。因此,Thumb指令只需要支持通用功能,必要时,可借助完善的ARM指令集,例如:所有异常自动进入ARM状态。 在编写Thumb指令时,先要使用伪指令CODE16声明,而且在ARM指令中要使用BX指令跳转到Thumb指令,以切换处理器状态。编写ARM指令时,可使用伪指令CODE32声明。 【1】.代码密度:单位存储空间中包含的指令的个数。例如 ARM指令是32位的,而Thumb指令时16位的,如果在1K 的存储空间中,可以放32条ARM指令,就可以放64条Thumb指令,因此在存放Thunb指令时,代码密度高。 Thumb指令集与ARM指令集的区别 Thumb指令集没有协处理器指令、信号量指令以及访问CPSR或SPSR的指令,没有乘加指令及64位乘法指令等,且指令的第二操作数受到限制;除了跳转指令B有条件执行功能外,其他指令均为无条件执行;大多数Thumb数据处理指令采用2地址格式。Thumb指令集与ARM指令集的区别一般有如下几点: ? 跳转指令 程序相对转移,特别是条件跳转与ARM代码下的跳转相比,在范围上有更多的限制,转向子程序是无条件的转移。 ? 数据处理指令

数据处理指令是对通用寄存器进行操作,在大多数情况下,操作的结果须放入其中一个操作数寄存器中,而不是第三个寄存器中。 数据处理操作比ARM状态的更少,访问寄存器R8—R15受到一定限制。 (除MOV和ADD指令访问寄存器R8—R15外,其他数据处理指令总是更新CPSR中ALU状态标志) 访问寄存器R8—R15的Thumb数据处理指令不能更新CPSR中的ALU状态标志 ? 单寄存器加载和存储指令 在Thumb状态下,单寄存器加载和存储指令只能访问寄存器 R0—R7 ? 批量寄存器加载和存储指令 LDM和STM指令可以将任何范围为R0——R7的寄存器子集加载或存储

ARM指令大全

目录 一、跳转指令 (4) 1、B指令 (4) 2、BL指令 (4) 3、BLX指令 (4) 4、BX指令 (5) 二、数据处理指令 (5) 1、MOV指令 (5) 2、MVN指令 (5) 3、CMP指令 (6) 4、CMN指令 (6) 5、TST指令 (6) 6、TEQ指令 (7) 7、ADD指令 (7) 8、ADC指令 (7) 9、SUB指令 (7) 10、~~~~C指令 (8) 11、R~~~~指令 (8) 12、RSC指令 (8) 13、AND指令 (9) 14、ORR指令 (9) 15、EOR指令 (9) 16、BIC指令 (9) 三、法指令与乘加指令 (10) 1、MUL指令 (10) 2、MLA指令 (10) 3、SMULL指令 (10) 4、SMLAL指令 (11) 5、UMULL指令 (11) 6、UMLAL指令 (11) 四、程序状态寄存器访问指令 (12) 1、MRS指令 (12) 2、MSR指令 (12) 五、加载/存储指令 (12) 1、LDR指令 (12) 2、LDRB指令 (13) 3、LDRH指令 (14) 4、STR指令 (14) 5、STRB指令 (14) 6、STRH指令 (15) 六、批量数据加载/存储指令 (15) LDM(或STM)指令 (15)

IA (15) IB (15) DA (15) DB (15) FD (15) ED (15) FA (15) EA (15) 七、数据交换指令 (16) 1、SWP指令 (16) 2、SWPB指令 (16) 八、移位指令(操作) (16) 1、LSL(或ASL) (17) 2、LSR (17) 3、ASR (17) 4、ROR (17) 5、RRX (17) 九、协处理器指令 (18) 1、CDP指令 (18) 2、LDC指令 (18) 3、STC指令 (18) 4、MCR指令 (19) 5、MRC指令 (19) 十、异常产生指令 (19) 1、SWI指令 (19) 2、BKPT指令 (20) 一、符号定义(Symbol Definition)伪指令 (20) 1、GBLA、GBLL和GBLS (20) 2、LCLA、LCLL和LCLS (21) 3、SETA、SETL和SETS (22) 4、RLIST (22) 二、数据定义(Data Definition)伪指令 (23) 1、DCB (23) 2、DCW(或DCWU) (23) 3、DCD(或DCDU) (24) 4、DCFD(或DCFDU) (24) 5、DCFS(或DCFSU) (25) 6、DCQ(或DCQU) (25) 7、SPACE (25) 8、MAP (26) 9、FILED (26) 三、汇编控制(Assembly Control)伪指令 (27) 1、IF、ELSE、ENDIF (27) 2、WHILE、WEND (28)

ARM 指令集表格

ARM指令集 ARM指令集由数据处理指令、跳转指令、Load/Store指令、程序状态寄存器指令、协处理 器指令和软件中断指令六大类构成。 1、数据处理指令 指令格式功能举例 MOV 数据传送指令MOV{}{S} , Rd=op1 MOV R0,#5 ;R0=5 MOV PC,R14 ;退出调 用者 MVN 数据取反传送指令MVN{}{S} , Rd=!op1 MVN R1,R2 ;将R2按 位取反结果存到R1 ADD 加法指令ADD{}{S} ,, Rd=Rn+op2 ADD R0,R1,R2,LSL#5 ; R0=R1+R2左移5位 ADC 带进位加法指令ADC{}{S} ,, Rd=Rn+op2+ca rry ADDS R0,R2,R4 ;低32 位相加,S表示结果影响条 件标志位的值 SUB 减法指令SUB{}{S} ,, Rd=Rn-op2 SUB R0,R1,#5 ; R0=R1-5 RSB 反向减法指令RSB{}{S} ,, Rd=op2-Rn RSB R0,R1,#5 ; R0=5-R1 SBC 带借位减法指令SBC{}{S} ,, Rd=Rn-op2-!ca rry SUBS R0,R2,R4 ; 低32位相减 RSC 带借位的反向减法RSC{}{S} ,, Rd=op2-Rn-!ca rry RSC R1,R5,R3 ;高32位 相减 MUL 32位乘法指令MUL{}{S} ,, Rd=Rn*op2 MULS R0,R1,R2 ;R0=R1*R2 MLA 32位乘法指令MLA{}{S} ,,, Rd=Rn*op2+op 3 MLA R0,R1,R2,R3 ;R0=R1*R2 +R3 SMULL 64位有符号数乘法指令SMULL{}{S} ,,, Rdh Rdl=Rn*op2 SMULL R2,R3,R7,R6 ;(R3,R2)=R 7*R6 SMLAL 64位有符号数乘加指令SMLAL{}{S} ,,, Rdh Rdl=Rn*op2+R dh Rdl SMLAL R2,R3,R7,R6 ;(R3,R2)=R 7*R6+(R3,R2) UMULL 64位无符号数乘法指令UMULL{}{S} ,,, Rdh Rdl=Rn*op2 UMUll R0,R1,R5,R8 ;(R1,R0)=R 5*R8 UMLAL 64位无符号数乘加指令UMLAL{}{S} ,,, Rdh Rdl=Rn*op2+R dh Rdl UMLAL R0,R1,R5,R8 ;(R1,R0)=R 5*R8+(R1,R0) AND 逻辑与指令AND{}{S} ,, Rd=Rn AND op2 AND R0,R0,#5 ;保持R0 的第0位和第2位,其余

ARM指令集详解(超详细!带实例!)

算术和逻辑指令 ADC : 带进位的加法 (Ad dition with C arry) ADC{条件}{S} , , dest = op_1 + op_2 + carry ADC将把两个操作数加起来,并把结果放置到目的寄存器中。它使用一个进位标志位,这样就可以做比32 位大的加法。下列例子将加两个128 位的数。 128 位结果: 寄存器0、1、2、和3 第一个128 位数: 寄存器4、5、6、和7 第二个128 位数: 寄存器8、9、10、和11。 ADDS R0, R4, R8 ; 加低端的字 ADCS R1, R5, R9 ; 加下一个字,带进位 ADCS R2, R6, R10 ; 加第三个字,带进位 ADCS R3, R7, R11 ; 加高端的字,带进位 如果如果要做这样的加法,不要忘记设置S 后缀来更改进位标志。 ADD : 加法 (Add ition) ADD{条件}{S} , , dest = op_1 + op_2 ADD将把两个操作数加起来,把结果放置到目的寄存器中。操作数 1 是一个寄存器,操作数2 可以是一个寄存器,被移位的寄存器,或一个立即值:

ADD R0, R1, R2 ; R0 = R1 + R2 ADD R0, R1, #256 ; R0 = R1 + 256 ADD R0, R2, R3,LSL#1 ; R0 = R2 + (R3 << 1) 加法可以在有符号和无符号数上进行。 AND : 逻辑与 (logical AND) AND{条件}{S} , , dest = op_1 AND op_2 AND将在两个操作数上进行逻辑与,把结果放置到目的寄存器中;对屏蔽你要在上面工作的位很有用。操作数1 是一个寄存器,操作数2 可以是一个寄存器,被移位的寄存器,或一个立即值: AND R0, R0, #3 ; R0 = 保持 R0 的位0和 1,丢弃其余的位。 AND 的真值表(二者都是1 则结果为1): Op_1 Op_2 结果 0 0 0 0 1 0 1 0 0 1 1 1 BIC : 位清除 (Bi t C lear) BIC{条件}{S} , , dest = op_1 AND (!op_2)

ARM指令集详解

好东西,说得非常详细的。要做底层的可以看看 ARM指令集详解 发布: 2010-3-19 11:42 | 作者: Saiu | 来源: MCU嵌入式领域 ARM可以用两套指令集:ARM指令集和Thumb指令集。本文介绍ARM指令集。在介绍ARM指令集之前,先介绍指令的格式。 1 指令格式 (1)基本格式 {}{S} ,{,} 其中,<>内的项是必须的,{}内的项是可选的,如是指令助记符,是必须的,而{}为指令执行条件,是可选的,如果不写则使用默认条件AL(无条件执行)。 opcode 指令助记符,如LDR,STR 等 cond 执行条件,如EQ,NE 等 S 是否影响CPSR 寄存器的值,书写时影响CPSR,否则不影响 Rd 目标寄存器 Rn 第一个操作数的寄存器 operand2 第二个操作数 指令格式举例如下: LDR R0,[R1] ;读取R1 地址上的存储器单元内容,执行条件AL BEQ DATAEVEN ;跳转指令,执行条件EQ,即相等跳转到DATAEVEN ADDS R1,R1,#1 ;加法指令,R1+1=R1 影响CPSR 寄存器,带有S SUBNES R1,R1,#0xD;条件执行减法运算(NE),R1-0xD=>R1,影响CPSR 寄存器,带有S (2)第2个操作数 在ARM 指令中,灵活的使用第2个操作数能提高代码效率,第2个操作数的形式如下: #immed_8r 常数表达式,该常数必须对应8 位位图,即常数是由一个8 位的常数循环移位偶数位得到。 合法常量 0x3FC、0、0xF0000000、200、0xF0000001等都是合法常量。 非法常量 0x1FE、511、0xFFFF、0x1010、0xF0000010等都是非法常量。 常数表达式应用举例如下:

arm指令集基础系列

ARM指令和指令系统: 指令是指示计算机某种操作的命令,指令的集合称为指令系统。指令系统的功能强弱很大程度上决定了这类计算机智能的高低,它集中地反应了微处理器的硬件功能和属性。 ARM指令在机器中的表示格式是用32位的二进制数表示。如ARM中有一条指令为ADDEQS R0,R1,#8; 其二进制代码形式为: 31~28 | 27~25 | 24~21 | 20 | 19~16 | 15~12 | 11~0 0000 | 001 | 0100 | 1 | 0001 | 0000 | 0000 0000 1000 cond |opcode | Rn | Rd | Op2 ARM指令格式一般如下: {}{s}{,} 格式中< >的内容是必不可少的,{ }中的内容可忽略 表示操作码。如ADD表示算术加法 {} 表示指令执行的条件域。如EQ、NE等,缺省为AL。 {S} 决定指令的执行结果是否影响CPSR的值,使用该后缀则指令执行结果影响CPSR 的值,否则不影响 表示目的寄存器 表示第一个操作数,为寄存器 表示第二个操作数,可以是立即数。寄存器和寄存器移位操作数 ARM指令后缀:S、! S后缀:指令中使用S后缀时,指令执行后程序状态寄存器的条件标志位将被刷新,不使用S后缀时,指令执行后程序状态寄存器的条件标志将不会发生变化。S后缀常用于对条件进行测试,如是否有溢出,是否进位等,根据这些变化,就可以进行一些判断,如是否大于,相等,从而影响指令执行的顺序。 !后缀:如果指令地址表达式中不含!后缀,则基址寄存器中的地址值不会发生变化。加上此后缀后,基址寄存器中的值(指令执行后)= 指令执行前的值+ 地址偏移量(1)!后缀必须紧跟在地址表达式后面,而地址表达式要有明确的地址偏移量 (2)!后缀不能用于R15(PC)的后面 (3)当用在单个地址寄存器后面时,必须确信这个寄存器有隐性的偏移量,例如“STMDB R1!,{R3,R5,R7}”。此时地址基址寄存器R1的隐性偏移量为4(一条指令占32位,即4个字节) 指令的条件码:31-28位4个字节存储,共16个条件码 条件码助记符后缀标志含义 0000 EQ Z置位相等 0001 NE Z清零不相等 0010 CS C置位无符号数大于或等于0011 CC C清零无符号数小于 0100 MI N置位负数 0101 PL N清零正数或零

ARM指令大全

ARM指令集详解 ARM可以用两套指令集:ARM指令集和Thumb指令集。本文介绍ARM指令集。在介绍ARM指令集之前,先介绍指令的格式。 1 指令格式 (1)基本格式 {}{S} ,{,} 其中,<>内的项是必须的,{}内的项是可选的,如是指令助记符,是必须的,而{}为指令执行条件,是可选的,如果不写则使用默认条件AL(无条件执行)。 opcode 指令助记符,如LDR,STR 等 cond 执行条件,如EQ,NE 等 S 是否影响CPSR 寄存器的值,书写时影响CPSR,否则不影响 Rd 目标寄存器 Rn 第一个操作数的寄存器 operand2 第二个操作数 指令格式举例如下: LDR R0,[R1] ;读取R1 地址上的存储器单元内容,执行条件AL BEQ DATAEVEN ;跳转指令,执行条件EQ,即相等跳转到DATAEVEN ADDS R1,R1,#1 ;加法指令,R1+1=R1 影响CPSR 寄存器,带有S SUBNES R1,R1,#0xD;条件执行减法运算(NE),R1-0xD=>R1,影响CPSR 寄存器,带有S (2)第2个操作数 在ARM 指令中,灵活的使用第2个操作数能提高代码效率,第2个操作数的形式如下: #immed_8r 常数表达式,该常数必须对应8 位位图,即常数是由一个8 位的常数循环移位偶数位得到。 合法常量 0x3FC、0、0xF0000000、200、0xF0000001等都是合法常量。 非法常量 0x1FE、511、0xFFFF、0x1010、0xF0000010等都是非法常量。 常数表达式应用举例如下: MOV R0,#1 ;R0=1 AND R1,R2,#0x0F ;R2 与0x0F,结果保存在R1 LDR R0,[R1],#-4 ;读取R1 地址上的存储器单元内容,且R1=R1-4 Rm 寄存器方式,在寄存器方式下操作数即为寄存器的数值。 寄存器方式应用举例: SUB R1,R1,R2 ;R1-R2=>R1 MOV PC,R0 ;PC=R0,程序跳转到指定地址 LDR R0,[R1],-R2 ;读取R1 地址上的存储器单元内容并存入R0,且R1=R1-R2 Rm, shift 寄存器移位方式。将寄存器的移位结果作为操作数,但RM 值保存不变,移位方法如下: ASR #n 算术右移n 位(1≤n≤32) LSL #n 逻辑左移n 位(1≤n≤31) LSR #n 逻辑左移n 位(1≤n≤32) ROR #n 循环右移n 位(1≤n≤31) RRX 带扩展的循环右移1位 type Rs 其中,type 为ASR,LSL,和ROR 中的一种;Rs 偏移量寄存器,低8位有效,若其值大于或等于32,

ARM指令系统

ARM指令系统.txt蜜蜂整日忙碌,受到赞扬;蚊子不停奔波,人见人打。多么忙不重要,为什么忙才重要。1.ARM指令的寻址方式有几种?试分别举例说明。 答: 立即数寻址:MOV R0,#15 寄存器寻址:ADD R0,R1,R2 寄存器间接寻址:LDR R0,[R4] 寄存器移位寻址:ADD R0,R1,R2,LSL #1 基址变址寻址:LDR R0,[R1,#4] 多寄存器寻址:LDMIA R0!,{R1-R4} 相对寻址:BL proc ;跳转到子程序proc处执行 … proc MOV R0,#1 堆栈寻址:STMFD R13!,{R0-R4}; LDMFD R13!,{R0-R4}; 2.ARM指令系统对字节、半字、字的存取是如何实现的? LDR/STR 字数据加载/存储指令 LDRB/STRB 字节数据加载/存储指令 LDRH/STRH 半字数据加载/存储指令 LDRSB/LDRSH 有符号数字节/半字加载指令 3.如何从ARM指令集跳转到Thumb指令集? BX指令跳转到指令中所指定的目标地址,并实现状态的切换。 Rm是一个表达目标地址的寄存器。当Rm中的最低位Rm[0]为1时,强制程 序从ARM指令状态跳到Thumb指令状态;当Rm中的最低位Rm[0]为0时, 强制程序从Thumb指令状态跳到ARM指令状态。 4.ARM指令集支持哪几种协处理器指令? CDP LDC STC MCR MRC 5.ARM指令的条件码有多少个?默认条件码是什么? 15个默认条件码是AL 6.ARM指令中的第二操作数有哪几种形式?试举例说明。 寄存器形式 MOV R1,R0 ;将寄存器R0的值传送到寄存器R1 立即数形式 MOV R0,#5 ;将立即数5传送到寄存器R0 被移位的寄存器形式 MOV R1,R0,LSL#3

ARM v7指令集

ARM v7指令集 内容简介:在本文档的汇编语法中,用# 前缀表示立即值,用& 表示十六进制值,用% 表示二进制值,用{花括号} 表示指令中可选的设置字段或位。下面表格中粗体的指令是核心ARM 指令,其他的是值得包含的位和片... 在本文档的汇编语法中,用# 前缀表示立即值,用& 表示十六进制值,用% 表示二进制值,用{花括号} 表示指令中可选的设置字段或位。下面表格中粗体的指令是核心ARM 指令,其他的是值得包含的位和片段、移位选项和汇编器 助记码(mnemonic)... 还列出了协处理器指令。但是用于RISC OS 机器的ARM 处理器不支持协处理器,只在一个可访问的芯片中提供了实际上的协处理器功能。其中包括设置ARM、cache、MMU 的设施,等... 指令意义最早的CPU / 注释 ADC带进位的加法- ADD加法- AND逻辑与- ASL 算术左移这是一个选项,不是指令 ASR 算术右移这是一个选项,不是指令 B分支- BIC位清除- BL带连接的分支- BX 分支到Thumb 代码StrongARM SA1110 ? CDP 协处理器数据操作- CMN比较取负的值- CMP比较值- EOR异或两个值- LDC 装载内存到协处理器- LDM装载多个寄存器- LDR装载寄存器- LDRB装载字节到寄存器- LDRH装载半字到寄存器StrongARM LDRSB装载有符号字节到寄存器StrongARM LDRSH装载有符号半字到寄存器StrongARM LSL 逻辑左移这是一个选项,不是指令 LSR 逻辑右移这是一个选项,不是指令 MCR 协处理器寄存器传送- MLA带累加的乘法- MOV传送值/寄存器到一个寄存器-

必须知道的ARM指令集介绍

常用的ARM指令集 我们知道CPU能直接识别的是机器码,也就是0和1组合的东东。不同的组合代表不同的含义,CPU就做不同的事情。按照道理来说,我们写的代码都应该是0和1组成的东西,但是如果真的是那样,那结果....。人类是聪明的,发明了汇编语言,汇编语言是用一些助记符,来代替机器码。当然,汇编语言是不能给CPU识别的,需要编译器将其编译成机器码。 汇编语言是和特定的CPU关联的,不同的CPU对应不同的汇编指令。好在,现在大多数嵌入式CPU都是使用ARM架构的,这样我们只需要学会ARM的汇编指令集,我们写的汇编程序就可以在这些使用ARM架构的CPU上运行。 ARM的汇编指令很多,很难将其全部记住,其实也没必要,我们只需要学会一些常用的ARM汇编指令即可。其他的用到再去查即可。下面我们就开始,必须知道的ARM指令学习。 一、ARM指令格式 ARM指令的基本格式如下: 其中<>号内是必须的,{}号内的项是可选的。 各项的说明如下: opcode :指令助记符 cond :执行条件 S :是否影响CPSR寄存器的值

Rd :目标寄存器(一般用来存放指令执行最后的结果) Rn :第1个操作数的寄存器 operand2 :第2个操作数 1. 其中,我们必须要研究一下operand2这个操作数,它有如下表现形式 (1)#immed_8r——常数表达式 该常数必须对应8位位图,即一个8位的常数通过循环右移偶数位得到,这个数我们也成立即数。 例如: MOV R0,#1 AND R1,R2,#0x0 例如:0x04800000它可以拿0x12循环右移位10位得到,所以它是一个合法的立即数。 有人会问,是不是每次我都要去判断一个数是不是立即数,我就需要从0x00~0xff里面找出一个数,将它循环右移偶数个位看是否能的得到,那太麻烦了。有没有一些快速识别的方法呢?肯定是有的呀,当然也不能完全依赖哦,最好就是拿编译器验证,不是,编译器就会报

常用ARM指令

常用ARM指令及汇编包括 1、ARM处理器寻址方式 2、指令集介绍 3、伪指令 4、ARM汇编程序设计 5、C与汇编混合编程 ARM处理器寻址方式 1、寄存器寻址:操作数的值在寄存器中,指令中的地址码字段指出的是寄存器编号,指令执行时直接取出寄存器值操作 MOV R1, R2 ;R2->R1 SUB R0, R1,R2 ;R1-R2 -> R0 2、立即寻址:立即寻址指令中的操作码字段后面的地址码部分就是操作数本身,也就是说,数据就包含在指令当中,取出指令就取出了可以立即使用的操作数 SUBS R0,R0,#1 ;R0-1 -> R0 MOV R0,#0xff00 ;0xff00 -> R0 注:立即数要以"#"为前缀,表示16进制数值时以"0x"表示 3、寄存器偏移寻址:是ARM指令集特有的寻址方式,当第2操作数是寄存器偏移方式时,第2个寄存器操作数在与第1个操作数结合之前选择进行移位操作 MOV R0,R2,LSL #3 ;R2的值左移3位,结果存入R0,即R0 = R2 * 8 ANDS R1,R1,R2,LSL R3 ;R2的值左移R3位,然后和R1相与操作,结果放入R1 寄存器偏移寻址可采用的移位操作如下 (1)、LSL(Logical Shift Left)逻辑左移,寄存器中字的低端空出补0 (2)、LSR(Logical Shift Right)逻辑右移,寄存器中字的高端空出补0 (3)、ASR(Arthmetic Shift Right)算术右移,移位中保持符号位不变,即如果源操作数为正数,字高端空出补0,否则补1 (4)、ROR(Rotate Right)循环右移,由字的低端移出的位填入高端空出的位 (5)、RRX(Rotate Right eXtended by 1 place),操作数右移一位,左侧空位由CPSR的C 填充

ARM指令集

ARM处理器的指令集 ARM处理器的指令找了好久都没有找到完整版,在这里我把所有用过的ARM指令都整理下来,方便大家参考。 ADC(不更改状态寄存器): 带进位的加法(Addition with Carry) ADCS(更改状态寄存器)R1, R5, R9 ; R1=R5+R9,带进位 ADD : 加法(Addition) ADD R0, R1, #256 ; R0 = R1 + 256 AND : 逻辑与(logical AND) AND R0, R0, #3 ; R0 = 保持R0 的位0和1,丢弃其余的位。 BIC : 位清除(Bit Clear) BIC R0, R0, #%1011 ; 清除R0 中的位0、1、和3。保持其余的不变。 EOR : 逻辑异或(logical Exclusive OR) EOR R0, R0, #3 ; 反转R0 中的位0 和1 MOV : 传送(Move) MOV PC, R14 ; PC=R14 MOVS PC, R14 ; 退出到调用者并恢复标志位(不遵从32-bit 体系) MVN : 传送取反的值(MoveNegative) MVN R0, #4 ; R0 = -5 ORR : 逻辑或(logical OR) ORR R0, R0, #3 ; 设置R0 中位0 和1 RSB : 反向减法(Reverse Subtraction) RSB R0, R1, R2 ; R0 = R2 - R1 RSC : 带借位的反向减法(Reverse Subtraction with Carry) RSC{条件}{S} , , dest = op_2 - op_1 - !carry SBC : 带借位的减法(Subtraction with Carry) SBC{条件}{S} , , dest = op_1 - op_2 - !carry

ARM总汇编指令集

ARM汇编指令集 1 跳转指令 1.1 跳转指令B: B LABLE ;跳转到标号LABEL处 B 0X1111 ;跳转到绝对地址0X1111处 1.2 带连接的跳转指令 BL: START … BL NEXT ;跳转到标号NEXT处,同时保存当前PC到R14中 … ;返回地址 … NEXT… ;子程序入口 MOV PC,R14 ;返回 1.3 带状态切换的跳转指令BX: MOV R0, #0X0201 BX R0 ;程序跳转到0x0200处,微处理器切换到Thumb状态(地址必须是4的倍数,否则产生不可预知的后果) 2算术运算指令 2.1不带进位加法指令ADD ADD R0, R1, R2 ;R0← (R1)+(R2) ADD R0, R1, #112 ;R0← (R1)+ 112 ADD R0, R1, R2, LSL #1 ;R0←(R1)+(R2<<1) ;将R2中的值左移1位,再与R1值相加,结果送R0 2.2带进位加法指令 ADC ADDS R0, R3, R6 ;加最低位字节,不带进位 ADCS R1, R4, R7 ;加第二个字,带进位 ADCS R2, R5,R8 ;加第三个字,带进位 ;三句话实现了96bit加法运算,由于ARM寄存器宽度只有32bit所以分三次相加 2.3 不带进位减法指令SUB ;S—进位标志 SUB R0, R1, R2 ;R0←(R1)- (R2) SUB R0, R1, #112 ;R0←(R1)- 112 SUB R0, R1 ,R2 LSL#1 ;R0←(R1)- (R2<<1) 2.4 带进位减法指令SBC SUBS R0, R3, R6 ;减最低位字节,不带进位 SBCS R1, R4, R7 ;减第二个字,带进位 SBCS R2, R5, R8 ;减第三个字,带进位 ;三句话实现了96bit减法运算,由于ARM寄存器宽度只有32bit所以分三次相减

相关主题
文本预览
相关文档 最新文档