第四章 ARM匯編語(yǔ)言程序設(shè)計(jì)課件_第1頁(yè)
第四章 ARM匯編語(yǔ)言程序設(shè)計(jì)課件_第2頁(yè)
第四章 ARM匯編語(yǔ)言程序設(shè)計(jì)課件_第3頁(yè)
第四章 ARM匯編語(yǔ)言程序設(shè)計(jì)課件_第4頁(yè)
第四章 ARM匯編語(yǔ)言程序設(shè)計(jì)課件_第5頁(yè)
已閱讀5頁(yè),還剩143頁(yè)未讀 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說(shuō)明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡(jiǎn)介

第四章ARM匯編語(yǔ)言程序設(shè)計(jì)ADS1.2環(huán)境下的匯編語(yǔ)句格式1.ARM匯編中,所有標(biāo)號(hào)必須在一行的頂格書寫,其后面不要添加“:”,而所有指令均不能頂格書寫。2.ARM匯編器對(duì)標(biāo)識(shí)符大小寫敏感,書寫標(biāo)號(hào)及指令時(shí)字母大小寫要一致,在ARM匯編程序中,一個(gè)ARM指令、偽指令、寄存器名可以全部為大寫字母,也可以全部為小寫字母,但不要大小寫混合使用(如果混合使用也能編譯通過(guò),如movSP,r6能編譯通過(guò),但是moVSP,r6則出現(xiàn)錯(cuò)誤)。3.注釋可以在一行的頂格書寫。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)ADS1.2環(huán)境下的匯編語(yǔ)句14.1偽操作

在ARM匯編語(yǔ)言程序里,有一些特殊指令助記符,這些助記符與指令系統(tǒng)的助記符不同,沒(méi)有相對(duì)應(yīng)的操作碼,通常稱這些特殊指令助記符為偽指令他們所完成的操作稱為偽操作。ARM指令系統(tǒng)4.1偽操作在ARM匯編語(yǔ)言程序里,2

偽指令:由匯編器處理偽指令是由匯編器對(duì)源程序匯編期間進(jìn)行處理的.偽指令僅在匯編過(guò)程中起作用,一旦匯編結(jié)束,偽指令的使命就完成。

指令:由ARM處理器執(zhí)行ARM指令系統(tǒng)偽指令:由匯編器處理ARM指令系統(tǒng)3在ARM的匯編程序中,有如下幾種偽指令: 符號(hào)定義偽指令、 數(shù)據(jù)定義偽指令、 匯編控制偽指令、 宏指令以及其他偽指令。ARM指令系統(tǒng)在ARM的匯編程序中,有如下幾種偽指令:ARM指令系統(tǒng)44.1.1符號(hào)定義偽指令

符號(hào)定義偽指令用于定義ARM匯編程序中的變量、對(duì)變量賦值以及定義寄存器的別名等操作。ARM指令系統(tǒng)4.1.1符號(hào)定義偽指令A(yù)RM指令系統(tǒng)54.1.1符號(hào)定義偽指令

常見的符號(hào)定義偽指令有如下幾種:— 用于定義全局變量的GBLA、GBLL和GBLS。— 用于定義局部變量的LCLA、LCLL和LCLS?!?用于對(duì)變量賦值的SETA、SETL、SETS?!?為通用寄存器列表定義名稱的RLIST。ARM指令系統(tǒng)4.1.1符號(hào)定義偽指令常見的符號(hào)定義偽指令有如下幾種:61、 GBLA、GBLL和GBLS語(yǔ)法格式:GBLA(GBLL或GBLS) 全局變量名GBLA、GBLL和GBLS偽指令用于定義一個(gè)ARM程序中的全局變量,并將其初始化。GBLA偽指令用于定義一個(gè)全局的數(shù)字變量,并初始化為0;GBLL偽指令用于定義一個(gè)全局的邏輯變量,并初始化為F(假);GBLS偽指令用于定義一個(gè)全局的字符串變量,并初始化為空;由于以上三條偽指令用于定義全局變量,因此在整個(gè)程序范圍內(nèi)變量名必須唯一。1、 GBLA、GBLL和GBLS7

GBLA Test1 ;定義一個(gè)全局的數(shù)字變量,變量名為Test1 Test1 SETA 0xaa ;將該變量賦值為0xaa GBLL Test2 ;定義一個(gè)全局的邏輯變量,變量名為Test2 Test2 SETL {TRUE} ;將該變量賦值為真 GBLS Test3 ;定義一個(gè)全局的字符串變量,變量名為Test3 Test3 SETS "Testing" ;將該變量賦值為"Testing" GBLA Test1 82、 LCLA、LCLL和LCLS語(yǔ)法格式:LCLA(LCLL或LCLS) 局部變量名LCLA、LCLL和LCLS偽指令用于定義一個(gè)ARM程序中的局部變量,并將其初始化。其中:LCLA偽指令用于定義一個(gè)局部的數(shù)字變量,并初始化為0;LCLL偽指令用于定義一個(gè)局部的邏輯變量,并初始化為F(假);LCLS偽指令用于定義一個(gè)局部的字符串變量,并初始化為空; 以上三條偽指令用于聲明局部變量ARM指令系統(tǒng)2、 LCLA、LCLL和LCLSARM指令系統(tǒng)9使用示例: LCLA Test4 ;聲明一個(gè)局部的數(shù)字變量,變量名為Test4 Test3 SETA 0xaa ;將該變量賦值為0xaa LCLL Test5 ;聲明一個(gè)局部的邏輯變量,變量名為Test5 Test4 SETL {TRUE};將該變量賦值為真 LCLS Test6 ;定義一個(gè)局部的字符串變量,變量名為Test6 Test6 SETS “Testing” ;將該變量賦值為“Testing”使用示例:103、 SETA、SETL和SETS語(yǔ)法格式:變量名 SETA(SETL或SETS)表達(dá)式

偽指令SETA、SETL、SETS給一個(gè)已經(jīng)定義的全局變量或局部變量賦值。SETA偽指令用于給一個(gè)數(shù)學(xué)變量賦值;SETL偽指令用于給一個(gè)邏輯變量賦值;SETS偽指令用于給一個(gè)字符串變量賦值;

其中,變量名為已經(jīng)定義過(guò)的全局變量或局部變量,表達(dá)式為將要賦給變量的值。3、 SETA、SETL和SETS11使用示例:

LCLA Test3 ;聲明一個(gè)局部的數(shù)字變量,變量名為Test3 Test3

SETA 0xaa;將該變量賦值為0xaa LCLL Test4 ;聲明一個(gè)局部的邏輯變量,變量名為Test4 Test4

SETL {TRUE} ;將該變量賦值為真ARM指令系統(tǒng)使用示例:ARM指令系統(tǒng)124、 RLIST名稱 RLIST {寄存器列表}RLIST偽指令可用于對(duì)一個(gè)通用寄存器列表定義名稱,使用該偽指令定義的名稱可在ARM指令LDM/STM中使用。在LDM/STM指令中,列表中的寄存器訪問(wèn)次序?yàn)楦鶕?jù)寄存器的編號(hào)由低到高,而與列表中的寄存器排列次序無(wú)關(guān)。4、 RLIST13RLIST使用示例:RegList RLIST {R0-R5,R8,R10}

將寄存器列表名稱定義為RegList,可在ARM指令LDM/STM中通過(guò)該名稱訪問(wèn)寄存器列表。例:RegList RLIST {r1-r13};必須頂格開始寫LDMIAr0,RegListRLIST使用示例:144.1.2數(shù)據(jù)定義(DataDefinition)偽指令

