综合资讯 技术文章 原文阅读 在线商城 下载专区 DATASHEET 技术论坛 商务频道

电子技术 | 技术资料 | 嵌入式系统 | 单片机专题 | DSP专题
EDA/PLD专题 | 电源技术专题 | 电子制作专题 | 其他综合 | 芯片选型

所在的位置:首页在线阅读嵌入式系统ARM应用系统开发详解正文
 
第4章 ARM程序设计基础

ARM编译器一般都支持汇编语言的程序设计和C/C++语言的程序设计,以及两者的混合编程。本章介绍ARM程序设计的一些基本概念,如ARM汇编语言的伪指令、汇编语言的语句格式和汇编语言的程序结构等,同时介绍C/C++和汇编语言的混合编程等问题。

本章的主要内容:

  ARM编译器所支持的伪指令

  汇编语言的语句格式

  汇编语言的程序结构

  相关的程序示例

4.1  ARM汇编器所支持的伪指令

ARM汇编语言程序里,有一些特殊指令助记符,这些助记符与指令系统的助记符不同,没有相对应的操作码,通常称这些特殊指令助记符为伪指令,他们所完成的操作称为伪操作。伪指令在源程序中的作用是为完成汇编程序作各种准备工作的,这些伪指令仅在汇编过程中起作用,一旦汇编结束,伪指令的使命就完成。

ARM的汇编程序中,有如下几种伪指令:符号定义伪指令、数据定义伪指令、汇编控制伪指令、宏指令以及其他伪指令。

4.1.1  符号定义(Symbol Definition)伪指令

符号定义伪指令用于定义ARM汇编程序中的变量、对变量赋值以及定义寄存器的别名等操作。常见的符号定义伪指令有如下几种:

       用于定义全局变量的GBLAGBLLGBLS

       用于定义局部变量的LCLALCLLLCLS

       用于对变量赋值的SETASETLSETS

       为通用寄存器列表定义名称的RLIST

1、  GBLAGBLLGBLS

语法格式:

GBLAGBLLGBLS     全局变量名

GBLAGBLLGBLS伪指令用于定义一个ARM程序中的全局变量,并将其初始化。其中:

GBLA伪指令用于定义一个全局的数字变量,并初始化为0

GBLL伪指令用于定义一个全局的逻辑变量,并初始化为F(假);

GBLS伪指令用于定义一个全局的字符串变量,并初始化为空;

由于以上三条伪指令用于定义全局变量,因此在整个程序范围内变量名必须唯一。

使用示例:

  GBLA    Test1                   ;定义一个全局的数字变量,变量名为Test1

  Test1   SETA    0xaa            ;将该变量赋值为0xaa

  GBLL    Test2                   ;定义一个全局的逻辑变量,变量名为Test2

  Test2   SETL    {TRUE}          ;将该变量赋值为真

  GBLS    Test3                   ;定义一个全局的字符串变量,变量名为Test3

  Test3   SETS    Testing      ;将该变量赋值为“Testing

2、  LCLALCLLLCLS

语法格式:

LCLALCLLLCLS      局部变量名

LCLALCLLLCLS伪指令用于定义一个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、  SETASETLSETS

语法格式:

变量名     SETASETLSETS     表达式

伪指令SETASETLSETS用于给一个已经定义的全局变量或局部变量赋值。

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-R5R8R10} ;将寄存器列表名称定义为RegList,可在ARM指令LDM/STM中通过该名称访问寄存器列表。

4.1.2  数据定义(Data Definition)伪指令

数据定义伪指令一般用于为特定的数据分配存储单元,同时可完成已分配存储单元的初始化。常见的数据定义伪指令有如下几种:

       DCB                     用于分配一片连续的字节存储单元并用指定的数据初始化。

       DCWDCWU   用于分配一片连续的半字存储单元并用指定的数据初始化。

       DCDDCDU    用于分配一片连续的字存储单元并用指定的数据初始化。

       DCFDDCFDU)用于为双精度的浮点数分配一片连续的字存储单元并用指定的数据初始化。

       DCFSDCFSU 用于为单精度的浮点数分配一片连续的字存储单元并用指定的数据初始化。

       DCQDCQU    用于分配一片以8字节为单位的连续的存储单元并用指定的数据初始化。

       SPACE                  用于分配一片连续的存储单元

       MAP                     用于定义一个结构化的内存表首地址

       FIELD                  用于定义一个结构化的内存表的数据域

