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   表示本數據域在內存表中所佔用的字節數.

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章