數(shù)據(jù)定義偽指令一般用于為特定的數(shù)據(jù)分配存儲(chǔ)單元,同時(shí)可完成已分配存儲(chǔ)單元的初始化。常見的數(shù)據(jù)定義偽指令有如下幾種:— DCB 用于分配一片連續(xù)的字節(jié)存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。— DCW(DCWU) 用于分配一片連續(xù)的半字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化?!?DCFD(DCFDU)用于為雙精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。ARM指令系統(tǒng)4.1.2數(shù)據(jù)定義(DataDefinition)偽指15— DCFS(DCFSU) 用于為單精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化?!?DCQ(DCQU) 用于分配一片以8字節(jié)為單位的連續(xù)的存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。DCD(DCDU)用于分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。SPACE 用于分配一片連續(xù)的存儲(chǔ)單元MAP用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表首地址FIELD用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表的數(shù)據(jù)域— DCFS(DCFSU) 用于為單精度的浮點(diǎn)數(shù)分配一片連續(xù)161、 DCB(DCB也可用“=”代替)語(yǔ)法格式:標(biāo)號(hào) DCB 表達(dá)式

DCB偽指令用于分配一片連續(xù)的字節(jié)存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。其中,表達(dá)式可以為0~255的數(shù)字或字符串。使用示例:Str DCB "Thisisatest!" ;分配一片連續(xù)的字節(jié)存儲(chǔ)單元并初始化。1、 DCB(DCB也可用“=”代替)172、 DCD(或DCDU,DCD也可用“&”代替)語(yǔ)法格式:標(biāo)號(hào) DCD(或DCDU) 表達(dá)式

DCD(或DCDU)偽指令用于分配一片連續(xù)的字存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。其中,表達(dá)式可以為程序標(biāo)號(hào)或數(shù)字表達(dá)式。

用DCD分配的字存儲(chǔ)單元是字對(duì)齊的,而用DCDU分配的字存儲(chǔ)單元并不嚴(yán)格字對(duì)齊。使用示例:DataTest DCD 4,5,6 ;分配一片連續(xù)的字存儲(chǔ)單元并初始化。2、 DCD(或DCDU,DCD也可用“&”代替)183、 SPACE(SPACE也可用“%”代替)語(yǔ)法格式:標(biāo)號(hào) SPACE 表達(dá)式SPACE偽指令用于分配一片連續(xù)的存儲(chǔ)區(qū)域并初始化為0。其中,表達(dá)式為要分配的字節(jié)數(shù)。使用示例:DataSpace SPACE 100 ;分配連續(xù)100字節(jié)的存儲(chǔ)單元并初始化為0。3、 SPACE(SPACE也可用“%”代替)194、 MAP,MAP也可用“^”代替。語(yǔ)法格式:MAP 表達(dá)式{,基址寄存器}MAP偽指令用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表的首地址。表達(dá)式可以為程序中的標(biāo)號(hào)或數(shù)學(xué)表達(dá)式,基址寄存器為可選項(xiàng),當(dāng)基址寄存器選項(xiàng)不存在時(shí),表達(dá)式的值即為內(nèi)存表的首地址,當(dāng)該選項(xiàng)存在時(shí),內(nèi)存表的首地址為表達(dá)式的值與基址寄存器的和。4、 MAP,MAP也可用“^”代替。20MAP偽指令通常與FIELD偽指令配合使用來(lái)定義結(jié)構(gòu)化的內(nèi)存表。使用示例:MAP 0x100,R0 ;定義結(jié)構(gòu)化內(nèi)存表首地址的值為0x100+R0。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)課件215、 FILEDFILED也可用“#”代替。語(yǔ)法格式:標(biāo)號(hào) FIELD 表達(dá)式FIELD偽指令用于定義一個(gè)結(jié)構(gòu)化內(nèi)存表中的數(shù)據(jù)域。表達(dá)式的值為當(dāng)前數(shù)據(jù)域在內(nèi)存表中所占的字節(jié)數(shù)。

FIELD偽指令常與MAP偽指令配合使用來(lái)定義結(jié)構(gòu)化的內(nèi)存表。MAP偽指令定義內(nèi)存表的首地址,F(xiàn)IELD偽指令定義內(nèi)存表中的各個(gè)數(shù)據(jù)域,并可以為每個(gè)數(shù)據(jù)域指定一個(gè)標(biāo)號(hào)供其他的指令引用。5、 FILEDFILED也可用“#”代替。22注意MAP和FIELD偽指令僅用于定義數(shù)據(jù)結(jié)構(gòu),并不實(shí)際分配存儲(chǔ)單元。使用示例:MAP 0x100 ;定義結(jié)構(gòu)化內(nèi)存表首地址的值為0x100。A FIELD 16 ;定義A的長(zhǎng)度為16字節(jié),位置為0x100B FIELD 32 ;定義B的長(zhǎng)度為32字節(jié),位置為0x110S FIELD 256 ;定義S的長(zhǎng)度為256字節(jié),位置為0x130注意MAP和FIELD偽指令僅用于定義數(shù)據(jù)結(jié)構(gòu),并不實(shí)際分23s3c44b0xARM芯片的初始化源程序:AREARamData,DATA,READWRITE^(_IRQ_BASEADDRESS);^即MAPHandleReset#4;#即DCDHandleUndef#4HandleSWI#4HandlePabort#4HandleDabort#4HandleReserved#4HandleIRQ#4HandleFIQ#4s3c44b0xARM芯片的初始化源程序:244.1.3匯編控制偽指令匯編控制偽指令用于控制匯編程序的執(zhí)行流程常用的匯編控制偽指令包括:— IF、ELSE、ENDIF— WHILE、WEND— MACRO、MEND— MEXITARM指令系統(tǒng)4.1.3匯編控制偽指令匯編控制251IF、ELSE、ENDIF(等價(jià)于[|])語(yǔ)法格式:IF 邏輯表達(dá)式 指令序列1ELSE 指令序列2ENDIFIF、ELSE、ENDIF偽指令能根據(jù)條件的成立與否決定是否執(zhí)行某個(gè)指令序列。當(dāng)IF后面的邏輯表達(dá)式為真,則執(zhí)行指令序列1,否則執(zhí)行指令序列2。其中,ELSE及指令序列2可以沒(méi)有,此時(shí),當(dāng)IF后面的邏輯表達(dá)式為真,則執(zhí)行指令序列1,否則繼續(xù)執(zhí)行后面的指令。1IF、ELSE、ENDIF(等價(jià)于[|])26IF、ELSE、ENDIF偽指令可以嵌套使用。使用示例: GBLL

Test;聲明一個(gè)全局的邏輯變量,變量名為Test ……

IF

Test=TRUE

指令序列1

ELSE

指令序列2

ENDIFIF、ELSE、ENDIF偽指令可以嵌套使用。27s3c44b0xARM芯片的初始化源程序:

GBLLTHUMBCODE;globallogicalvariable,init.to{false}[{CONFIG}=16 ;[=if,{CONFIG}:build-invariabledefinedbyARM;assembler,16=assembingThumbcode.THUMBCODESETL {TRUE};SETL:setthelogicvariableCODE32;changingfromThumbstatetoArmstate|;|=elseTHUMBCODESETL {FALSE}];]=endifs3c44b0xARM芯片的初始化源程序:282、 WHILE、WEND語(yǔ)法格式:WHILE 邏輯表達(dá)式 指令序列WEND

