ARM教材第5章ARM汇编程序设计(共16页).doc

上传人:飞****2 文档编号:16790019 上传时间:2022-05-18 格式:DOC 页数:16 大小:64KB
返回 下载 相关 举报
ARM教材第5章ARM汇编程序设计(共16页).doc_第1页
第1页 / 共16页
ARM教材第5章ARM汇编程序设计(共16页).doc_第2页
第2页 / 共16页
点击查看更多>>
资源描述

《ARM教材第5章ARM汇编程序设计(共16页).doc》由会员分享,可在线阅读,更多相关《ARM教材第5章ARM汇编程序设计(共16页).doc(16页珍藏版)》请在得力文库 - 分享文档赚钱的网站上搜索。

1、精选优质文档-倾情为你奉上第5章 ARM汇编程序设计ARM编译器,如ADS集成开发环境,一般都支持汇编语言的程序设计。本章介绍ARM程序设计的一些基本概念,如ARM汇编语言的伪指令、汇编语言的语句格式和汇编语言的程序结构等到,并在些基础上介绍一些常用的ARM汇编子程序的设计。4.1 ARM伪指令ARM汇编程序由汇编指令、伪指令和宏指令组成,伪指令不介汇编指令那样在处理器的运行期间执行,而是在汇编器对汇编程序进行汇编时处理。宏是一段独立的汇编程序代码,它是通过伪指令定义的,在程序中宏指令即调用宏指令。当程序被汇编时,汇编程序对每个宏调用进行展开,用宏定义代汇编程序中的宏指令。由于指令也发球汇编伪

2、指令的一部分,因此本书将宏指令放在汇编伪一起介绍。与单片机汇编程序设计一样,在ARM汇编语言程序里,有一些特殊指令助记符,这些助记符与指令系统的助记符不同,它们没有相对应的操作友码,通常称这些特殊指令助记符为伪指令,它们所完成的操作称为伪操作。伪指令在源程序中的作用是为完成汇编程序做各种准备工作,这些伪指令仅在汇编过程中起作用,一旦汇编结束,伪指令的使命就完成了。在ARM的光荣称号程序中,有如下几种伪指令:符号定义伪指令、数据定义伪指令、汇编控制伪指令以及其他伪指令。4.1.1 符号定义(Symbol Definition)伪指令符号定义伪指令用于定义ARM汇编程序中的变量、对变量赋值以及定义

3、寄存器的别名等。常见的符号定义伪指令有以下几种:l 用于宝玉局变量的GBLA、GBLL和GBLS;l 用于定义局部变量的LCLA、SETL、LCLS;l 用于对变量赋值的SETA、SETL、SETS;l 为通用寄存器列表定义名称的RLIST;l 为一个协处理器的寄存器定义名称的伪指令CN;l 为一个协处理器定义名称的伪指令CP;l 为一个CFP寄存器定义名称的伪指令DN和SN;l 为一个FPA浮点寄存器定义名称的伪指令FN。1. GBLA、GBLL和GBLS语法格式:GBLA(GBLL或GBLS) 全局变量名GBLA、GBLL和GBLS伪指令用于定义一个ARM程序中的全局变量,并半其初始化。其

4、中:GBLA伪指令用于定义一个全局的数字变量,并初始化为0;GBLL伪指令用于定义一个全局的逻辑变量,并初始化为F(假);GBLS伪指令用于定义一个全局的字符串变量,并初始化为空。由于以上三条伪指令用于定义全局变量,因此在整个程序范围内变量名必须唯一。使用示例:GBLA Number1 ;定义一个全局的数字变量,变量名为Number1Number1 SETA 0xaa ;将Number1变量赋值为0xaaGBLL True1 ;定义一个全局的逻辑变量,变量名为True1True1 SETLTRUE;将True1变量赋值为真GBLS String1 ;定义一个全局的字符串变量,变量名为Strin

