AMR中宏的使用与结构化内存表的建立

来源:互联网 发布:php 检测类是否存在 编辑:程序博客网 时间:2024/05/16 03:08
MACRO$HandlerLabel HANDLER $HandleLabel$HandlerLabelsubsp,sp,#4;decrement sp(to store jump address)stmfdsp!,{r0};PUSH the work register to stack(lr does't push because it return to original address)ldr     r0,=$HandleLabel;load the address of HandleXXX to r0ldr     r0,[r0] ;load the contents(service routine start address) of HandleXXXstr     r0,[sp,#4]      ;store the contents(ISR) of HandleXXX to stackldmfd   sp!,{r0,pc}     ;POP the work register and pc(jump to ISR)MEND

这个宏定义$HandlerLabel HANDLER$HandleLabel,第一个$HandlerLabel是函数名字(或者是标号),第二个$HandleLabel(注意与第一个$HandlerLabel)少了一个r,是一个函数入口地址。

后面的代码可以证明

       ^  _ISR_STARTADDRESS              ;_ISR_STARTADDRESS=0x33FF_FF00

HandleReset  #  4

HandleUndef #  4

这段代码的意思是,由0x33FF_FF00为首地址建立一个表,之中的^与MAP指令同意,MAP 用于定义一个结构化的内存表的首地址.此时,内存表的位置计数器{VAR}设置

为该地址值{VAR}为汇编器的内置变量.^与MAP 同义.

其中的#号与FIELD同意,表示占用4的字节的数据域。

因此可以看出HandleReset是一个地址,而不是HandlerReset。

MACRO具体使用

格式:

MACRO  

[$标号]   宏名    [$参数1,$参数2,$参数3  ··········]

                            指令序列

MEND

其中标号与 参数都是可选部分,但是宏名是必须具备的部分。   其中标号与参数都会在宏指令被展开的时候替换为用户定义的符号,在MACRO与MEND 之间的指令序列称为宏定义体。在宏定义体的第一行应声明宏的原型(包含宏名、所需参数)

注意标号的使用方式,如果宏体内部要使用标号,那么在宏声明的时候就要使用一个标号进行声明,使这个标号成为一个主标号,在宏语句段内的其他标号都必须由这个主标号构成。并且在宏语句段内,所有标号前加“$”号。

MACRO                           ;宏定义 CALL     $Function,$dat1,$dat2  ;宏名称为 CALL,带 3 个参数IMPORT   $Function              ;声明外部子程序 MOV      R0,$dat1               ;设置子程序参数,R0=$dat1MOV      R1,$dat2        BL       Function               ;调用子程序 MEND                          ;宏定义结束

CALL     FADD1,#3,#2          ;宏调用

汇编预处理后,宏调用将被展开,程序清单如下:

       …

      IMPORT    FADD1    

      MOV       R0,#3

      MOV       R1,#3

      BL        FADD1

      …

在此要注意一个比较迷惑人的问题,就是在宏定义时,不是一定要求先是宏名,在带参数(当然可以不带参数),也可以像第一个实例一样,中间的HANDLER是宏名,其他收尾两个是参数。



MAP与FIELD的具体使用

MAP用于定义一个结构化的内存表的首地址.此时,内存表的位置计数器{VAR}设置为该地址值{VAR}为汇编器的内置变量.^与MAP 同义. 伪指令格式:

MAP  expr,{base_register}

其中   expr         数字表达式或程序中的标号.当指令中没有 base_register 时,expr 即为结构化内存表的首地址.

base_register  一个寄存器.当指令中包含这一项时,结构化内存表的首地 址为 expr 与 base_register 寄存器值的和.

伪指令应用举例如下;

MAP     0x00,R9 ;定义内存表的首地址为 R9

Timer    FIELD  4        ;定义数据域 Timer,长度为 4 字节

Attrib   FIELD  4        ;定义数据域 Attrib,长度为 4 字节

String   FIELD 100       ;定义数据域 String,长度为 100 字节

         …

ADR   R9,DataStart ;设置 R9 的值,即设置结构化的内存表地址

LDR   R0,Atrrib    ;相当于 LDR,R0,[R9,#4]

MAP伪指令和 FIELD 伪指令配合使用,用于定义结构化的内存表结构

FIELD用于定义一个结构化内存表中的数据域.#与FIELD 同义.

伪指令格式:

{tabel}   FIELD  expr

其中

      label 当指令中包含这一项时,label的值为当前内存表的位置计数 器{VAR}的值,汇编编译器处理了这条 FIELD 伪指令后,内存表 计数器的值将加上 expr.

          expr   表示本数据域在内存表中所占用的字节数.

原创粉丝点击