WHILE、WEND偽指令可以嵌套使用。WHILE、WEND偽指令能根據(jù)條件的成立與否決定是否循環(huán)執(zhí)行某個(gè)指令序列。當(dāng)WHILE后面的邏輯表達(dá)式為真,則執(zhí)行指令序列,該指令序列執(zhí)行完畢后,再判斷邏輯表達(dá)式的值,若為真則繼續(xù)執(zhí)行,一直到邏輯表達(dá)式的值為假。2、 WHILE、WEND29使用示例:

GBLA

Counter ;聲明一個(gè)全局的數(shù)學(xué)變量,變量名為CounterCounter SETA 3 ;由變量Counter控制循環(huán)次數(shù)……

WHILE Counter<10

指令序列

WEND使用示例:303、 MACRO、MEND語(yǔ)法格式:

MACRO$標(biāo)號(hào) 宏名

$參數(shù)1,$參數(shù)2,…… 指令序列MEND

MACRO、MEND偽指令可以將一段代碼定義為一個(gè)整體,稱為宏指令,然后就可以在程序中通過(guò)宏指令多次調(diào)用該段代碼。其中,$標(biāo)號(hào)在宏指令被展開時(shí),標(biāo)號(hào)會(huì)被替換為用戶定義的符號(hào);

宏指令可以使用一個(gè)或多個(gè)參數(shù),當(dāng)宏指令被展開時(shí),這些參數(shù)被相應(yīng)的值替換。MACRO、MEND偽指令可以嵌套使用。3、 MACRO、MEND31s3c44b0x初始化程序里的例子:

MACRO$HandlerLabelHANDLER$HandleLabel$HandlerLabelsub sp,sp,#4

stmfd