5、g1 String1 SETS“Testing” ;将String1变量赋值为“Testing”2. LCLA、LCLL和 LCLS语法格式:LCLA(LCLL或LCLS) 局部变量名LCLA、LCLL和LCLS伪指令用于定义一个ARM程序中的局部变量,并将其初始化,其中:l LCLA伪指令用于定义一个局部的数字变量,并初始化为0;l LCLL伪指令用于定义一个局部的逻辑变量,并初始化为F(假);l LCLS伪指令用于定义一个局部的字符串变量,并初始化为空。以上三条伪指令用于声明局部变量,在其作用范围内变量名必须唯一。使用示例:LCLA Number2 ;声明一个局部的数字变量,变量名为Num

6、ber2 Number2 SETA 0xaa ;将Number2 变量赋值为0xaaLCLL Logic2 ;声明一个局部的逻辑变量,变量名为Logic2Logic2 SETLTRUE ;将Logic2变量赋值为真LCLS String2 ;定义一个局部的字符串变量,变量名为String2String2 SETS“Testing” ;将String2变量赋值为“Testing”3. SETA、SETL和SETS语法格式:变量名 SETA(SETL或SETS) 表达式伪指令SETA、SETL、SETS用于给一个已经定义的全局变量或局部变量赋值。l SETA伪指令用于给一个数学变量赋值;l SET

7、L伪指令用于给一个逻辑变量赋值;l SETS伪指令用于给一个字符串变量赋值。其中,变量名是已经定义地的全局变量或局部变量,表达式是将要赋给变量的值。使用示例:LCLA Number3 ;声明一个局部的数字变量,变量名为Number3Number3 SETA 0xaa ;将Number3变量赋值为0xaaLCLL Logic3 ;声明一个局部的逻辑变量,变量名为Logic3Logic3 SETLTRUE了 ;将Logic3变量赋值为真4. RLIST语法格式:名称 RLIST 寄存器列表RLIST伪指令可用于对一个通用寄存器列表定义名称,使用该伪指令定义的名称可在ARM指令LDM/STM中使用。

8、在LDM/STM指令中,列表中的寄存器访问次序为根据寄存器的编号由低到高,而与列表中的寄存器排列次序无关。使用示例:RegList RLIST R0-R5,R8,R10 ;将寄存器列表名称定义为RegList,可在ARM ;指令LDM/STM中通过该名称访问寄存器列表STMD SP!,RegList ;保存寄存器列表RegList 到堆栈由于在编程过程中,协处理器汇编伪指令使用的情况比较少,因此本书不对CP、DN和SN以及FN等协处理器伪指令进行介绍。4.1.2 数据定义(DATA Definition)伪指令数据定义伪指令一般用于为特定的数据分配存储单元,同时可完成对已分配存储单元的初始化。

9、常见的数据定义伪指令有如下几种:l DCB用于分配一片连续的字节存储单元并用指定的数据初始化;l DCW(DCWU)用于分配一片连续的半字节存储单元并用指定的数据初始 化;l DCD(DCDU)用于分配一片连续的字存储单元并用指定的数据初始 化;l DCFS(DCFSU)用于为双精度的浮点数分配一片连续的字存储单元并用指定的数据初始化;l DCQ(DCQU)用于分配一片以8字节个为单位的连续的存储单元并用指定的数据初始化;l DCDO用于分配一段字的内存单元,将每个单元的内容初始化为该单元相对于基址寄存器的偏移量;l DCI用于分配一段字的内丰单元,并用单精度的浮点数据初始化,指定内存单元存放

10、的是代码,而不是数据;l SPACE用于分配一片连续的存储单元;l MAP用于定义一个结构化的内存表首地址;l FIELD用于定义一个结构化的内在表的数据域;l LTORG用于声明一个文字池。1.语法格式:标号 DCB 表达式DCB伪指令用于分配一片连续的字节存储单元并用伪指令中的表达式初始化。其中,表达式可以为0255的数字或字符吕,DC也可用“=”代替。使用示例:标号 DCB 表达式DCB伪指令用于分配一片连续的字节存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为0255的数字或字符串,DCB也可用“=”代替。使用示例:String DCB “This is a test!” ;

