在ARM匯編語言程序里,有一些特殊指令助記符,這些助記符與指令系統(tǒng)的助記符不同,沒有相對應的操作碼,通常稱這些特殊指令助記符為偽指令,他們所完成的操作稱為偽操作。偽指令在源程序中的作用是為完成匯編程序作各種準備工作的,這些偽指令僅在匯編過程中起作用,一旦匯編結束,偽指令的使命就完成。在ARM的匯編程序中,有如下幾種偽指令:符號定義偽指令、數(shù)據(jù)定義偽指令、匯編控制偽指令、宏指令以及其他偽指令。
符號定義(SymbolDefinition)偽指令
符號定義偽指令用于定義ARM匯編程序中的變量、對變量賦值以及定義寄存器的別名等操作。
常見的符號定義偽指令有如下幾種:
—用于定義全局變量的GBLA、GBLL和GBLS。
—用于定義局部變量的LCLA、LCLL和LCLS。
—用于對變量賦值的SETA、SETL、SETS。
—為通用寄存器列表定義名稱的RLIST。
1、GBLA、GBLL和GBLS
語法格式:
GBLA(GBLL或GBLS)全局變量名
GBLA、GBLL和GBLS偽指令用于定義一個ARM程序中的全局變量,并將其初始化。其中:
GBLA偽指令用于定義一個全局的數(shù)字變量,并初始化為0;
GBLL偽指令用于定義一個全局的邏輯變量,并初始化為F(假);
GBLS偽指令用于定義一個全局的字符串變量,并初始化為空;
由于以上三條偽指令用于定義全局變量,因此在整個程序范圍內(nèi)變量名必須唯一。
使用示例:
GBLATest1;定義一個全局的數(shù)字變量,變量名為Test1
Test1SETA0xaa;將該變量賦值為0xaa
GBLLTest2;定義一個全局的邏輯變量,變量名為Test2
Test2SETL{TRUE};將該變量賦值為真
GBLSTest3;定義一個全局的字符串變量,變量名為Test3
Test3SETS“Testing”;將該變量賦值為“Testing”
本文引用地址:http://www.biyoush.com/article/201611/322954.htm2、LCLA、LCLL和LCLS
語法格式:
LCLA(LCLL或LCLS)局部變量名
LCLA、LCLL和LCLS偽指令用于定義一個ARM程序中的局部變量,并將其初始化。其中:
LCLA偽指令用于定義一個局部的數(shù)字變量,并初始化為0;
LCLL偽指令用于定義一個局部的邏輯變量,并初始化為F(假);
LCLS偽指令用于定義一個局部的字符串變量,并初始化為空;
以上三條偽指令用于聲明局部變量,在其作用范圍內(nèi)變量名必須唯一。
使用示例:
LCLATest4;聲明一個局部的數(shù)字變量,變量名為Test4
Test3SETA0xaa;將該變量賦值為0xaa
LCLLTest5;聲明一個局部的邏輯變量,變量名為Test5
Test4SETL{TRUE};將該變量賦值為真
LCLSTest6;定義一個局部的字符串變量,變量名為Test6
Test6SETS“Testing”;將該變量賦值為“Testing”
3、SETA、SETL和SETS
語法格式:
變量名SETA(SETL或SETS)表達式
偽指令SETA、SETL、SETS用于給一個已經(jīng)定義的全局變量或局部變量賦值。
SETA偽指令用于給一個數(shù)學變量賦值;
SETL偽指令用于給一個邏輯變量賦值;
SETS偽指令用于給一個字符串變量賦值;
其中,變量名為已經(jīng)定義過的全局變量或局部變量,表達式為將要賦給變量的值。
使用示例:
LCLATest3;聲明一個局部的數(shù)字變量,變量名為Test3
Test3SETA0xaa;將該變量賦值為0xaa
LCLLTest4;聲明一個局部的邏輯變量,變量名為Test4
Test4SETL{TRUE};將該變量賦值為真
4、RLIST
語法格式:
名稱RLIST{寄存器列表}
RLIST偽指令可用于對一個通用寄存器列表定義名稱,使用該偽指令定義的名稱可在ARM指令LDM/STM中使用。在LDM/STM指令中,列表中的寄存器訪問次序為根據(jù)寄存器的編號由低到高,而與列表中的寄存器排列次序無關。
使用示例:
RegListRLIST{R0-R5,R8,R10};將寄存器列表名稱定義為RegList,可在ARM指令LDM/STM中通過該名稱訪問寄存器列表。
數(shù)據(jù)定義(DataDefinition)偽指令
數(shù)據(jù)定義偽指令一般用于為特定的數(shù)據(jù)分配存儲單元,同時可完成已分配存儲單元的初始化。
常見的數(shù)據(jù)定義偽指令有如下幾種:
—DCB用于分配一片連續(xù)的字節(jié)存儲單元并用指定的數(shù)據(jù)初始化。
—DCW(DCWU)用于分配一片連續(xù)的半字存儲單元并用指定的數(shù)據(jù)初始化。
—DCD(DCDU)用于分配一片連續(xù)的字存儲單元并用指定的數(shù)據(jù)初始化。
—DCFD(DCFDU)用于為雙精度的浮點數(shù)分配一片連續(xù)的字存儲單元并用指定的數(shù)據(jù)初始
化。
—DCFS(DCFSU)用于為單精度的浮點數(shù)分配一片連續(xù)的字存儲單元并用指定的數(shù)據(jù)初
始化。
—DCQ(DCQU)用于分配一片以8字節(jié)為單位的連續(xù)的存儲單元并用指定的數(shù)據(jù)初始
化。
—SPACE用于分配一片連續(xù)的存儲單元
—MAP用于定義一個結構化的內(nèi)存表首地址
—FIELD用于定義一個結構化的內(nèi)存表的數(shù)據(jù)域
1、DCB
語法格式:
標號DCB表達式
DCB偽指令用于分配一片連續(xù)的字節(jié)存儲單元并用偽指令中指定的表達式初始化。其中,表達式可以為0~255的數(shù)字或字符串。DCB也可用“=”代替。
使用示例:
StrDCB“Thisisatest!”;分配一片連續(xù)的字節(jié)存儲單元并初始化。
2、DCW(或DCWU)
語法格式:
標號DCW(或DCWU)表達式
DCW(或DCWU)偽指令用于分配一片連續(xù)的半字存儲單元并用偽指令中指定的表達式初始化。
其中,表達式可以為程序標號或數(shù)字表達式。。
用DCW分配的字存儲單元是半字對齊的,而用DCWU分配的字存儲單元并不嚴格半字對齊。
使用示例:
DataTestDCW1,2,3;分配一片連續(xù)的半字存儲單元并初始化。
3、DCD(或DCDU)
語法格式:
標號DCD(或DCDU)表達式
DCD(或DCDU)偽指令用于分配一片連續(xù)的字存儲單元并用偽指令中指定的表達式初始化。其中,表達式可以為程序標號或數(shù)字表達式。DCD也可用“&”代替。
用DCD分配的字存儲單元是字對齊的,而用DCDU分配的字存儲單元并不嚴格字對齊。
使用示例:
DataTestDCD4,5,6;分配一片連續(xù)的字存儲單元并初始化。
4、DCFD(或DCFDU)
語法格式:
標號DCFD(或DCFDU)表達式
DCFD(或DCFDU)偽指令用于為雙精度的浮點數(shù)分配一片連續(xù)的字存儲單元并用偽指令中指定的表達式初始化。每個雙精度的浮點數(shù)占據(jù)兩個字單元。用DCFD分配的字存儲單元是字對齊的,而用DCFDU分配的字存儲單元并不嚴格字對齊。
使用示例:
FDataTestDCFD2E115,-5E7;分配一片連續(xù)的字存儲單元并初始化為指定的雙精度數(shù)。
5、DCFS(或DCFSU)
語法格式:
標號DCFS(或DCFSU)表達式
DCFS(或DCFSU)偽指令用于為單精度的浮點數(shù)分配一片連續(xù)的字存儲單元并用偽指令中指定的表達式初始化。每個單精度的浮點數(shù)占據(jù)一個字單元。用DCFS分配的字存儲單元是字對齊的,而用DCFSU分配的字存儲單元并不嚴格字對齊。
使用示例:
FDataTestDCFS2E5,-5E-7;分配一片連續(xù)的字存儲單元并初始化為指定的單精度數(shù)。
6、DCQ(或DCQU)
語法格式:
標號DCQ(或DCQU)表達式
DCQ(或DCQU)偽指令用于分配一片以8個字節(jié)為單位的連續(xù)存儲區(qū)域并用偽指令中指定的表達式初始化。
用DCQ分配的存儲單元是字對齊的,而用DCQU分配的存儲單元并不嚴格字對齊。
使用示例:
DataTestDCQ100;分配一片連續(xù)的存儲單元并初始化為指定的值。
7、SPACE
語法格式:
標號SPACE表達式
SPACE偽指令用于分配一片連續(xù)的存儲區(qū)域并初始化為0。其中,表達式為要分配的字節(jié)數(shù)。
SPACE也可用“%”代替。
使用示例:
DataSpaceSPACE100;分配連續(xù)100字節(jié)的存儲單元并初始化為0。
8、MAP
語法格式:
MAP表達式{,基址寄存器}
MAP偽指令用于定義一個結構化的內(nèi)存表的首地址。MAP也可用“^”代替。
表達式可以為程序中的標號或數(shù)學表達式,基址寄存器為可選項,當基址寄存器選項不存在時,表達式的值即為內(nèi)存表的首地址,當該選項存在時,內(nèi)存表的首地址為表達式的值與基址寄存器的和。
MAP偽指令通常與FIELD偽指令配合使用來定義結構化的內(nèi)存表。
使用示例:
MAP0x100,R0;定義結構化內(nèi)存表首地址的值為0x100+R0。
9、FILED
語法格式:
標號FIELD表達式
FIELD偽指令用于定義一個結構化內(nèi)存表中的數(shù)據(jù)域。FILED也可用“#”代替。
表達式的值為當前數(shù)據(jù)域在內(nèi)存表中所占的字節(jié)數(shù)。
FIELD偽指令常與MAP偽指令配合使用來定義結構化的內(nèi)存表。MAP偽指令定義內(nèi)存表的首地址,FIELD偽指令定義內(nèi)存表中的各個數(shù)據(jù)域,并可以為每個數(shù)據(jù)域指定一個標號供其他的指令引用。
注意MAP和FIELD偽指令僅用于定義數(shù)據(jù)結構,并不實際分配存儲單元。
使用示例:
MAP0x100;定義結構化內(nèi)存表首地址的值為0x100。
AFIELD16;定義A的長度為16字節(jié),位置為0x100
BFIELD32;定義B的長度為32字節(jié),位置為0x110
SFIELD256;定義S的長度為256字節(jié),位置為0x130
匯編控制(AssemblyControl)偽指令
匯編控制偽指令用于控制匯編程序的執(zhí)行流程,常用的匯編控制偽指令包括以下幾條:
—IF、ELSE、ENDIF
—WHILE、WEND
—MACRO、MEND
—MEXIT
1、IF、ELSE、ENDIF
語法格式:
IF邏輯表達式
指令序列1
ELSE
指令序列2
ENDIF
IF、ELSE、ENDIF偽指令能根據(jù)條件的成立與否決定是否執(zhí)行某個指令序列。當IF后面的邏輯表達式為真,則執(zhí)行指令序列1,否則執(zhí)行指令序列2。其中,ELSE及指令序列2可以沒有,此時,當IF后面的邏輯表達式為真,則執(zhí)行指令序列1,否則繼續(xù)執(zhí)行后面的指令。
IF、ELSE、ENDIF偽指令可以嵌套使用。
使用示例:
GBLLTest;聲明一個全局的邏輯變量,變量名為Test……
IFTest=TRUE
指令序列1
ELSE
指令序列2
ENDIF
2、WHILE、WEND
語法格式:
WHILE邏輯表達式
指令序列
WEND
WHILE、WEND偽指令能根據(jù)條件的成立與否決定是否循環(huán)執(zhí)行某個指令序列。當WHILE后面的邏輯表達式為真,則執(zhí)行指令序列,該指令序列執(zhí)行完畢后,再判斷邏輯表達式的值,若為真則繼續(xù)執(zhí)行,一直到邏輯表達式的值為假。
WHILE、WEND偽指令可以嵌套使用。
使用示例:
GBLACounter;聲明一個全局的數(shù)學變量,變量名為Counter
CounterSETA3;由變量Counter控制循環(huán)次數(shù)
……
WHILECounter<10
指令序列
WEND
3、MACRO、MEND
語法格式:
$標號宏名$參數(shù)1,$參數(shù)2,……
指令序列
MEND
MACRO、MEND偽指令可以將一段代碼定義為一個整體,稱為宏指令,然后就可以在程序中通過宏指令多次調(diào)用該段代碼。其中,$標號在宏指令被展開時,標號會被替換為用戶定義的符號,宏指令可以使用一個或多個參數(shù),當宏指令被展開時,這些參數(shù)被相應的值替換。
宏指令的使用方式和功能與子程序有些相似,子程序可以提供模塊化的程序設計、節(jié)省存儲空間并提高運行速度。但在使用子程序結構時需要保護現(xiàn)場,從而增加了系統(tǒng)的開銷,因此,在代碼較短且需要傳遞的參數(shù)較多時,可以使用宏指令代替子程序。
包含在MACRO和MEND之間的指令序列稱為宏定義體,在宏定義體的第一行應聲明宏的原型(包含宏名、所需的參數(shù)),然后就可以在匯編程序中通過宏名來調(diào)用該指令序列。在源程序被編譯時,匯編器將宏調(diào)用展開,用宏定義中的指令序列代替程序中的宏調(diào)用,并將實際參數(shù)的值傳遞給宏定義中的形式參數(shù)。
MACRO、MEND偽指令可以嵌套使用。
評論