必須掌握的基礎(chǔ)知識(shí):ARM 匯編的一些規(guī)范
A.5.1文件格式
ARM 源程序文件(即源文件)為文件格式,可以使用任一文本編輯器編寫程序代碼。
在一個(gè)項(xiàng)目中,至少要有一個(gè)匯編源文件或C 程序文件,可以有多個(gè)匯編源文件或多個(gè)C 程序文件,或者C 程序文件和匯編文件兩者的組合。
A.5.2ARM 匯編的一些規(guī)范
(1)匯編語句格式
ARM 匯編中,所有標(biāo)號(hào)必須在一行的頂格書寫,其后面不要添加“:”,而所有指令均不能頂格書寫。ARM 匯編器對(duì)標(biāo)識(shí)符大小寫敏感,書寫標(biāo)號(hào)及指令時(shí)字母大小寫要一致,在ARM 匯編程序中,一個(gè)ARM 指令、偽指令、寄存器名可以全部為大寫字母,也可以全部為小寫字母,但不要大小寫混合使用。注釋使用“;”,注釋內(nèi)容由“;”開始到此行結(jié)束,注釋可以在一行的頂格書寫。
格式:[標(biāo)號(hào)] <指令|條件|S> <操作數(shù)>[;注釋]
源程序中允許有空行,適當(dāng)?shù)夭迦肟招锌梢蕴岣咴创a的可讀性。如果單行太長,可以使用字符“”將其分行,“”后不能有任何字符,包括空格和制表符等。對(duì)于變量的設(shè)置,常量的定義,其標(biāo)識(shí)符必須在一行的頂格書寫。
匯編指令正確的例子和錯(cuò)誤的例子如下:
正確的例子:
…
Str1 SETS My string1.” ;設(shè)置字符串變量Str1
Count RN R0 ;定義寄存器名Count
USR_STACK EQU 64 ;定義常量
START LDR R0,=0x1123456 ;R0=0x123456H
MOV R1,#0
LOOP
MOV R2,#3
…
錯(cuò)誤的例子:
START MOV R0,#1 ;標(biāo)號(hào)START 沒有頂格寫
ABC: MOV R1,#2 ;標(biāo)號(hào)后不能帶:
MOV R2,#3 ;命令不允許頂格書寫
loop Mov R2,#3 ;指令中大小寫混合
B Loop ;無法跳轉(zhuǎn)到Loop 標(biāo)號(hào)
(2)標(biāo)號(hào)
在ARM 匯編中,標(biāo)號(hào)代表一個(gè)地址,段內(nèi)標(biāo)號(hào)的地址在匯編時(shí)確定,而段外標(biāo)號(hào)的地址值在連接時(shí)確定。根據(jù)標(biāo)號(hào)的生成方式,可以有以下3 鐘:
基于PC 的標(biāo)號(hào)
基于PC 的標(biāo)號(hào)時(shí)位于目標(biāo)指令前的標(biāo)號(hào)或程序中的數(shù)據(jù)定義偽指令前的標(biāo)號(hào),這種標(biāo)號(hào)在匯編時(shí)將被處理成PC 值加上或減去一個(gè)數(shù)字常量。它常用于表示跳轉(zhuǎn)指令的目標(biāo)地址,或者代碼段中所嵌入的少量數(shù)據(jù)。
基于寄存器的標(biāo)號(hào)
基于寄存器的標(biāo)號(hào)通常用MAP 和FILED 偽指令定義,也可以用于EQU 偽指令定義,這種標(biāo)號(hào)在匯編時(shí)被處理成寄存器的值加上或減去一個(gè)數(shù)字常量。它常用于訪問位于數(shù)據(jù)段中的數(shù)據(jù)。
絕對(duì)地址
絕對(duì)地址是一個(gè)32 為的數(shù)字量,它可以尋址的范圍為0~232-1,可以直接尋址整個(gè)內(nèi)存空間。
(3)局部標(biāo)號(hào)
局部標(biāo)號(hào)主要用于局部范圍代碼中,在宏定義也是很有用的。局部標(biāo)號(hào)是一個(gè)0~99 之間的十進(jìn)制數(shù)字,可重復(fù)定義,局部標(biāo)號(hào)后面可以緊接一個(gè)通常表示該局部變量作用范圍的符號(hào)。局部變量的作用范圍為當(dāng)前段,也可以用偽指令ROUT 來定義局部標(biāo)號(hào)的作用范圍。
局部標(biāo)號(hào)定義格式:N{routname}
其中:N 局部標(biāo)號(hào),為0~99。
routname 局部標(biāo)號(hào)作用范圍的名稱,由ROUT 偽指令定義。
局部標(biāo)號(hào)引用格式:
%{F|B}{A|T} N{routname}
其中: % 表示局部標(biāo)號(hào)引用操作。
F 指示編譯器只向前搜索
B 指示編譯器只向后搜索
A 指示編譯器搜索宏的所有嵌套層次
T 指示編譯器搜索宏的當(dāng)前層
如果F 和B 都沒有指定,則編譯器先向前搜索,再向后搜索。如果A 和T 都沒有指定,編譯器搜索所有從宏的當(dāng)前層次到宏的最高層次,比當(dāng)前層次的層次不再搜索。
如果指定了routname,編譯器向前搜索最近的ROUT 偽指令,若routname 與該ROUT偽指令定義的名稱不匹配,編譯器報(bào)告錯(cuò)誤,匯編失敗。
示例如下:
routintA ROUT
…
3routineA
BEQ %4routineA
BGE %3
4routineA
…
otherstuff ROUT
…
(4)符號(hào)
在ARM 匯編中,符號(hào)可以代表地址、變量、數(shù)字常量。當(dāng)符號(hào)代表地址時(shí)又稱為標(biāo)號(hào),符號(hào)就是變量的變量名、數(shù)字常量的名稱、標(biāo)號(hào),符號(hào)的命名規(guī)則如下:
a. 符號(hào)由大小寫字母、數(shù)字以及下劃線組成;
b. 除局部標(biāo)號(hào)以數(shù)字開頭外,其它的符號(hào)不能以數(shù)字開頭;
c. 符號(hào)區(qū)分大小寫,且所有字符都是有意義的;
d. 符號(hào)在其作用域范圍你必須是唯一的;
e. 符號(hào)不能與系統(tǒng)內(nèi)部或系統(tǒng)預(yù)定義的符號(hào)同名;
f. 符號(hào)不要與指令助記符、偽指令同名。
(5)常量
數(shù)字常數(shù)
數(shù)字常量有三種表示方式:
十進(jìn)制數(shù),如:12,5,876,0。
十六進(jìn)制數(shù),如0x4387,0xFF0, 0x1。
n 進(jìn)制數(shù),用n-XXX 表示,其中n 為2~9,XXX 為具體的數(shù)。如2-010111,8-4363156等。
字符常量
字符常量由一對(duì)單引號(hào)及中間字符串表示,標(biāo)準(zhǔn)C 語言中的轉(zhuǎn)義符也可使用。如果需要包含雙引號(hào)或“”,必須使用“”和”,必須使用“”和$代替。如下示例:
Hello SETS “Hello World!”
Errorl SETS “The parameter ““VFH””error$$2”
布爾常量
布爾常量的邏輯真為{TRUE},邏輯假為{FALSE}。如下示例:
testno SETS {FALSE}
(6)段定義
ARM 匯編程序設(shè)計(jì)采用分段式設(shè)計(jì),一個(gè)ARM 源程序至少需要一個(gè)代碼段,大的程序可以包含多個(gè)代碼段及數(shù)據(jù)段。
ARM 匯編程序經(jīng)過匯編處理后生成一個(gè)可執(zhí)行的映象文件,該文件通常包含下面3部分內(nèi)容:
一個(gè)或多代碼段。代碼段通常是只讀的。
零個(gè)或多個(gè)包含初始化值的數(shù)據(jù)段。這些數(shù)據(jù)段通常是可讀寫的。
零