11、分配一片连续的字节存储单元并初始化Parameter DCB 0x33,0x44,0x55DCB -1,-2,0,1,2, ;分配一片连续的字节存储单元并初始化2. DCW(或DCWU)语法格式:标号 DCW(或DCWU) 表达式DCW(或DCWU)伪指令用于为双精度的浮点数分配一片连续的字节存储单元,并用伪指令中指定的表达式初始化。每个双精度的浮点数占据两个字单元,用DCFD分配的字存储单元是字对齐的,而用DCFDU分配的字存储单元并不严格字对齐。使用示例:Fdata DCFD 0,2E115。-5E7 ;分配一片连续的字存储单元并初始化为指定的双精度数5. DCFS(或DCFSU) 语法格

12、式:标号 DCFS(或DCFSU) 表达式DCFS(或DCFSU)伪指令用于为单精度的浮点数分配一片连续的字存储单元,并用伪指令中指定的表达式初始化。每个单精度的浮点数占据一个字单元,用DCFS分配的字存储单元是字对齐的,而用DCFSU分配的字存储单元并不严格对齐。使用示例:Sdata DCFS 1,2E5,-5E7 ;分配一片连续的字存储单元并初始化为指定的单精度数6. DCQ(或DCQU)语法格式:标号 DCQ(或DCQU) 表达式DCQ(或DCQU)伪指令用于分配一片以8个字节为单位的连续存储区域,并用伪指令中指定的表达式初始化。用DCQ分配的存储单元是字对齐的,而用DCQU分配的存储单

13、元并不严格对齐。使用示例:Data DCQ 100,1000 ;分配一片连续的存储单元并初始化为指定的值7. DCDO语法格式:标号 DCDO 表达式DCDO用于分配一段字内存单元,并将每个单元的内容初始化为该单元相对于表态基址寄存器的偏移量。DCDO伪指令作为表态基址寄存器R9的偏移量分配内在单元,该指令需要内在字对齐。使用示例:IMPORT externsysData DCDO externnsys ;分配32位的字单元,其值勤为标号externnsys 基于R9的偏移量8. DCI语法格式:标号 DCI 表达式DCI用于分配一段字节的内在单元,并用伪指令中指定的表达式初始化。指定内在单元

14、存放的是代码而不是数据,在Thumb代码中,DCI分配的是半字节的内存代码单元。使用示例:MACRO ;这个宏指令将指令 newinstr Rd,Rm定义为相应的机器指令Newinstr $Rd,$RmDCI 0Xe15f0f10:CR:(8$Rd:SHL:12):OR;$Rm ;这是存放的是指令MEND9. SPACE语法格式:标号 SPACE 表达式SPACE伪指令用于分配一片连续的存储区域并初始化为0。其中,表达式是要分配的字节数,SPACE也可用“%”代替。使用示例:DataSpace SPACE 1000 ;分配连续1000B的存储单元并初始化为010. MAP语法格式:MAP 表达

15、式,基址寄存器MAP伪指令用于定义一个结构化的内存表的首地址,MAP也可用“”代替。表达式可以是程序中的标号或数学表达式,基址寄存器为可选项,当基址寄存器选项不存在时,表达式的值即为内存表的首地址,当该选项存在时,内存表的首地址为表达式的值与基址寄存器的和。MAP伪指令通常与FIELD伪指令配合使用来定义结构化的内存表。使用示例:MAP 0x10,R0 ;定义结构化内存表首地址的值为0x10+R011. FIELD语法格式:标号 FIELD 表达式FIELD伪指令用于定义一个结构化内存表中的数据域。FIELD也可用“#”代替,表达式的值是当前数据域在内存表中所占的字节数。FIELD伪指令常与M