sp!,{r0} ldr r0,=$HandleLabelldr r0,[r0] str r0,[sp,#4] ldmfdsp!,{r0,pc} MENDs3c44b0x初始化程序里的例子:32HandlerEINT4567 HANDLERHandleEINT4567將被匯編成如下語(yǔ)句:HandlerEINT4567sub sp,sp,#4 stmfdsp!,{r0} ldr r0,=HandleEINT4567;ldr r0,[r0] str r0,[sp,#4]ldmfdsp!,{r0,pc}HandlerEINT4567 HANDLERHandl33HandleEINT4567的定義:

^(_IRQ_BASEADDRESS+0x100)HandleADC#4………………..HandleEINT4567#4HandleEINT4567的定義:34圖1

執(zhí)行subsp,sp,#4時(shí)堆棧的情況圖1執(zhí)行subsp,sp,#4時(shí)堆棧的情況35圖2

執(zhí)行stmfdsp!,{r0},將R0壓入堆棧后綴“!”表示最后的地址寫回到sp中圖2執(zhí)行stmfdsp!,{r0},將R0壓入36圖3執(zhí)行strr0,[sp,#4],將R0壓入堆棧(R0)=HandleXXX處的內(nèi)容,不改變SP的值。

圖3執(zhí)行strr0,[sp,#4],將R0壓入堆棧37圖4ldmfdsp!,{r0,pc}

;彈出兩個(gè)字,分別送到R0,PC

圖4ldmfdsp!,{r0,pc}

;彈出兩個(gè)字,384、 MEXIT語(yǔ)法格式:MEXITMEXIT用于從宏定義中跳轉(zhuǎn)出去。ARM指令系統(tǒng)4、 MEXITARM指令系統(tǒng)394.1.6其他常用的偽指令還有一些其他的偽指令,在匯編程序中經(jīng)常會(huì)被使用,包括以下幾條:AREAALIGNCODE16、CODE32ENTRYENDEQUEXPORT(或GLOBAL)IMPORTEXTERNGET(或INCLUDE)INCBINRNROUT4.1.6其他常用的偽指令還有一401、 AREA語(yǔ)法格式:AREA 段名 屬性1,屬性2,……

AREA偽指令用于定義一個(gè)代碼段或數(shù)據(jù)段。其中,段名若以數(shù)字開頭,則該段名需用“|”括起來(lái),如|1_test|。屬性字段表示該代碼段(或數(shù)據(jù)段)的相關(guān)屬性,多個(gè)屬性用逗號(hào)分隔。常用的屬性如下:

CODE屬性:用于定義代碼段,默認(rèn)為READONLY。

DATA屬性:用于定義數(shù)據(jù)段,默認(rèn)為READWRITE。1、 AREA41READONLY屬性:指定本段為只讀,代碼段默認(rèn)為READONLY。READWRITE屬性:指定本段為可讀可寫,數(shù)據(jù)段的默認(rèn)屬性為READWRITE。ALIGN屬性:使用方式為ALIGN表達(dá)式。在默認(rèn)時(shí),ELF(可執(zhí)行連接文件)的代碼段和數(shù)據(jù)段是按字對(duì)齊的,表達(dá)式的取值范圍為0~31,相應(yīng)的對(duì)齊方式為2表達(dá)式次方。

ThisisnotthesameasthewaythattheALIGNdirectiveisspecified.

DonotuseALIGN=0orALIGN=1forcodesections.

READONLY屬性:指定本段為只讀,代碼段默認(rèn)為READO42使用示例:AREA Init,CODE,READONLY;該偽指令定義了一個(gè)代碼段,段名為Init,屬性為只讀

AREAMyDATA,DATA,READONLY,ALIGN=14;(以214字節(jié)對(duì)齊.)使用示例:432、 ALIGN語(yǔ)法格式:ALIGN {表達(dá)式{,偏移量}} ALIGN偽指令可通過(guò)添加填充字節(jié)的方式,使當(dāng)前位置滿足一定的對(duì)其方式|。其中,表達(dá)式的值用于指定對(duì)齊方式,可能的取值為2的冪,如1、2、4、8、16等。若未指定表達(dá)式,則將當(dāng)前位置對(duì)齊到下一個(gè)字的位置。偏移量也為一個(gè)數(shù)字表達(dá)式,如果不是用在AREA里的,應(yīng)該是表達(dá)式+偏移量.2、 ALIGN44將兩個(gè)字節(jié)的數(shù)據(jù)放在同一個(gè)字的第一個(gè)字節(jié)和第四個(gè)字節(jié)中,帶offset的ALIGN對(duì)齊:AREAoffsetFxample,CODEDCB0x31;第一個(gè)字節(jié)保存0x31ALIGN4,3;字對(duì)齊DCB0x32;第四個(gè)字節(jié)保存0x32內(nèi)容是:0x32000031上面的語(yǔ)句ALIGN4,3意義是:字對(duì)齊,然后再加上偏移3將兩個(gè)字節(jié)的數(shù)據(jù)放在同一個(gè)字的第一個(gè)45ExamplesAREAcacheable,CODE,ALIGN=3rout1;code;alignedon8-byteboundary;codeMOVpc,lr;alignedonlyon4-byteboundaryALIGN8;nowalignedon8-byteboundaryrout2;codeExamples463、 CODE16、CODE32語(yǔ)法格式: CODE16(或CODE32)CODE16偽指令通知編譯器,其后的指令序列為16位的Thumb指令。CODE32偽指令通知編譯器,其后的指令序列為32位的ARM指令。3、 CODE16、CODE3247若在匯編源程序中同時(shí)包含ARM指令和Thumb指令時(shí),可用CODE16偽指令通知編譯器其后的指令序列為16位的Thumb指令,CODE32偽指令通知編譯器其后的指令序列為32位的ARM指令。因此,在使用ARM指令和Thumb指令混合編程的代碼里,可用這兩條偽指令進(jìn)行切換

注意:他們只通知編譯器其后指令的類型,并不能對(duì)處理器進(jìn)行狀態(tài)的切換。若在匯編源程序中同時(shí)包含ARM指令和48

AREA Init,CODE,READONLY…… CODE32;通知編譯器其后的指令為32位的ARM指令 LDR R0,=NEXT+1 ;將跳轉(zhuǎn)地址放入寄存R0 BX R0 ;程序跳轉(zhuǎn)到新的位置執(zhí)行,并將處理器切換到Thumb工作狀態(tài) …… CODE16 ;通知編譯器其后的指令為16位的Thumb指令NEXT LDR R3,=0x3FF ……END ;程序結(jié)束 AREA Init,CODE,READONLY494、 ENTRY語(yǔ)法格式: ENTRY

ENTRY偽指令用于指定匯編程序的入口點(diǎn)。在一個(gè)完整的匯編程序中至少要有一個(gè)ENTRY(也可以有多個(gè),當(dāng)有多個(gè)ENTRY時(shí),程序的真正入口點(diǎn)由鏈接器指定),但在一個(gè)源文件里最多只能有一個(gè)ENTRY(可以沒(méi)有)。4、 ENTRY50使用示例: AREA Init,CODE,READONLY ENTRY;指定應(yīng)用程序的入口點(diǎn)……使用示例:515、 END語(yǔ)法格式: END

END偽指令用于通知編譯器已經(jīng)到了源程序的結(jié)尾,用于指示匯編編譯器源文件已結(jié)束.每一個(gè)匯編源文件均要使用一個(gè).使用示例: AREA Init,CODE,READONLY …… END ;指定應(yīng)用程序的結(jié)尾5、 END526、 EQU,可用“*”代替。語(yǔ)法格式:名稱 EQU 表達(dá)式{,類型}名稱必須頂格寫EQU偽指令用于為程序中的常量、標(biāo)號(hào)等定義一個(gè)等效的字符名稱,類似于C語(yǔ)言中的#define。名稱為EQU偽指令定義的字符名稱,當(dāng)表達(dá)式為32位的常量時(shí),可以指定表達(dá)式的數(shù)據(jù)類型,可以有以下三種類型:CODE16、CODE32和DATA6、 EQU,可用“*”代替。53使用示例:Test EQU 50;定義標(biāo)號(hào)Test的值為50Addr EQU 0x55,CODE32 ;定義Addr的值為0x55,且該處為32位的ARM指令。使用示例:547、 EXPORT(或GLOBAL)語(yǔ)法格式: EXPORT 標(biāo)號(hào){[WEAK]}

EXPORT偽指令用于在程序中聲明一個(gè)全局的標(biāo)號(hào),該標(biāo)號(hào)可在其他的文件中引用。EXPORT可用GLOBAL代替。

標(biāo)號(hào)在程序中區(qū)分大小寫

[WEAK]選項(xiàng)聲明其他的同名標(biāo)號(hào)優(yōu)先于該標(biāo)號(hào)被引用。7、 EXPORT(或GLOBAL)55使用示例:

AREA Init,CODE,READONLY EXPORT Stest;聲明一個(gè)可全局引用的標(biāo)號(hào)Stest……END

使用示例:568、 IMPORT語(yǔ)法格式:

IMPORT 標(biāo)號(hào){[WEAK]}

IMPORT偽指令用于通知編譯器要使用的標(biāo)號(hào)在其他的源文件中定義,但要在當(dāng)前源文件中引用,而且無(wú)論當(dāng)前源文件是否引用該標(biāo)號(hào),該標(biāo)號(hào)均會(huì)被加入到當(dāng)前源文件的符號(hào)表中。

[WEAK]選項(xiàng)表示當(dāng)所有的源文件都沒(méi)有定義這樣一個(gè)標(biāo)號(hào)時(shí),編譯器也不給出錯(cuò)誤信息,在多數(shù)情況下將該標(biāo)號(hào)置為0,若該標(biāo)號(hào)為B或BL指令引用,則將B或BL指令置為NOP操作。8、 IMPORT57使用示例: AREA Init,CODE,READONLY IMPORT Main ;通知編譯器當(dāng)前文件要引用標(biāo)號(hào)Main,但Main在其他源文件中定義 …… END

使用示例:589、 EXTERN語(yǔ)法格式:EXTERN 標(biāo)號(hào){[WEAK]}

EXTERN偽指令用于通知編譯器要使用的標(biāo)號(hào)在其他的源文件中定義,但要在當(dāng)前源文件中引用,如果當(dāng)前源文件實(shí)際并未引用該標(biāo)號(hào),該標(biāo)號(hào)就不會(huì)被加入到當(dāng)前源文件的符號(hào)表中。9、 EXTERN59使用示例: AREA Init,CODE,READONLY EXTERN Main ;通知編譯器當(dāng)前文件要引用標(biāo)號(hào)Main,但Main在其他源文件中定義 …… END 使用示例:6010、 GET(或INCLUDE)語(yǔ)法格式: GET 文件名

GET偽指令用于將一個(gè)源文件包含到當(dāng)前的源文件中,并將被包含的源文件在當(dāng)前位置進(jìn)行匯編處理。

使用方法與C語(yǔ)言中的“include”相似。

GET偽指令只能用于包含源文件,包含目標(biāo)文件需要使用INCBIN偽指令

10、 GET(或INCLUDE)61使用示例:

AREA Init,CODE,READONLY

GET a1.s;通知編譯器當(dāng)前源文件包含源文件a1.s

GET C:\a2.s;通知編譯器當(dāng)前源文件包含源文件C:\a2.s

……

END 使用示例:6211、 INCBIN語(yǔ)法格式:

INCBIN 文件名INCBIN偽指令用于將一個(gè)目標(biāo)文件或數(shù)據(jù)文件包含到當(dāng)前的源文件中,被包含的文件不作任何變動(dòng)的存放在當(dāng)前文件中,編譯器從其后開始繼續(xù)處理。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)課件63使用示例:

AREA Init,CODE,READONLY

INCBIN a1.dat ;通知編譯器當(dāng)前源文件包含文件a1.dat

INCBIN C:\a2.txt ;通知編譯器當(dāng)前源文件包含文件

C:\a2.txt

……

END 使用示例:6412、 RN語(yǔ)法格式:名稱 RN 表達(dá)式RN偽指令用于給一個(gè)寄存器定義一個(gè)別名。采用這種方式可以方便程序員記憶該寄存器的功能。其中,名稱為給寄存器定義的別名,表達(dá)式為寄存器的編碼。名稱必須頂格寫.使用示例:Temp RN R0 ;將R0定義一個(gè)別名Temp12、 RN6513、 ROUT語(yǔ)法格式:{名稱} ROUTROUT偽指令用于給一個(gè)局部變量定義作用范圍。在程序中未使用該偽指令時(shí),局部變量的作用范圍為所在的AREA,而使用ROUT后,局部變量的作為范圍為當(dāng)前ROUT和下一個(gè)ROUT之間。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)課件6614、 NOP語(yǔ)法格式:NOP空操作,不影響CPSR中的條件標(biāo)志位第四章ARM匯編語(yǔ)言程序設(shè)計(jì)課件674.4匯編語(yǔ)言程序格式

1)在ARM(Thumb)匯編語(yǔ)言程序中,以程序段為單位組織代碼。2)段是相對(duì)獨(dú)立的指令或數(shù)據(jù)序列,具有特定的名稱。3)段可以分為代碼段和數(shù)據(jù)段,代碼段的內(nèi)容為執(zhí)行代碼,數(shù)據(jù)段存放代碼運(yùn)行時(shí)需要用到的數(shù)據(jù)。一個(gè)匯編程序至少應(yīng)該有一個(gè)代碼段,當(dāng)程序較長(zhǎng)時(shí),可以分割為多個(gè)代碼段和數(shù)據(jù)段,多個(gè)段在程序編譯鏈接時(shí)最終形成一個(gè)可執(zhí)行的映象文件。

