一面学习,一面总结,一面记录。
以下是整理在网上找到的一些资料,简单整理记录一下,方便以后查阅。
ARM处理器的指令集能够分为跳转指令、数据处理指令、程序状态寄存器(PSR)处理指令、载入/存储指令、协处理器指令和异常产生指令6大指令。
一、跳转指令
Ⅱ.直接向程序计数器PC写入跳转地址值。通过向程序计数器PC写入跳转地址值,能够实如今4GB的地址空间中的随意跳转,在跳转之前结合使用MOV LR,PC等类似指令,能够保存将来的返回地址值。从而实如今4GB连续的线性地址空间的子程序调用。
ARM指令集中的跳转指令能够完毕从当前指令向前或向后的32MB的地址空间的跳转,包含下面4条指令:
1、B指令
BEQ Label
2、BL指令
该指令是实现子程序调用的一个基本但经常使用的手段。
以 下指令:
BL Label ;当程序无条件跳转到标号Label处运行时,同一时候将当前的 PC值保存到
;R14(LR)中
3、BLX指令
BLX指令从ARM指令集跳转到指令中所指定的目标地址,并将处理器的工作状态有ARM状态切换到Thumb状态,该指令同一时候将PC的当前内容保存到寄存 器R14中。因此,当子程序使用Thumb指令集。而调用者使用ARM指令集时,能够通过BLX指令实现子程序的调用和处理器工作状态的切换。
同一时候,子程 序的返回能够通过将寄存器R14值拷贝到PC中来完毕。
4、BX指令
二、数据处理指令
数据传送指令用于在寄存器和存储器之间进行数据的双向传输。
算术逻辑运算指令完毕经常使用的算术与逻辑的运算,该类指令不但将运算结果保存在目的寄存器中,同一时候更新CPSR中的对应条件标志位。
比較指令不保存运算结果。仅仅更新CPSR中对应的条件标志位。
数据处理指令共下面16条。
1、MOV指令(传送)
MOV R1。R0。LSL#3 ;将寄存器R0的值左移3位后传送到R1
2、MVN指令(求反)
当中S决定指令的操作是否影响CPSR中条件标志位的值。当没有S时指令不更新CPSR中条件标志位的值。
指令演示样例:
MVN R0,#0 。将 马上数0取反传送到寄存器R0中,完毕后R0=-1
3、CMP指令(比較)
CMP R1,#100 ;将寄存器R1的值与马上数100相减。并根 据结果设置CPSR的标志位
4、CMN指令(负数比較)
CMN R1。#100 ;将寄存器R1的值与马上数100相加。并依据 结果设置CPSR的标志位
5、TST指令(測试)
指令演示样例:
TST R1。#%1 。用于測试在寄存器R1中是否设置了最低位(%表 示二进制数)
TST R1,#0xffe ;将寄存器R1的值与马上数0xffe按位与。并依据 结果设置CPSR
;的标志位
6、TEQ指令(測试相等)
;的标志位
7、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)
8、ADC指令(带进位相加)
ADC R3,R7。R11 ; 加第四个字,带进位
9、SUB指令(相减)
指令演示样例:
SUB R0,R1,R2 ; R0 = R1 - R2
SUB R0,R1,#256 ; R0 = R1 - 256
SUB R0,R2,R3,LSL#1 ; R0 = R2 - (R3 << 1)
10、~~~~C指令
SUBS R0,R1,R2 ;R0 = R1 - R2 - 。C,并依据结果设置CPSR的进位标志位
11、R~~~~指令
指令演示样例:
R~~~~ R0。R1,R2 ; R0 = R2 – R1
R~~~~ R0。R1,#256 。 R0 = 256 – R1
R~~~~ R0。R2,R3。LSL#1 。 R0 = (R3 << 1) - R2
12、RSC指令(反向带进位减)
操作数1应是一个寄存器,操作数2能够是一个寄存器。被移位 的寄存器,或一个马上数。
该指令使用进位标志来表示借位,这样就能够做大于32位的减法。注意不要忘记设置S后缀来更改进位标志。该指令可用于有符号数或 无符号数的减法运算。
指令演示样例:
RSC R0,R1,R2 ;R0 = R2 – R1 - !
C
13、AND指令(逻辑位 与)
该指令经常使用于屏蔽操作数1的某些位。
指令演示样例:
AND R0,R0。#3 。该指令保持R0的0、1位,其余位清零。
14、ORR指令(逻辑位 或)
ORR R0,R0,#3 ;该指令设置R0的0、1位,其余位保持不变。
15、EOR指令(逻辑位 异或)
该指令经常使用于反转操作数1的某些位。
指令演示样例:
EOR R0。R0,#3 。该指令反转R0的0、1位,其余位保持不变。
16、BIC指令(位清零)
操作数2为32位的掩码。假设在掩码中设置了某一位。则清除这一位。未设置的掩码位保持不 变。
指令演示样例:
BIC R0。R0。#%1011 ;该指令清除R0中的位 0、1、和 3,其余的位保持不变。
三、乘法指令与乘加指令
乘法指令与乘加指令共同拥有下面6条:
1、MUL指令(相乘)
当中。操作数1和操 作数2均为32位的有符号数或无符号数。
指令演示样例:
MUL R0,R1。R2 ;R0 = R1 × R2
MULS R0,R1。R2 ;R0 = R1 × R2,同一时候设置CPSR中的相关条件标志位
2、MLA指令(带累加的相乘)
指令演示样例:
MLA R0,R1,R2,R3 ;R0 = R1 × R2 + R3
MLAS R0,R1。R2。R3 ;R0 = R1 × R2 + R3。同一时候设置CPSR中的相关条件标志位
3、SMULL指令
。R1 = (R2 × R3)的高32位
4、SMLAL指令
指令演示样例:
SMLAL R0,R1。R2,R3 ;R0 = (R2 × R3)的低32位 + R0
;R1 = (R2 × R3)的高32位 + R1
5、UMULL指令
。R1 = (R2 × R3)的高32位
6、UMLAL指令
对于目的寄存器Low。在指令运行前存放64位加数的低32位,指令运行后存放结果的低32位。对于目的寄存器High,在指令运行前存放64位加数的高32位。指令运行后存放结果的高32位。
指令演示样例:
UMLAL R0,R1,R2。R3 ;R0 = (R2 × R3)的低32位 + R0
。R1 = (R2 × R3)的高32位 + R1
四、程序状态寄存器訪问指令
1、MRS指令
MRS R0,SPSR 。传送 SPSR的内容到R0
2、MSR指令
<域>用于设置程序状态寄存器中须要 操作的位,32位的程序状态寄存器可分为4个域:
位[31:24]为条件位域。用f表示;
位[23:16]为状态位域。用s表示。
位[15:8] 为扩展位域,用x表示。
位[7:0] 为控制位域,用c表示;
该指令通经常使用于恢复或改变程序状态寄存器的内容,在使用时,一般要在MSR指令中指明将要操作的域。
指令演示样例:
MSR CPSR,R0 ;传送R0的内容到CPSR
MSR SPSR,R0 。传送R0的内容到SPSR
MSR CPSR_c,R0 ;传送R0的内容到SPSR。但只改动CPSR中的控制位域
五、载入/存储指令。ARM微处理器支持载入/存储指令用于在寄存器和存储器之间传送数据,载入指令用于将存储器中的数据传送到寄存器,存储 指令则完毕相反的操作。经常使用的载入存储指令例如以下:
1、LDR指令
LDR指令的格式为:
LDR{条件} 目的寄存器,<存储器地址>当程序计数器PC作为 目的寄存器时,指令从存储器中读取的字数据被当作目的地址。从而能够实现程序流程的跳转。该指令在程序设计 中比較经常使用,且寻址方式灵活多样,请读者认真掌握。
指令演示样例:
LDR R0,[R1] 。将存储器地址为R1的字数据读入寄存器R0。
LDR R0,[R1,R2] ;将存储器地址为R1+R2的字数据读入寄存器R0。
LDR R0,[R1,#8] ;将存储器地址为R1+8的字数据读入寄存器R0。
LDR R0,[R1,R2] !
;将存储器地址为R1+R2的字数据读入寄存器R0,并将新地 址
;R1+R2写入R1。
LDR R0,[R1,#8] !
;将存储器地址为R1+8的字数据读入寄存器R0。并将新地址 R1
。+8写入R1。
LDR R0,[R1]。R2 ;将存储器地址为R1的字数据读入寄存器R0,并将新地址 R1+
;R2写入R1。
LDR R0,[R1。R2。LSL#2]! ;将存储器地址为R1+R2×4的字数据读入寄存器R0,并
;将新地址R1+R2×4写入R1。
LDR R0。[R1],R2,LSL#2 ;将存储器地址为R1的字数据读入 寄存器R0。并将新地
。址R1+R2×4写入R1。
2、LDRB指令
当程序计数器PC作为目的寄存器时。指令从存储器中读取的字数据被当作目 的地址,从而能够实现程序流程的跳转。
指令演示样例:
LDRB R0,[R1] 。将存储器地址为R1的字节数据读入寄存器 R0,并将R0的高24
。位清零。
LDRB R0。[R1,#8] ;将存储器地址为R1+8的字节数据读入寄存器R0,并将 R0的
;高24位清零。
3、LDRH指令
LDRH R0。[R1,R2] ;将存储器地址为R1+R2的半字数据读入寄存器R0。并将 R0的
;高16位清零。
4、STR指令
STR R0,[R1,#8] ;将R0中的字数据写入以R1+8为地址的存储器中。
5、STRB指令
指令演示样例:
STRB R0。[R1] 。将寄存器R0中的字节数据写入以R1为地 址的存储器中。
STRB R0,[R1,#8] 。将寄存器R0中的字节数据写入以R1+8为地址的存 储器中。
6、STRH指令
指令演示样例:
STRH R0,[R1] 。将寄存器R0中的半字数据写入以R1为地址的 存储器中。
STRH R0。[R1。#8] ;将寄存器R0中的半字数据写入以R1+8 为地址的存储器中。
六、批量数据载入/存储指令。
ARM微处理器所支持批量数据载入/存储指令能够一次在一片连续的存储器单元和多个寄存器之间传送数据,批量载入指令 用于将一片连续的存储器中的数据传送到多个寄存器。批量数据存储指令则完毕相反的操作。
经常使用的载入存储指令例如以下:LDM(或STM)指令
LDM(或STM)指令的格式为:同一时候,该后缀还表 示传入或传出的是用户模式下的寄存器,而不是当前模式下的寄存器。
指令演示样例:
STMFD R13!,{R0,R4-R12,LR} ;将寄存器列表中的寄存器(R0,R4到R12,LR)存入堆栈。
LDMFD R13!, {R0,R4-R12。PC} ;将堆栈内容恢复到寄存器(R0,R4到R12,LR)。
七、数据交换指令
1、SWP指令
显然,当源寄存 器1和目的寄存器为同一个寄存器时。指令交换该寄存器和存储器的内容。
指令演示样例:
SWP R0,R1,[R2] 。将R2所指向的存储器中的字数据传送到R0,同一时候将R1 中的字数据传送到R2所指向的存储单元。
SWP R0,R0,[R1] ;该指令完毕将R1所指向的存储器中的字数 据与R0中的数据交换。
2、SWPB指令
显然。当源寄存器1和目的寄存器为同一个寄存器时。指令交换该寄存器和存储器的内容。
指令演示样例:
SWPB R0,R1,[R2] 。将R2所指向的存储器中的字节数据传送到 R0。R0的高24位清零。同一时候将R1中的低8位数据传送到R2所指向的存储单元。
SWPB R0,R0,[R1] ;该指令完毕将R1所指向的存储器中的 字节数据与R0中的低8位数据交换。
八、移位指令
1、LSL(或ASL)
当中,操作数能够是通用寄存器,也能够是马上数(0~31)。
操作演示样例
MOV R0, R1, LSL #2 ;将R1中的内容左移两位后传送到R0 中。
2、LSR
操作演示样例:
MOV R0, R1, LSR #2 ;将R1中的内容右移两位后传送到R0 中,左端用零来填充。
3、ASR
当中。操作数能够是通用寄存器。也能够是立 即数(0~31)。
操作演示样例:
MOV R0, R1, ASR #2 。将R1中的内容右移两位后传送到R0 中,左端用第31位的值来填充。
4、ROR
操作演示样例:
MOV R0, R1, ROR #2 。将R1中的内容循环右移两位后传送到R0 中。
5、RRX
操作演示样例:
MOV R0, R1, RRX #2 。将R1中的内容进行带扩展的循环右移两位 后传送到R0中。
九、协处理器指令
1、CDP指令
CDP指令用于ARM处理器通知ARM协处理器运行特定的操作,若协处理器不能成功完毕特定的操作,则产生没有定义指令异常。当中协处理器操作码1和协处理 器操作码2为协处理器将要运行的操作。目的寄存器和源寄存器均为协处理器的寄存器,指令不涉及ARM处理器的寄存器和存储器。
指令演示样例:
CDP P3,2。C12,C10,C3,4 。该指令完毕协处理器P3的初始化
2、LDC指令
当中,{L}选项表示指 令为长读取操作,如用于双精度数据的传输。
指令演示样例:
LDC P3。C4,[R0] ;将ARM处理器的寄存器R0所指向的存储器中的字数 据传送到协处理器P3的寄存器C4中。
3、STC指令
STC P3,C4,[R0] 。将协处理器P3的寄存器C4中的字数据传送到ARM处理 器的寄存器R0所指向的存储器中。
4、MCR指令
MCR指令的格式为:
MCR P3。3,R0。C4,C5。6 ;将ARM处理器寄存器R0中的数据传送到协处 理器P3的寄存器C4和C5中。
5、MRC指令
MRC P3。3,R0,C4,C5,6 ;该指令将协处理器P3的寄存器中的数据传送到 ARM处理器寄存器中。
十、异常产生指令
1、SWI指令
指令演示样例:
SWI 0x02 。该指令调用操作系统编号位02的系统例程。
2、BKPT指令
ARM汇编伪指令
伪指令在源程序中的作用是为完毕汇编程序作各种准备工作的,这些伪指令仅在汇编过程中起作用。一旦汇编结束,伪指令的使命就完 成。
在ARM 的汇编程序中。有例如以下几种伪指令:符号定义伪指令、数据定义伪指令、汇编控制伪指令、宏指令以及其它伪指令。
一、符号定义(Symbol Definition)伪指令
常见的符号定义伪指令有例如以下几种:
— 用于定义全局变量的GBLA 、GBLL 和GBLS 。
— 用于定义局部变量的LCLA 、LCLL 和LCLS 。
— 用于对变量赋值的SETA 、SETL 、SETS 。
— 为通用寄存器列表定义名称的RLIST 。
Test2 SETL {TRUE} ;将该变量赋值为真。
GBLS Test3 ; 定义一个全局的字符串变量,变量名为 Test3。
Test3 SETS “Testing” 。将该变量赋值为"Testing”。
2、LCLA、LCLL 和LCLS
语法格式:
LCLA (LCLL 或 LCLS )局部变量名
LCLA 、LCLL 和LCLS 伪指令用于定义一个ARM 程序中的局部变量,并将其初始化。当中:
LCLA伪指令用于定义一个局部的数字变量。并初始化为0 ;
LCLL伪指令用于定义一个局部的逻辑变量。并初始化为F(假);
LCLS伪指令用于定义一个局部的字符串变量,并初始化为空;
以上三条伪指令用于声明局部变量。在其作用范围内变量名必须唯一。
使用演示样例:
LCLA Test4 ; 声明一个局部的数字变 量,变量名为Test4。
Test3 SETA 0xaa 。 将该变量赋值为0xaa。
LCLL Test5 ; 声明一个局部的逻辑变 量,变量名为Test5。
Test4 SETL {TRUE} ;将该变量赋值为真。
LCLS Test6 。 定义一个局部的字 符串变量,变量名为Test6。
Test6 SETS “Testing” 。将该变量赋值为 "Testing”。
3、SETA、SETL 和SETS
语法格式:
变量名 SETA (SETL 或 SETS )表达式
伪指令 SETA 、SETL 、SETS 用于给一个已经定义的全局变量或局部变量赋值。
SETA伪指令用于给一个数学变量赋值;
SETL伪指令用于给一个逻辑变量赋值;
SETS伪指令用于给一个字符串变量赋值;
当中,变量名为已经定义过的全局变量或局部变量,表达式为将要赋给变量的值。
使用演示样例:
LCLA Test3 。 声明一个局部的数字变量,变量名为 Test3。
Test3 SETA 0xaa 。 将该变量赋值为0xaa。
LCLL Test4 ; 声明一个局部的逻辑变量。变量名为 Test4。
Test4 SETL {TRUE} ;将该变量赋值为真。
4 、RLIST
语法格式:
名称 RLIST { 寄存器列表 }
RLIST伪指令可用于对一个通用寄存器列表定义名称,使用该伪指令定义的名称可在ARM 指令 LDM/STM中使用。在LDM/STM指令中,列表中的寄存器訪问次序为依据寄存器的编号由低到高。而与列表中的寄存器排列次序无关。
使用演示样例:
RegList RLIST {R0-R5 。R8 ,R10} 。将寄存器列表名称定义为 RegList 。可在ARM指令LDM/STM中通过该名称訪问寄存器列表。
二、数据定义(Data Definition)伪指令
— DCW(DCWU)用于分配一片连续的半字存储单元并用指定的数据初始化。
— DCD (DCDU)用于分配一片连续的字存储单元并用指定的数据初始化。
— DCFD(DCFDU)用于为双精度的浮点数分配一片连续的字存储单元并用指定的数据初始化。
— DCFS(DCFSU)用于为单精度的浮点数分配一片连续的字存储单元并用指定的数据初始化。
— DCQ(DCQU)用于分配一片以8字节为单位的连续的存储单元并用指定的数据初始化。
— SPACE 用于分配一片连续的存储单元。
— MAP 用于定义一个结构化的内存表首地址。
— FIELD 用于定义一个结构化的内存表的数据域。
DCB 也可用“=”取代。
使用演示样例:
Str DCB “This is a test” ;分配一片连续的字节存储单元并初始化。
2、DCW(或DCWU)
语法格式:
标号 DCW (或DCWU) 表达式
DCW(或DCWU)伪指令用于分配一片连续的半字存储单元并用伪指令中指定的表达式初始化。
当中。表达式能够为程序标号或数字表达式。
用DCW分配的字存储单元是半字对齐的。而用DCWU分配的字存储单元并不严格半字对齐。
使用演示样例:
DataTest DCW 1 ,2 。3 。分配一片连续的半字存储单元并初始化。
3、DCD(或DCDU)
语法格式:
标号 DCD(或DCDU) 表达式
DCD(或DCDU)伪指令用于分配一片连续的字存储单元并用伪指令中指定的表达式初始化。当中。表达式能够为程序标号或数字表达式。DCD也可 用"&” 取代。
用DCD分配的字存储单元是字对齐的。而用DCDU分配的字存储单元并不严格字对齐。
使用演示样例:
DataTest DCD 4 ,5 ,6 。分配一片连续的字存储单元并初始化。
4、DCFD(或DCFDU)
语法格式:
标号 DCFD(或DCFDU) 表达式
DCFD(或DCFDU)伪指令用于为双精度的浮点数分配一片连续的字存储单元并用伪指令中指定的表达式初始化。每一个双精度的浮点数占领两个字单元。
用 DCFD分配的字存储单元是字对齐的。而用DCFDU分配的字存储单元并不严格字对齐。
使用演示样例: FDataTest DCFD 2E115 ,-5E7 ;分配一片连续的字存储单元并初始化 为指定的双精度数。
5、DCFS(或DCFSU)
语法格式:
标号 DCFS(或DCFSU) 表达式
DCFS(或DCFSU)伪指令用于为单精度的浮点数分配一片连续的字存储单元并用伪指令中指定的表达式初始化。每一个单精度的浮点数占领一个字单元。用 DCFS分配的字存储单元是字对齐的。而用DCFSU分配的字存储单元并不严格字对齐。
使用演示样例:
FDataTest DCFS 2E5 ,-5E -7 。分配一片连续的字存储单元并初始化为 指定的单精度数。
6、DCQ(或DCQU)
语法格式:
标号 DCQ(或DCQU) 表达式
DCQ(或DCQU)伪指令用于分配一片以8个字节(双字)为单位的连续存储区域并用伪指令中指定的表达式 初始化。 用DCQ分配的存储单元是字对齐的,而用DCQU 分配的存储单元并不严格字对齐。
使用演示样例:
DataTest DCQ 100 ;分配一片连续的存储单元并初始化为指定的值。
7、SPACE
语法格式:
标号 SPACE 表达式
SPACE伪指令用于分配一片连续的存储区域并初始化为0 。当中,表达式为要分配的字节数。
SPACE也可用“ % ”取代。
使用演示样例:
DataSpace SPACE 100 ;分配连续100字节的存储单元并初始化为0 。
8、MAP
语法格式:
MAP 表达式 { 。基址寄存器 }
MAP伪指令用于定义一个结构化的内存表的首地址。MAP也可用“^” 取代。
表达式能够为程序中的标号或数学表达式,基址寄存器为可选项,当基址寄存器选项不存在时,表达式的值即为内存表的首地址。当该选项存在时。内存表的首地址 为表达式的值与基址寄存器的和。
MAP伪指令通常与FIELD伪指令配合使用来定义结构化的内存表。
使用演示样例:
MAP 0x100 ,R0 。定义结构化内存表首地址的值为0x100+R0 。
9、FILED
语法格式:
标号 FIELD 表达式
FIELD伪指令用于定义一个结构化内存表中的数据域。FILED 也可用“#” 取代。
表达式的值为当前数据域在内存表中所占的字节数。
FIELD伪指令常与MAP伪指令配合使用来定义结构化的内存表。MAP伪指令定义内存表的首地址,FIELD伪指令定义内存表中的各个数据域,并能够为 每一个数据域指定一个标号供其它的指令引用。
注意MAP和FIELD伪指令仅用于定义数据结构。并不实际分配存储单元。
使用演示样例:
MAP 0x100 。 定义结构化内存表首地址的值为0x100。
A FIELD 16 。 定义A的长度为16字节。位置为0x100。
B FIELD 32 ; 定义B的长度为32字节,位置为0x110。
S FIELD 256 ;定义S的长度为256字节,位置为0x130。
三、汇编控制(Assembly Control)伪指令
汇编控制伪指令用于控制汇编程序的运行流程,经常使用的汇编控制伪指令包含下面几条:
— IF 、ELSE 、ENDIF
— WHILE 、WEND
— MACRO 、MEND
— MEXIT
当IF后面的逻辑表达式为真。则运行指令序列1 。否则运行指令序列2 。当中。ELSE及指令序列2能够没有,此时。当IF后面的逻辑表达式为真。则运行指令序列1 ,否则继续运行后面的指令。
IF 、ELSE 、ENDIF伪指令能够嵌套使用。
使用演示样例:
GBLL Test 。声明一个全局的逻辑变量,变量名为Test
IF Test = TRUE
指令序列 1
ELSE
指令序列 2
ENDIF
2、WHILE、WEND
语法格式:
WHILE 逻辑表达式
指令序列
WEND
WHILE 、WEND伪指令能依据条件的成立与否决定是否循环运行某个指令序列。当WHILE后面的逻辑表达式为真,则运行指令序列,该指令序列运行完成后。再推断 逻辑表达式的值,若为真则继续运行。一直到逻辑表达式的值为假。
WHILE 、WEND伪指令能够嵌套使用。
使用演示样例:
GBLA Counter ; 声明一个全局的数学变量。变量名为Counter
Counter SETA 3 。由变量Counter 控制循环次数
……
WHILE Counter < 10
指令序列
WEND
MEXIT用于从宏定义中跳转出去。
四、其它经常使用的伪指令
— ROUT
属性字段表示该代码段(或数据段)的相关属性,多个属性用逗号分隔。经常使用的属性例如以下:
— CODE 属性:用于定义代码段,默觉得READONLY 。
— DATA 属性:用于定义数据段,默觉得READWRITE 。
— READONLY 属性:指定本段为仅仅读,代码段默觉得READONLY 。
— READWRITE 属性:指定本段为可读可写。数据段的默认属性为READWRITE 。
— ALIGN 属性:使用方式为ALIGN表达式。在默认时,ELF(可运行连接文件)的代码段和数据段是按字对齐的,表达式的取值范围为0~31,对应的对齐方式为2 表达式次方。
— COMMON 属性:该属性定义一个通用的段。不包括不论什么的用户代码和数据。
各源文件里同名的COMMON段共享同一段存储单元。
一个汇编语言程序至少要包括一个段,当程序太长时,也能够将程序分为多个代码段和数据段。
使用演示样例:
AREA Init ,CODE ,READONLY ; 该伪指令定义了一个代码段,段 名为Init ,属性为仅仅读。
2、ALIGN
语法格式:
ALIGN { 表达式 { 。偏移量 }}
ALIGN伪指令可通过加入填充字节的方式,使当前位置满足一定的对齐方式。当中。表达式的值用于指定对齐方式,可能的取值为2的幂。如1 、2 、4 、8 、16 等。
若未指定表达式。则将当前位置对齐到下一个字的位置。
偏移量也为一个数字表达式,若使用该字段,则当前位置的对齐方式为:2的表达式次幂+偏移 量。
使用演示样例:
AREA Init 。CODE ,READONLY ,ALIEN=3 ;指定后面的指令为8 字节对齐。
指令序列
END
3、CODE16、CODE32
语法格式:
CODE16(或CODE32)
CODE16伪指令通知编译器,其后的指令序列为16位的Thumb指令。
CODE32伪指令通知编译器。其后的指令序列为32位的ARM指令。
若在汇编源程序中同一时候包括ARM指令和Thumb指令时,可用CODE16伪指令通知编译器其后的指令序列为16位的Thumb指令。CODE32伪指令 通知编译器其后的指令序列为32位的ARM指令。因此。在使用ARM指令和Thumb指令混合编程的代码里,可用这两条伪指令进行切换,但注意他们仅仅通知 编译器其后指令的类型,并不能对处理器进行状态的切换。
使用演示样例:
AREA Init 。CODE ,READONLY ……
CODE32 ; 通知编译器其后的指令为32位的 ARM指令
LDR R0 。=NEXT+1 ;将跳转地址放入寄存器R0
BX R0 。 程序跳转到新的位置运行, 并将处理器切换到Thumb工作状态
……
CODE16 ; 通知编译器其后的指令为16位的 Thumb指令
NEXT LDR R3,=0x3FF
……
END 。
4、ENTRY
语法格式:
ENTRY
ENTRY伪指令用于指定汇编程序的入口点。
在一个完整的汇编程序中至少要有一个ENTRY(也能够有多个,当有多个ENTRY时。程序的真正入口点由链 接器指定)。但在一个源文件中最多仅仅能有一个ENTRY(能够没有)。
使用演示样例:
AREA Init ,CODE ,READONLY
ENTRY ; 指定应用程序的入口点
……
5、END
语法格式:
END
END伪指令用于通知编译器已经到了源程序的结尾。
使用演示样例:
AREA Init ,CODE ,READONLY
……
END ;指定应用程序的结尾
6、EQU
语法格式:
名称 EQU 表达式 { ,类型 }
EQU伪指令用于为程序中的常量、标号等定义一个等效的字符名称,类似于C语言中的#define 。当中EQU可用“*”取代。名称为EQU伪指令定义的字符名称。当表达式为32位的常量时,能够指定 表达式的数据类型,能够有下面三种类型:
CODE16 、CODE32 和DATA
使用演示样例:
Test EQU 50 。 定义标号Test 的值为50。
Addr EQU 0x55 ,CODE32 。 定义Addr的值为0x55 。且该处为32位的ARM指令。
7、EXPORT(或GLOBAL)
语法格式:
EXPORT 标号 {[WEAK]}
EXPORT伪指令用于在程序中声明一个全局的标号,该标号可在其它的文件里引用。EXPORT 可用GLOBAL取代。标号在程序中区分大写和小写。[WEAK] 选项声明其它的同名标号优先于该标号被引用。
使用演示样例:
AREA Init ,CODE ,READONLY
EXPORT Stest 。声明一个可全局引用的标号Stest
END
END
9、EXTERN
语法格式:
EXTERN 标号 {[WEAK]}
EXTERN伪指令用于通知编译器要使用的标号在其它的源文件里定义,但要在当前源文件里引用,假设当前源文件实际并未引用该标号,该 标号就不会被增加到当前源文件的符号表中。标号在程序中区分大写和小写, [WEAK] 选项表示当全部的源文件都未定义这样一个标号时。编译器也不给出错误信息。在多数情况下将该标号置为0 ,若该标号为B或BL指令引用,则将B或BL指令置为NOP操作。
使用演示样例:
AREA Init 。CODE 。READONLY
EXTERN Main ;通知编译器当前文件要引用标号Main,但Main在其它源文件里定 义。
END
10、GET(或INCLUDE)
语法格式:
GET 文件名称
GET伪指令用于将一个源文件包括到当前的源文件里。并将被包括的源文件在当前位置进行汇编处理。可 以使用INCLUDE取代GET。
汇编程序中经常使用的方法是在某源文件里定义一些宏指令。用EQU定义常量的符号名称,用MAP和FIELD定义结构化的数据类型,然后用GET伪指令将这个 源文件包括到其它的源文件里。
用法与C 语言中的"include” 相似。
GET伪指令仅仅能用于包括源文件。包括目标文件须要使用INCBIN伪指令
使用演示样例:
AREA Init ,CODE ,READONLY
GET a1.s 。 通知编译器当前源文件包括源文件a1.s
GET C:\a2.s ; 通知编译器当前源文件包括源文件C:\a2.s
END
在程序中未使用该伪指令时,局部 变量的作用范围为所在的AREA,而使用ROUT后,局部变量的作为范围为当前ROUT和下一个ROUT之间。
3. BNE 与 BEQ
TST R0, #0X8
BNE SuspendUp 。BNE指令是“不相等(或不为0)跳转指令”:
LDR R1,#0x00000000
先进行and运算。假设R0的第四位不为1,则结果为零。则设置zero=1(继续以下的LDR指令);
否则。假设R0的第四位为1。zero=0(跳到SuspendUp处运行)。
tst 和bne连用: 先是用tst进行位与运算。然后将位与的结果与0比較,假设不为0,则跳到bne紧跟着的标记(如bne sleep,则跳到sleep处)。
tst 和beq连用: 先是用tst进行位与运算。然后将位与的结果与0比較。假设为0,则跳到beq紧跟着的标记(如bne AAAA,则跳到AAAA处)。
原文:http://www.cnblogs.com/mengfanrong/p/5239930.html