16、AP伪指令配合使用来定义结构化的内存表,MAP伪指令定义内存表的首地址,FIELD伪指令定义内存表中的各个数据域,并可以为每个数据域指定一个标号供其他的指令引用。注意MAP和FIELD伪指令仅用于定义数据结构,并不实际分配存储单元。使用示例:MAP 0x100 ;定义结构化内存表首地址的值为为0x100A FIELD 16 ;定义A的长度为16B,位置为0x100B FIELD 32 ;定义B的长度为32B,位置为0x110(0x100的存放了 ;16个字节) S FIELD 256 ;定义S的长度为256B,位置为0x130(0x100后存放了 ;32个字节)12. LTORG语法格式:LT

17、ORGLTORG用于声明一个文字池。在使用LDR伪指令时,要在适当的地址加入LTORG声明文字池,这样就会把要加载的数据保存在文字池中,再用ARM的加载指令读出数据。如果没有使用LTORG声明文字池,则汇编器会在程序末尾自动声明,使用LTORG声明文字池的目的可以在程序代码的任何位置存储加载的数据。使用示例:LDR R0,=0x12345ADD R1,R1,R0MOV PC,LRLTORG ;声明文字池,此处存放0x12345 ;其他代码4.1.3 汇编控制(Assembly Control)伪指令汇编控制伪指令用于控制汇编程序的执行流程,包括以下常用的汇编控制伪指令:l IF、ELSE、EN

18、DIFl WHILE、WENDl MACRO、MENDl MEXIT1.IF、ELSE、ENDIF语法格式:IF 逻辑表达式 指令序列1ELSE 指令序列2ENDIFIF、ELSE、ENDIF伪指令能够根据条件的成立与否决定是否执行某个指令序列。若IF后面的逻辑表达式为真,则执行指令序列1,否则执行指令序列2。其中,ELSE及指令序列2可以没有,此时,若IF后面的逻辑表达式为真,则执行指令序列1,否则继续执行后面的指令。IF、ELSE、ENDIF伪指令可以嵌套使用。使用示例:GBLS Version ;定义一个全局的字符串变量,变量名为VersionIF Version=“V1” 指令序列1E

19、LSE 指令序列2ENDIF2. WHILE、WEND语法格式:WHILE 逻辑表达式 指令序列WENDWHILE、WEND伪指令能够根据条件的成立与否决定是否循环执行某个指令序列。若WHILE后面的逻辑表达式为真,则执行指令序列,该指令序列执行完毕后,再判断逻辑表达式的值,若为真则继续执行,一直到逻辑表达式的值为假。WHILE、WEND伪指令可以嵌套使用。使用示例如下:GBLA Counter ;声明一个全局的的数字变量,变量名为Counter,作为循环计数器WHILE Counter”、“=”、“/=”“”运算符以X和Y表示两个逻辑表达式,以上的运算符代表运算如下:X=Y 表示X等于YXY

20、 表示X大于YX=Y 表示X大于等于YX=Y 表示X小于等于YX/=Y 表示X不等于YXY 表示X不等于Yl “LAND”、“LOR”、“LNOT”及“LEOR”运算符以X和Y表示两个逻辑表达式,以上的逻辑运算符代表运算如下:X:LAND:Y 表示将X和Y作逻辑与的操作X:LOR:Y 表示将X和Y作逻辑或的操作:LNOT:Y 表示将将Y作逻辑非的操作X:LEOR:Y 表示将X和Y按位作逻辑异或的操作3) 字符串表达式及运算符字符串表达式一般由字符串常量、字符串变量、运算符和括号构成,编译器所支持的字符串最大长度为512B。常用的字符串表达式相关的运算符如下:l LEN运算符LEN运算符返回字的