4.4匯編語(yǔ)言程序格式68可執(zhí)行映象文件通常由以下幾部分構(gòu)成:—

一個(gè)或多個(gè)代碼段,代碼段的屬性為只讀?!?/p>

零個(gè)或多個(gè)包含初始化數(shù)據(jù)的數(shù)據(jù)段,數(shù)據(jù)段的屬性為可讀寫?!?/p>

零個(gè)或多個(gè)不包含初始化數(shù)據(jù)的數(shù)據(jù)段,數(shù)據(jù)段的屬性為可讀寫。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)課件69 AREA Init,CODE,READONLY ENTRYStart LDR R0,=0x3FF5000 LDR R1,0xFF STR R1,[R0] LDR R0,=0x3FF5008 LDR R1,0x01 STR R1,[R0] ┉┉ END

;;每一個(gè)匯編源程序結(jié)尾處都必須有一條END偽指令,指示程序的結(jié)束。 AREA Init,CODE,READONLY70匯編語(yǔ)言的子程序調(diào)用

在ARM匯編語(yǔ)言程序中,子程序的調(diào)用一般是通過(guò)BL指令來(lái)實(shí)現(xiàn)的。在程序中,使用指令: BL 子程序名

該指令完成:將子程序的返回地址存放在連接寄存器LR中,同時(shí)將程序計(jì)數(shù)器PC指向子程序的入口點(diǎn),當(dāng)子程序執(zhí)行完畢需要返回調(diào)用處時(shí),只需要將存放在LR中的返回地址重新拷貝給程序計(jì)數(shù)器PC即可。

在調(diào)用子程序的同時(shí),也可以完成參數(shù)的傳遞和從子程序返回運(yùn)算的結(jié)果,通常可以使用寄存器R0~R3完成。匯編語(yǔ)言的子程序調(diào)用71 子程序調(diào)用實(shí)例: AREA Init,CODE,READONLY ENTRYStart BL PRINT_TEXT ┉┉PRINT_TEXT ┉┉ MOV PC,BL ┉┉ END 子程序調(diào)用實(shí)例: 724.6 匯編語(yǔ)言程序示例I_ISPC EQU0x1e00024EXTINTPND EQU0x1d20054 AREAMyIRQ_ISR,CODE,READONLY CODE32;ARM代碼 EXPORTReIn_EINT4567ISRReIn_EINT4567ISR

sublr,lr,#44.6 匯編語(yǔ)言程序示例73

ldrr0,=0x1d2002c;rPDATE地址為0x1d2002c

ldrr1,[r0];將rPDATE的內(nèi)容送給r1

orrr1,r1,#8

strr1,[r0];將((rPDATE)&&0x1f7)的結(jié)果送給rPDATE,PE3==1鈴不響

ldrr3,=0x4000001;局部標(biāo)號(hào)

subr3,r3,#1cmpr3,#0bne%b1;%表示對(duì)局部標(biāo)號(hào)的引用

movspc,lr

ENDldrr0,=0x1d2002c;rPDATE地74第四章ARM匯編語(yǔ)言程序設(shè)計(jì)ADS1.2環(huán)境下的匯編語(yǔ)句格式1.ARM匯編中,所有標(biāo)號(hào)必須在一行的頂格書寫,其后面不要添加“:”,而所有指令均不能頂格書寫。2.ARM匯編器對(duì)標(biāo)識(shí)符大小寫敏感,書寫標(biāo)號(hào)及指令時(shí)字母大小寫要一致,在ARM匯編程序中,一個(gè)ARM指令、偽指令、寄存器名可以全部為大寫字母,也可以全部為小寫字母,但不要大小寫混合使用(如果混合使用也能編譯通過(guò),如movSP,r6能編譯通過(guò),但是moVSP,r6則出現(xiàn)錯(cuò)誤)。3.注釋可以在一行的頂格書寫。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)ADS1.2環(huán)境下的匯編語(yǔ)句754.1偽操作

在ARM匯編語(yǔ)言程序里,有一些特殊指令助記符,這些助記符與指令系統(tǒng)的助記符不同,沒(méi)有相對(duì)應(yīng)的操作碼,通常稱這些特殊指令助記符為偽指令他們所完成的操作稱為偽操作。ARM指令系統(tǒng)4.1偽操作在ARM匯編語(yǔ)言程序里,76

偽指令:由匯編器處理偽指令是由匯編器對(duì)源程序匯編期間進(jìn)行處理的.偽指令僅在匯編過(guò)程中起作用,一旦匯編結(jié)束,偽指令的使命就完成。

指令:由ARM處理器執(zhí)行ARM指令系統(tǒng)偽指令:由匯編器處理ARM指令系統(tǒng)77在ARM的匯編程序中,有如下幾種偽指令: 符號(hào)定義偽指令、 數(shù)據(jù)定義偽指令、 匯編控制偽指令、 宏指令以及其他偽指令。ARM指令系統(tǒng)在ARM的匯編程序中,有如下幾種偽指令:ARM指令系統(tǒng)784.1.1符號(hào)定義偽指令

符號(hào)定義偽指令用于定義ARM匯編程序中的變量、對(duì)變量賦值以及定義寄存器的別名等操作。ARM指令系統(tǒng)4.1.1符號(hào)定義偽指令A(yù)RM指令系統(tǒng)794.1.1符號(hào)定義偽指令

常見的符號(hào)定義偽指令有如下幾種:— 用于定義全局變量的GBLA、GBLL和GBLS?!?用于定義局部變量的LCLA、LCLL和LCLS?!?用于對(duì)變量賦值的SETA、SETL、SETS?!?為通用寄存器列表定義名稱的RLIST。ARM指令系統(tǒng)4.1.1符號(hào)定義偽指令常見的符號(hào)定義偽指令有如下幾種:801、 GBLA、GBLL和GBLS語(yǔ)法格式:GBLA(GBLL或GBLS) 全局變量名GBLA、GBLL和GBLS偽指令用于定義一個(gè)ARM程序中的全局變量,并將其初始化。GBLA偽指令用于定義一個(gè)全局的數(shù)字變量,并初始化為0;GBLL偽指令用于定義一個(gè)全局的邏輯變量,并初始化為F(假);GBLS偽指令用于定義一個(gè)全局的字符串變量,并初始化為空;由于以上三條偽指令用于定義全局變量,因此在整個(gè)程序范圍內(nèi)變量名必須唯一。1、 GBLA、GBLL和GBLS81

