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

		MACRO
$HandlerLabel HANDLER $HandleLabel

$HandlerLabel
	sub	sp,sp,#4	;decrement sp(to store jump address)
	stmfd	sp!,{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 r0
	ldr     r0,[r0]	 ;load the contents(service routine start address) of HandleXXX
	str     r0,[sp,#4]      ;store the contents(ISR) of HandleXXX to stack
	ldmfd   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=$dat1
MOV      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   表示本数据域在内存表中所占用的字节数.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值