21、长度(字符数),以X表示字符串表达式,其语法格式如下:LEN:Xl CHR运算符CHR运算符将0255之间的整数转换为一个字符,以M表示某一个整数,其语法格式如下:CHR:Ml STR运算符STR运算符将一个数字表达式或逻辑表达式转换为一个字符串。对于数字表达式,STR运算符将其转换为一个十六进制格式的字符串;对于逻辑表达式,STR运算符将其转换为字符串T或F,其语法格式如下:STR:X其中,X为一个数字表达式或逻辑表达式。l LEFT运算符LEFT运算符返回某个字符串左端的规定长度的一个子串,其语法格式如下:X:LEFT:Y其中,X为源字符串,Y为一个整数,表示要返回的字符个数。l CC运算

22、符CC运算符用于将两个字符串连接成一个字符串,其语法格式如下:X:CC:Y其中,X为源字符串1,Y为源字符串2,CC运算符将Y连接到X的后面。4) 与寄存器和程序计数器(PC)相关的表达式及运算符如下:l BASE运算符BASE运算符返回基于寄存器的表达式中寄存器的编号,其语法格式如下:BASE:X其中,X为与寄存器相关的表达式。l INDEX运算符INDEX运算符返回基于寄存器的表达式中相对于其基址寄存器的偏移量,其语法格式如下:INDEX:X其中,X为与寄存器相关的表达式。55) 其他常用运算符l ?运算符?运算符返回基某代码行所生成的可执行代码的长度,例如:?X返回定义符号X的代码行所生

23、成的可执行代码的字节数。l DEF运算符DEF运算符判断是否定义某个符号,例如:DEF:X如果符号X已经定义,则结果为真,否则为假。4.3 汇编语言程序设计4.3.1 汇编语言的程序结构在ARM(Thumb)汇编语言程序中,以程序段为单位组织代码。段是相对独立的指令或数据序列,具有特定的名称。段可以分为代码段和数据段,代码段的内容为执行代码,数据段存放代码运行时需要用到的数据。一个汇编程序至少应该有一个代码段,当程序较长时,可以分割为多个代码段和数据段,多个段在程序编译外链接时最终形成一个可执行的映象文件。可执行映象文件通常由以下几部分构成:l 一个或多个代码段,代码段的属性为可读/写。l 零

24、个或我个包含初始化数据的数据段,数据段的属性为可读/写。l 零个或我个不包含初始化数据的数据段,数据段的属性为可读/写。链接器系统默认或用户设定的规则,将各个段安排在存储器中的相应位置。因此源程序中段之间的相对位置与可执行的映象文件中段的相对位置一般不会相同。以下是一个汇编语言源程序的基本结构:AREA Init,CODE,READONLYENTRYStartLDR R0,=0x3FF5000LDR R1,0xFFSTR R1,R0LDR R0,=0x3FF5008LDR R1,0x01STR R1,R0END在汇编语言程序中,用AREA伪指令定义一个段,并说明所定义段的相关属性,本例定义一个

25、名为Init的代码段,属性为只读。ENTRY伪指令标识程序的入口点,接下来是指令序列,程序的末尾为END伪指令,该伪指令告诉编译器源文件的结束,每一个汇编程序段都必须有一条END伪指令,指示代码段的结束。4.3.2 汇编语言的子程序调用在ARM汇编语言程序中,子程序的调用一般都是通过BL指令来实现的。在程序中,使用指令“BL子程序名”即可完成子程序的调用。该指令在执行时完成如下操作:将子程序的返回地址存放在链接寄存器LR中,同时将程序计数器PC指向子程序的入口点,当子程序执行完毕返回调用处时,只需要将存放在LR中的返回地址重新复制给程序计数器PC即可。在调用子的同时,也可以完成参数的传递和从子

26、程序返回运算的结果,通常可以使用寄存器R0R3完成。以下是使用BL指令调用子程序的汇编语言程序的基本结构:AREA Init,CODE,READONLYENTRYStartLDR R0,=0x3FF5000LDR R1,0xFFSTR R1,R0LDR R0,=0x3FF5008LDR R1,0x01STR R1,R0BL PRINT_TEXTPRINT_TEXTMOV PC,BLEND4.3.3 宏定义及其作用使用宏定义可以提高程序的可读性,简化程序代码和同步修改。ARM宏定义与标准C语言的#define相似,只在源程序中进行字符的简单替代。宏定义从MACRO伪指令开始,到MEND结束,并可