GBLA Test1 ;定義一個(gè)全局的數(shù)字變量,變量名為Test1 Test1 SETA 0xaa ;將該變量賦值為0xaa GBLL Test2 ;定義一個(gè)全局的邏輯變量,變量名為Test2 Test2 SETL {TRUE} ;將該變量賦值為真 GBLS Test3 ;定義一個(gè)全局的字符串變量,變量名為Test3 Test3 SETS "Testing" ;將該變量賦值為"Testing" GBLA Test1 822、 LCLA、LCLL和LCLS語(yǔ)法格式:LCLA(LCLL或LCLS) 局部變量名LCLA、LCLL和LCLS偽指令用于定義一個(gè)ARM程序中的局部變量,并將其初始化。其中:LCLA偽指令用于定義一個(gè)局部的數(shù)字變量,并初始化為0;LCLL偽指令用于定義一個(gè)局部的邏輯變量,并初始化為F(假);LCLS偽指令用于定義一個(gè)局部的字符串變量,并初始化為空; 以上三條偽指令用于聲明局部變量ARM指令系統(tǒng)2、 LCLA、LCLL和LCLSARM指令系統(tǒng)83使用示例: LCLA Test4 ;聲明一個(gè)局部的數(shù)字變量,變量名為Test4 Test3 SETA 0xaa ;將該變量賦值為0xaa LCLL Test5 ;聲明一個(gè)局部的邏輯變量,變量名為Test5 Test4 SETL {TRUE};將該變量賦值為真 LCLS Test6 ;定義一個(gè)局部的字符串變量,變量名為Test6 Test6 SETS “Testing” ;將該變量賦值為“Testing”使用示例:843、 SETA、SETL和SETS語(yǔ)法格式:變量名 SETA(SETL或SETS)表達(dá)式

偽指令SETA、SETL、SETS給一個(gè)已經(jīng)定義的全局變量或局部變量賦值。SETA偽指令用于給一個(gè)數(shù)學(xué)變量賦值;SETL偽指令用于給一個(gè)邏輯變量賦值;SETS偽指令用于給一個(gè)字符串變量賦值;

其中,變量名為已經(jīng)定義過(guò)的全局變量或局部變量,表達(dá)式為將要賦給變量的值。3、 SETA、SETL和SETS85使用示例:

LCLA Test3 ;聲明一個(gè)局部的數(shù)字變量,變量名為Test3 Test3

SETA 0xaa;將該變量賦值為0xaa LCLL Test4 ;聲明一個(gè)局部的邏輯變量,變量名為Test4 Test4

SETL {TRUE} ;將該變量賦值為真ARM指令系統(tǒng)使用示例:ARM指令系統(tǒng)864、 RLIST名稱 RLIST {寄存器列表}RLIST偽指令可用于對(duì)一個(gè)通用寄存器列表定義名稱,使用該偽指令定義的名稱可在ARM指令LDM/STM中使用。在LDM/STM指令中,列表中的寄存器訪問(wèn)次序?yàn)楦鶕?jù)寄存器的編號(hào)由低到高,而與列表中的寄存器排列次序無(wú)關(guān)。4、 RLIST87RLIST使用示例:RegList RLIST {R0-R5,R8,R10}

將寄存器列表名稱定義為RegList,可在ARM指令LDM/STM中通過(guò)該名稱訪問(wèn)寄存器列表。例:RegList RLIST {r1-r13};必須頂格開始寫LDMIAr0,RegListRLIST使用示例:884.1.2數(shù)據(jù)定義(DataDefinition)偽指令

數(shù)據(jù)定義偽指令一般用于為特定的數(shù)據(jù)分配存儲(chǔ)單元,同時(shí)可完成已分配存儲(chǔ)單元的初始化。常見的數(shù)據(jù)定義偽指令有如下幾種:— DCB 用于分配一片連續(xù)的字節(jié)存儲(chǔ)單元并用指定的數(shù)據(jù)初始化?!?DCW(DCWU) 用于分配一片連續(xù)的半字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化?!?DCFD(DCFDU)用于為雙精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。ARM指令系統(tǒng)4.1.2數(shù)據(jù)定義(DataDefinition)偽指89— DCFS(DCFSU) 用于為單精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化?!?DCQ(DCQU) 用于分配一片以8字節(jié)為單位的連續(xù)的存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。DCD(DCDU)用于分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。SPACE 用于分配一片連續(xù)的存儲(chǔ)單元MAP用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表首地址FIELD用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表的數(shù)據(jù)域— DCFS(DCFSU) 用于為單精度的浮點(diǎn)數(shù)分配一片連續(xù)901、 DCB(DCB也可用“=”代替)語(yǔ)法格式:標(biāo)號(hào) DCB 表達(dá)式

DCB偽指令用于分配一片連續(xù)的字節(jié)存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。其中,表達(dá)式可以為0~255的數(shù)字或字符串。使用示例:Str DCB "Thisisatest!" ;分配一片連續(xù)的字節(jié)存儲(chǔ)單元并初始化。1、 DCB(DCB也可用“=”代替)912、 DCD(或DCDU,DCD也可用“&”代替)語(yǔ)法格式:標(biāo)號(hào) DCD(或DCDU) 表達(dá)式

DCD(或DCDU)偽指令用于分配一片連續(xù)的字存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。其中,表達(dá)式可以為程序標(biāo)號(hào)或數(shù)字表達(dá)式。

用DCD分配的字存儲(chǔ)單元是字對(duì)齊的,而用DCDU分配的字存儲(chǔ)單元并不嚴(yán)格字對(duì)齊。使用示例:DataTest DCD 4,5,6 ;分配一片連續(xù)的字存儲(chǔ)單元并初始化。2、 DCD(或DCDU,DCD也可用“&”代替)923、 SPACE(SPACE也可用“%”代替)語(yǔ)法格式:標(biāo)號(hào) SPACE 表達(dá)式SPACE偽指令用于分配一片連續(xù)的存儲(chǔ)區(qū)域并初始化為0。其中,表達(dá)式為要分配的字節(jié)數(shù)。使用示例:DataSpace SPACE 100 ;分配連續(xù)100字節(jié)的存儲(chǔ)單元并初始化為0。3、 SPACE(SPACE也可用“%”代替)934、 MAP,MAP也可用“^”代替。語(yǔ)法格式:MAP 表達(dá)式{,基址寄存器}MAP偽指令用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表的首地址。表達(dá)式可以為程序中的標(biāo)號(hào)或數(shù)學(xué)表達(dá)式,基址寄存器為可選項(xiàng),當(dāng)基址寄存器選項(xiàng)不存在時(shí),表達(dá)式的值即為內(nèi)存表的首地址,當(dāng)該選項(xiàng)存在時(shí),內(nèi)存表的首地址為表達(dá)式的值與基址寄存器的和。4、 MAP,MAP也可用“^”代替。94MAP偽指令通常與FIELD偽指令配合使用來(lái)定義結(jié)構(gòu)化的內(nèi)存表。使用示例:MAP 0x100,R0 ;定義結(jié)構(gòu)化內(nèi)存表首地址的值為0x100+R0。第四章ARM匯編語(yǔ)言程序設(shè)計(jì)課件955、 FILEDFILED也可用“#”代替。語(yǔ)法格式:標(biāo)號(hào) FIELD 表達(dá)式FIELD偽指令用于定義一個(gè)結(jié)構(gòu)化內(nèi)存表中的數(shù)據(jù)域。表達(dá)式的值為當(dāng)前數(shù)據(jù)域在內(nèi)存表中所占的字節(jié)數(shù)。