1、  DCB

语法格式:

标号        DCB       表达式

DCB伪指令用于分配一片连续的字节存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为0255的数字或字符串。DCB也可用“=”代替。

使用示例:

Str DCB This is a test!” ;分配一片连续的字节存储单元并初始化。

2、  DCW(或DCWU

语法格式:

标号        DCW(或DCWU      表达式

DCW(或DCWU)伪指令用于分配一片连续的半字存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为程序标号或数字表达式。。

DCW分配的字存储单元是半字对齐的,而用DCWU分配的字存储单元并不严格半字对齐。

使用示例:

DataTest    DCW     123  ;分配一片连续的半字存储单元并初始化。

3、  DCD(或DCDU

语法格式:

标号        DCD(或DCDU       表达式

DCD(或DCDU)伪指令用于分配一片连续的字存储单元并用伪指令中指定的表达式初始化。其中,表达式可以为程序标号或数字表达式。DCD也可用“&”代替。

DCD分配的字存储单元是字对齐的,而用DCDU分配的字存储单元并不严格字对齐。

使用示例:

DataTest    DCD     456  ;分配一片连续的字存储单元并初始化。

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-5E7  ;分配一片连续的字存储单元并初始化为指定的单精度数。

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     0x100R0       ;定义结构化内存表首地址的值为0x100R0

9、  FILED

语法格式:

标号        FIELD    表达式

FIELD伪指令用于定义一个结构化内存表中的数据域。FILED也可用“#”代替。

表达式的值为当前数据域在内存表中所占的字节数。

FIELD伪指令常与MAP伪指令配合使用来定义结构化的内存表。MAP伪指令定义内存表的首地址,FIELD伪指令定义内存表中的各个数据域,并可以为每个数据域指定一个标号供其他的指令引用。

注意MAPFIELD伪指令仅用于定义数据结构,并不实际分配存储单元。

使用示例:

MAP     0x100           ;定义结构化内存表首地址的值为0x100

        A       FIELD   16      ;定义A的长度为16字节,位置为0x100

        B       FIELD   32      ;定义B的长度为32字节,位置为0x110

        S       FIELD   256     ;定义S的长度为256字节,位置为0x130

4.1.3  汇编控制(Assembly Control)伪指令

汇编控制伪指令用于控制汇编程序的执行流程,常用的汇编控制伪指令包括以下几条:

       IFELSEENDIF

       WHILEWEND

       MACROMEND

       MEXIT

1、  IFELSEENDIF

语法格式:

IF     逻辑表达式

        指令序列1

ELSE

        指令序列2

ENDIF

IFELSEENDIF伪指令能根据条件的成立与否决定是否执行某个指令序列。当IF后面的逻辑表达式为真,则执行指令序列1,否则执行指令序列2。其中,ELSE及指令序列2可以没有,此时,当IF后面的逻辑表达式为真,则执行指令序列1,否则继续执行后面的指令。

IFELSEENDIF伪指令可以嵌套使用。

使用示例:

GBLL  Test                    ;声明一个全局的逻辑变量,变量名为Test

……

IF    Test = TRUE

  指令序列1

ELSE

  指令序列2

ENDIF

2、  WHILEWEND

语法格式:

WHILE    逻辑表达式

        指令序列

WEND

WHILEWEND伪指令能根据条件的成立与否决定是否循环执行某个指令序列。当WHILE后面的逻辑表达式为真,则执行指令序列,该指令序列执行完毕后,再判断逻辑表达式的值,若为真则继续执行,一直到逻辑表达式的值为假。

WHILEWEND伪指令可以嵌套使用。

使用示例:

GBLA  Counter                 ;声明一个全局的数学变量,变量名为Counter

Counter     SETA     3             ;由变量Counter控制循环次数

……

WHILE Counter < 10

  指令序列

WEND

3、  MACROMEND

语法格式:

$标号      宏名       $参数1$参数2……     

指令序列

MEND

MACROMEND伪指令可以将一段代码定义为一个整体,称为宏指令,然后就可以在程序中通过宏指令多次调用该段代码。其中,$标号在宏指令被展开时,标号会被替换为用户定义的符号,

宏指令可以使用一个或多个参数,当宏指令被展开时,这些参数被相应的值替换。

       宏指令的使用方式和功能与子程序有些相似,子程序可以提供模块化的程序设计、节省存储空间并提高运行速度。但在使用子程序结构时需要保护现场,从而增加了系统的开销,因此,在代码较短且需要传递的参数较多时,可以使用宏指令代替子程序。

       包含在MACROMEND之间的指令序列称为宏定义体,在宏定义体的第一行应声明宏的原型(包含宏名、所需的参数),然后就可以在汇编程序中通过宏名来调用该指令序列。在源程序被编译时,汇编器将宏调用展开,用宏定义中的指令序列代替程序中的宏调用,并将实际参数的值传递给宏定义中的形式参数。

MACROMEND伪指令可以嵌套使用。

4、  MEXIT

语法格式:

MEXIT

MEXIT用于从宏定义中跳转出去。

4.1.4  其他常用的伪指令

还有一些其他的伪指令,在汇编程序中经常会被使用,包括以下几条:

       AREA

       ALIGN

       CODE16CODE32

       ENTRY

       END

       EQU

       EXPORT(或GLOBAL

       IMPORT

       EXTERN

       GET(或INCLUDE

       INCBIN

       RN

       ROUT

1、  AREA

语法格式:

AREA      段名       属性1,属性2…… 

AREA伪指令用于定义一个代码段或数据段。其中,段名若以数字开头,则该段名需用“|”括起来,如|1_test|

属性字段表示该代码段(或数据段)的相关属性,多个属性用逗号分隔。常用的属性如下:

       CODE属性:用于定义代码段,默认为READONLY

       DATA属性:用于定义数据段,默认为READWRITE

       READONLY属性:指定本段为只读,代码段默认为READONLY

       READWRITE属性:指定本段为可读可写,数据段的默认属性为READWRITE

       ALIGN属性:使用方式为ALIGN 表达式。在默认时,ELF(可执行连接文件)的代码段和数据段是按字对齐的,表达式的取值范围为031,相应的对齐方式为2表达式次方。

       COMMON属性:该属性定义一个通用的段,不包含任何的用户代码和数据。各源文件中同名的COMMON段共享同一段存储单元。

一个汇编语言程序至少要包含一个段,当程序太长时,也可以将程序分为多个代码段和数据段。

使用示例:

AREA  InitCODEREADONLY

指令序列

;该伪指令定义了一个代码段,段名为Init,属性为只读

2、  ALIGN

语法格式:

ALIGN     {表达式{,偏移量}}   

ALIGN伪指令可通过添加填充字节的方式,使当前位置满足一定的对其方式|。其中,表达式的值用于指定对齐方式,可能的取值为2的幂,如124816等。若未指定表达式,则将当前位置对齐到下一个字的位置。偏移量也为一个数字表达式,若使用该字段,则当前位置的对齐方式为:2的表达式次幂+偏移量。

使用示例:

AREA  InitCODEREADONLYALIEN3   ;指定后面的指令为8字节对齐。

指令序列

END

3、  CODE16CODE32

语法格式:

CODE16(或CODE32

CODE16伪指令通知编译器,其后的指令序列为16位的Thumb指令。

CODE32伪指令通知编译器,其后的指令序列为32位的ARM指令。

若在汇编源程序中同时包含ARM指令和Thumb指令时,可用CODE16伪指令通知编译器其后的指令序列为16位的Thumb指令,CODE32伪指令通知编译器其后的指令序列为32位的ARM指令。因此,在使用ARM指令和Thumb指令混合编程的代码里,可用这两条伪指令进行切换,但注意他们只通知编译器其后指令的类型,并不能对处理器进行状态的切换。

使用示例:

AREA  InitCODEREADONLY

……

CODE32                ;通知编译器其后的指令为32位的ARM指令

LDR   R0,=NEXT1    ;将跳转地址放入寄存器R0

BX    R0              ;程序跳转到新的位置执行,并将处理器切换到Thumb工作状态

……

CODE16                    通知编译器其后的指令为16位的Thumb指令

NEXT   LDR  R3,=0x3FF  

……

END                       ;程序结束