27、以使用参数。宏首先要定义,然后才能使用。使用时直接书写宏名,并根据对应的宏定义格式设置输入参数或书写桔等。当源程序被汇编时,汇编器将展开每一个宏调用,用宏定义代替程序中的宏调用,并使用实际的参数值取代宏定义时的形参。下面的例子是一个宏调用,程序中定义了一个宏CALL,用于调用子程序,调用时设置所需要调用的子程序名Function及两个入口参数dat1和dat2。由于宏定义体中使用的是MOV指令,所以dat1参数只能为8位的立即数或通用寄存器。MACRO ;宏定义CALLSubfunction Function,dat1,dat2 ;宏名为CALLSubfunction,带3个参数LMPORT

28、Function ;声明外子程序名MOV R0,dat1 ;设置子程序参数R0=dat1MOV R1,dat2BL Function ;调用子程序MENDCALL FADD1,#3,#2 ;宏调用汇编处理后,宏调用将被展开,程序如下:IMPORT FADD1MOV R0,#3MOV R1,#2BL FADD14.3.4 数据比较跳转汇编程序可以使用CMP指令进行两个数的比较,然后根据比较结果实现程序的跳转,代码如下:CMP R5,#10 BEQ BRANCH1 ;如果R5为10,则跳转到BRANCH1CMP R1,R2ADDHI R1,R1,#1 ;如果R1R2,则R1=R1+1ADDLS R

29、1,R1,#2 ;如果R1=R2,则R1=R1+2ANDS R1,R1,#0x80 ;R1=R1&0x80,并设置相应的标志位BNE WAIT ;如果R1的第7位0,则跳转到WAIT4.3.5 循环下面的程序代码为汇编循环程序的例子,指定了循环的次数,每循环一次进行减1操作,并判断结果是否为0,如果为0则退出循环。MOV R0,#10LOOPSUBS R0,R0,#1BEN LOOP4.3. 6 数据块复制可以使用存储器访问指令LDM/STM进行读取和存储,进行数据块的复制,示例代码如下: LDR R0,=DATA_DST ;指向数据的目标地址 LDR R1,=DATA_SRC ;指向数据源地

30、址 MOV R1,#10 ;复制数据块的大小为10*N个字节LOOP LDMIA R1!,R2-R9 STMIA R0!,R2-R9 SUBS R10,R10,#1 BNE LOOP 4.3. 7 堆栈操作可以使用存储器访问指令LDM/STM实现堆栈操作,用于子程序的寄存器保护。在使用堆栈前,首先需要分配好堆栈空间,设置好寄存器R13(即堆栈指针SP),否则操作会失败。OUTDAT STMFD SP!R0-R7 ;寄存器入栈 BL DELAY LDMFD SP!R0-R7,PC ;寄存器出栈4.3. 8 查表操作查表操作是汇编程序经常使用的一种功能,代码如下:LDR R3,=DISP_TAB ;字模表的首地址LDR R2,R3,R5,LSL,#2 ;根据R5的值查表,取出相应的值。每个字模 地 ;占2个字节,故偏移量为2下面的表为0F的字模DISP_TAB DCD 0xF9,0xA4,0x99,0x92

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 教育专区 > 教案示例

本站为文档C TO C交易模式,本站只提供存储空间、用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。本站仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知得利文库网,我们立即给予删除!客服QQ:136780468 微信:18945177775 电话:18904686070

工信部备案号:黑ICP备15003705号-8 |  经营许可证:黑B2-20190332号 |   黑公网安备:91230400333293403D

© 2020-2023 www.deliwenku.com 得利文库. All Rights Reserved 黑龙江转换宝科技有限公司 

黑龙江省互联网违法和不良信息举报
举报电话:0468-3380021 邮箱:hgswwxb@163.com