FIELD偽指令常與MAP偽指令配合使用來(lái)定義結(jié)構(gòu)化的內(nèi)存表。MAP偽指令定義內(nèi)存表的首地址,F(xiàn)IELD偽指令定義內(nèi)存表中的各個(gè)數(shù)據(jù)域,并可以為每個(gè)數(shù)據(jù)域指定一個(gè)標(biāo)號(hào)供其他的指令引用。5、 FILEDFILED也可用“#”代替。96注意MAP和FIELD偽指令僅用于定義數(shù)據(jù)結(jié)構(gòu),并不實(shí)際分配存儲(chǔ)單元。使用示例:MAP 0x100 ;定義結(jié)構(gòu)化內(nèi)存表首地址的值為0x100。A FIELD 16 ;定義A的長(zhǎng)度為16字節(jié),位置為0x100B FIELD 32 ;定義B的長(zhǎng)度為32字節(jié),位置為0x110S FIELD 256 ;定義S的長(zhǎng)度為256字節(jié),位置為0x130注意MAP和FIELD偽指令僅用于定義數(shù)據(jù)結(jié)構(gòu),并不實(shí)際分97s3c44b0xARM芯片的初始化源程序:AREARamData,DATA,READWRITE^(_IRQ_BASEADDRESS);^即MAPHandleReset#4;#即DCDHandleUndef#4HandleSWI#4HandlePabort#4HandleDabort#4HandleReserved#4HandleIRQ#4HandleFIQ#4s3c44b0xARM芯片的初始化源程序:984.1.3匯編控制偽指令匯編控制偽指令用于控制匯編程序的執(zhí)行流程常用的匯編控制偽指令包括:— IF、ELSE、ENDIF— WHILE、WEND— MACRO、MEND— MEXITARM指令系統(tǒng)4.1.3匯編控制偽指令匯編控制991IF、ELSE、ENDIF(等價(jià)于[|])語(yǔ)法格式:IF 邏輯表達(dá)式 指令序列1ELSE 指令序列2ENDIFIF、ELSE、ENDIF偽指令能根據(jù)條件的成立與否決定是否執(zhí)行某個(gè)指令序列。當(dāng)IF后面的邏輯表達(dá)式為真,則執(zhí)行指令序列1,否則執(zhí)行指令序列2。其中,ELSE及指令序列2可以沒(méi)有,此時(shí),當(dāng)IF后面的邏輯表達(dá)式為真,則執(zhí)行指令序列1,否則繼續(xù)執(zhí)行后面的指令。1IF、ELSE、ENDIF(等價(jià)于[|])100IF、ELSE、ENDIF偽指令可以嵌套使用。使用示例: GBLL

Test;聲明一個(gè)全局的邏輯變量,變量名為Test ……

IF

Test=TRUE

指令序列1

ELSE

指令序列2

ENDIFIF、ELSE、ENDIF偽指令可以嵌套使用。101s3c44b0xARM芯片的初始化源程序:

GBLLTHUMBCODE;globallogicalvariable,init.to{false}[{CONFIG}=16 ;[=if,{CONFIG}:build-invariabledefinedbyARM;assembler,16=assembingThumbcode.THUMBCODESETL {TRUE};SETL:setthelogicvariableCODE32;changingfromThumbstatetoArmstate|;|=elseTHUMBCODESETL {FALSE}];]=endifs3c44b0xARM芯片的初始化源程序:1022、 WHILE、WEND語(yǔ)法格式:WHILE 邏輯表達(dá)式 指令序列WEND

WHILE、WEND偽指令可以嵌套使用。WHILE、WEND偽指令能根據(jù)條件的成立與否決定是否循環(huán)執(zhí)行某個(gè)指令序列。當(dāng)WHILE后面的邏輯表達(dá)式為真,則執(zhí)行指令序列,該指令序列執(zhí)行完畢后,再判斷邏輯表達(dá)式的值,若為真則繼續(xù)執(zhí)行,一直到邏輯表達(dá)式的值為假。2、 WHILE、WEND103使用示例:

GBLA

Counter ;聲明一個(gè)全局的數(shù)學(xué)變量,變量名為CounterCounter SETA 3 ;由變量Counter控制循環(huán)次數(shù)……

WHILE Counter<10

指令序列

WEND使用示例:1043、 MACRO、MEND語(yǔ)法格式:

MACRO$標(biāo)號(hào) 宏名

$參數(shù)1,$參數(shù)2,…… 指令序列MEND

MACRO、MEND偽指令可以將一段代碼定義為一個(gè)整體,稱為宏指令,然后就可以在程序中通過(guò)宏指令多次調(diào)用該段代碼。其中,$標(biāo)號(hào)在宏指令被展開時(shí),標(biāo)號(hào)會(huì)被替換為用戶定義的符號(hào);

宏指令可以使用一個(gè)或多個(gè)參數(shù),當(dāng)宏指令被展開時(shí),這些參數(shù)被相應(yīng)的值替換。MACRO、MEND偽指令可以嵌套使用。3、 MACRO、MEND105s3c44b0x初始化程序里的例子:

MACRO$HandlerLabelHANDLER$HandleLabel$HandlerLabelsub sp,sp,#4

stmfd

sp!,{r0} ldr r0,=$HandleLabelldr r0,[r0] str r0,[sp,#4] ldmfdsp!,{r0,pc} MENDs3c44b0x初始化程序里的例子:106HandlerEINT4567 HANDLERHandleEINT4567將被匯編成如下語(yǔ)句:HandlerEINT4567sub sp,sp,#4 stmfdsp!,{r0} ldr r0,=HandleEINT4567;ldr r0,[r0] str r0,[sp,#4]ldmfdsp!,{r0,pc}HandlerEINT4567 HANDLERHandl107HandleEINT4567的定義:

^(_IRQ_BASEADDRESS+0x100)HandleADC#4………………..HandleEINT4567#4HandleEINT4567的定義:108圖1

執(zhí)行subsp,sp,#4時(shí)堆棧的情況圖1執(zhí)行subsp,sp,#4時(shí)堆棧的情況109圖2

執(zhí)行stmfdsp!,{r0},將R0壓入堆棧后綴“!”表示最后的地址寫回到sp中圖2執(zhí)行stmfdsp!,{r0},將R0壓入110圖3執(zhí)行strr0,[sp,#4],將R0壓入堆棧(R0)=HandleXXX處的內(nèi)容,不改變SP的值。

圖3執(zhí)行strr0,[sp,#4],將R0壓入堆棧111圖4ldmfdsp!,{r0,pc}

;彈出兩個(gè)字,分別送到R0,PC

圖4ldmfdsp!,{r0,pc}

;彈出兩個(gè)字,1124、 MEXIT語(yǔ)法格式:MEXITMEXIT用于從宏定義中跳轉(zhuǎn)出去。ARM指令系統(tǒng)4、 MEXITARM指令系統(tǒng)1134.1.6其他常用的偽指令還有一些其他的偽指令,在匯編程序中經(jīng)常會(huì)被使用,包括以下幾條:AREAALIGNCODE16、CODE32ENTRYENDEQUEXPORT(或GLOBAL)IMPORTEXTERNGET(或INCLUDE)INCBINRNROUT4.1.6其他常用的偽指令還有一1141、 AREA語(yǔ)法格式:AREA 段名 屬性1,屬性2,……

AREA偽指令用于定義一個(gè)代碼段或數(shù)據(jù)段。其中,段名若以數(shù)字開頭,則該段名需用“|”括起來(lái),如|1_test|。屬性字段表示該代碼段(或數(shù)據(jù)段)的相關(guān)屬性,多個(gè)屬性用逗號(hào)分隔。常用的屬性如下:

CODE屬性:用于定義代碼段,默認(rèn)為READONLY。

DATA屬性:用于定義數(shù)據(jù)段,默認(rèn)為READWRITE。1、 AREA115READONLY屬性:指定本段為只讀,代碼段默認(rèn)為READONLY。READWRITE屬性:指定本段為可讀可寫,數(shù)據(jù)段的默認(rèn)屬性為READWRITE。ALIGN屬性:使用方式為ALIGN表達(dá)式。在默認(rèn)時(shí),ELF(可執(zhí)行連接文件)的代碼段和數(shù)據(jù)段是按字對(duì)齊的,表達(dá)式的取值范圍為0~31,相應(yīng)的對(duì)齊方式為2表達(dá)式次方。

ThisisnotthesameasthewaythattheALIGNdirectiveisspecified.

DonotuseALIGN=0orALIGN=1forcodesections.

READONLY屬性:指定本段為只讀,代碼段默認(rèn)為READO116使用示例:AREA Init,CODE,READONLY;該偽指令定義了一個(gè)代碼段,段名為Init,屬性為只讀

AREAMyDATA,DATA,READONLY,ALIGN=14;(以214字節(jié)對(duì)齊.)使用示例:1172、 ALIGN語(yǔ)法格式:ALIGN {表達(dá)式{,偏移量}} ALIGN偽指令可通過(guò)添加填充字節(jié)的方式,使當(dāng)前位置滿足一定的對(duì)其方式|。其中,表達(dá)式的值用于指定對(duì)齊方式,可能的取值為2的冪,如1、2、4、8、16等。若未指定表達(dá)式,則將當(dāng)前位置對(duì)齊到下一個(gè)字的位置。偏移量也為一個(gè)數(shù)字表達(dá)式,如果不是用在AREA里的,應(yīng)該是表達(dá)式+偏移量.2、 ALIGN118將兩個(gè)字節(jié)的數(shù)據(jù)放在同一個(gè)字的第一個(gè)字節(jié)和第四個(gè)字節(jié)中,帶offset的ALIGN對(duì)齊:AREAoffsetFxample,CODEDCB0x31;第一個(gè)字節(jié)保存0x31ALIGN4,3;字對(duì)齊DCB0x32;第四個(gè)字節(jié)保存0x32內(nèi)容是:0x32000031上面的語(yǔ)句ALIGN4,3意義是:字對(duì)齊,然后再加上偏移3將兩個(gè)字節(jié)的數(shù)據(jù)放在同一個(gè)字的第一個(gè)119ExamplesAREAcacheable,CODE,ALIGN=3rout1;code;alignedon8-byteboundary;codeMOVpc,lr;alignedonlyon4-byteboundaryALIGN8;nowalignedon8-byteboundaryrout2;codeExamples1203、 CODE16、CODE32語(yǔ)法格式: CODE16(或CODE32)CODE16偽指令通知編譯器,其后的指令序列為16位的Thumb指令。CODE32偽指令通知編譯器,其后的指令序列為32位的ARM指令。3、 CODE16、CODE32121若在匯編源程序中同時(shí)包含ARM指令和Thumb指令時(shí),可用CODE16偽指令通知編譯器其后的指令序列為16位的Thumb指令,CODE32偽指令通知編譯器其后的指令序列為32位的ARM指令。因此,在使用ARM指令和Thumb指令混合編程的代碼里,可用這兩條偽指令進(jìn)行切換

注意:他們只通知編譯器其后指令的類型,并不能對(duì)處理器進(jìn)行狀態(tài)的切換。若在匯編源程序中同時(shí)包含ARM指令和122

AREA Init,CODE,READONLY…… CODE32;通知編譯器其后的指令為32位的ARM指令 LDR R0,=NEXT+1 ;將跳轉(zhuǎn)地址放入寄存R0 BX R0 ;程序跳轉(zhuǎn)到新的位置執(zhí)行,并將處理器切換到Thumb工作狀態(tài) …… CODE16 ;通知編譯器其后的指令為16位的Thumb指令NEXT LDR R3,=0x3FF ……END ;程序結(jié)束 AREA Init,CODE,READONLY1234、 ENTRY語(yǔ)法格式: ENTRY

ENTRY偽指令用于指定匯編程序的入口點(diǎn)。在一個(gè)完整的匯編程序中至少要有一個(gè)ENTRY(也可以有多個(gè),當(dāng)有多個(gè)ENTRY時(shí),程序的真正入口點(diǎn)由鏈接器指定),但在一個(gè)源文件里最多只能有一個(gè)ENTRY(可以沒(méi)有)。4、 ENTRY124使用示例: AREA Init,CODE,READONLY ENTRY;指定應(yīng)用程序的入口點(diǎn)……使用示例:1255、 END語(yǔ)法格式: END

END偽指令用于通知編譯器已經(jīng)到了源程序的結(jié)尾,用于指示匯編編譯器源文件已結(jié)束.每一個(gè)匯編源文件均要使用一個(gè).使用示例: AREA Init,CODE,READONLY …… END ;指定應(yīng)用程序的結(jié)尾5、 END1266、 EQU,可用“*”代替。語(yǔ)法格式:名稱 EQU 表達(dá)式{,類型}名稱必須頂格寫EQU偽指令用于為程序中的常量、標(biāo)號(hào)等定義一個(gè)等效的字符名稱,類似于C語(yǔ)言中的#define。名稱為EQU偽指令定義的字符名稱,當(dāng)表達(dá)式為32位的常量時(shí),可以指定表達(dá)式的數(shù)據(jù)類型,可以有以下三種類型:CODE16、CODE32和DATA6、 EQU,可用“*”代替。127使用示例:Test EQU 50;定義標(biāo)號(hào)Test的值為50Addr EQU 0x55,CODE32 ;定義Addr的值為0x55,且該處為32位的ARM指令。使用示例:1287、 EXPORT(或GLOBAL)語(yǔ)法格式: EXPORT 標(biāo)號(hào){[WEAK]}

EXPORT偽指令用于在程序中聲明一個(gè)全局的標(biāo)號(hào),該標(biāo)號(hào)可在其他的文件中引用。EXPORT可用GLOBAL代替。

標(biāo)號(hào)在程序中區(qū)分大小寫

[WEAK]選項(xiàng)聲明其他的同名標(biāo)號(hào)優(yōu)先于該標(biāo)號(hào)被引用。7、 EXPORT(或GLOBAL)129使用示例:

AREA Init,CODE,READONLY EXPORT Stest;聲明一個(gè)可全局引用的標(biāo)號(hào)Stest……END

使用示例:1308、 IMPORT語(yǔ)法格式:

IMPORT 標(biāo)號(hào){[WEAK]}

IMPORT偽指令用于通知編譯器要使用的標(biāo)號(hào)在其他的源文件中定義,但要在當(dāng)前源文件中引用,而且無(wú)論當(dāng)前源文件是否引用該標(biāo)號(hào),該標(biāo)號(hào)均會(huì)被加入到當(dāng)前源文件的符號(hào)表中。

[WEAK]選項(xiàng)表示當(dāng)所有的源文件都沒(méi)有定義這樣一個(gè)標(biāo)號(hào)時(shí),編譯器也不給出錯(cuò)誤信息,在多數(shù)情況下將該標(biāo)號(hào)置為0,若該標(biāo)號(hào)為B或BL指令引用,則將B或BL指令置為NOP操作。8、 IMPORT131使用示例: AREA Init,CODE,READONLY IMPORT Main ;通知編譯器當(dāng)前文件要引用標(biāo)號(hào)Main,但Mai

溫馨提示

  • 1. 本站所有資源如無(wú)特殊說(shuō)明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁(yè)內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒(méi)有圖紙預(yù)覽就沒(méi)有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫(kù)網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。

評(píng)論

0/150

提交評(píng)論