Load/Store內(nèi)存訪問指令
— LDR 字?jǐn)?shù)據(jù)加載指令 — LDRB 字節(jié)數(shù)據(jù)加載指令 — LDRH 半字?jǐn)?shù)據(jù)加載指令 — STR 字?jǐn)?shù)據(jù)存儲(chǔ)指令 — STRB 字節(jié)數(shù)據(jù)存儲(chǔ)指令 — STRH 半字?jǐn)?shù)據(jù)存儲(chǔ)指令
數(shù)據(jù)處理指令
MOV 數(shù)據(jù)傳送指令,有效數(shù)字不能超過2位16進(jìn)制(8位二進(jìn)制),MOV r2,#0xf800合法,MOV r2,0x1510錯(cuò)誤,執(zhí)行后原數(shù)據(jù)丟失。
MVN 數(shù)據(jù)取反傳送指令 ;取反的有效數(shù)字不能超過2位16進(jìn)制(8位二進(jìn)制),MVN r2,#0xf800合法,MVN r2,0x1510錯(cuò)誤,執(zhí)行后原數(shù)據(jù)丟失。 CMP 比較指令 — CMN 反值比較指令 — TST 位測(cè)試指令 — TEQ 相等測(cè)試指令 — ADD 加法指令 — ADC 帶進(jìn)位加法指令 — SUB 減法指令 — SBC 帶借位減法指令 — RSB 逆向減法指令 — RSC 帶借位的逆向減法指令 — AND 按位與指令 — ORR 按位或指令 — EOR 按位異或指令 — BIC 位清除指令
乘法與乘加指令
— MUL 32位乘法指令 — MLA 32位乘加指令 — SMULL 64位有符號(hào)數(shù)乘法指令 — SMLAL 64位有符號(hào)數(shù)乘加指令 — UMULL 64位無(wú)符號(hào)數(shù)乘法指令 — UMLAL 64位無(wú)符號(hào)數(shù)乘加指令
狀態(tài)寄存器訪問指令
— MRS 程序狀態(tài)寄存器到通用寄存器的數(shù)據(jù)傳送指 — MSR 通用寄存器到程序狀態(tài)寄存器的數(shù)據(jù)傳送指令
移位指令
— LSL 邏輯左移 — ASL 算術(shù)左移 — LSR 邏輯右移 — ASR 算術(shù)右移 — ROR 循環(huán)右移 — RRX 帶擴(kuò)展的循環(huán)右移
跳轉(zhuǎn)指令
— B 跳轉(zhuǎn)指令 — BL 帶返回的跳轉(zhuǎn)指令 — BLX 帶返回和狀態(tài)切換的跳轉(zhuǎn)指令 — BX 帶狀態(tài)切換的跳轉(zhuǎn)指令
協(xié)處理器指令
— LDC 協(xié)處理器數(shù)據(jù)加載指令 — STC 協(xié)處理器數(shù)據(jù)存儲(chǔ)指令 — MCR ARM處理器寄存器到協(xié)處理器寄存器的數(shù)據(jù)傳送指令 — MRC 協(xié)處理器寄存器到ARM處理器寄存器的數(shù)據(jù)傳送指令 — CDP 協(xié)處理器數(shù)操作指令
其他常用的偽指令
— AREA
— ALIGN
— CODE16 、 CODE32
— ENTRY
— END
— EQU
— EXPORT (或 GLOBAL )
— IMPORT
— EXTERN
— GET (或 INCLUDE )
— INCBIN
— RN
— ROUT
1) ARM雜項(xiàng)偽指令
1. ADR偽指令:小范圍的地址讀取偽指令。
ADR指令將基于PC相對(duì)偏移的地址值讀取到寄存器中。在匯編編譯源程序時(shí),ADR偽指令被編譯器替換成一條合適的指令。通常編譯器用一條ADD指令或SUB指令來(lái)實(shí)現(xiàn)該ADR偽指令的功能。
指令格式:ADR{cond} register ,expr
Register 加載的寄存器
Expr 程序相對(duì)偏移或寄存器相對(duì)偏移的表達(dá)式
非字對(duì)齊地址在-255~255字節(jié)范圍內(nèi);
字對(duì)齊地址在-1020~1020字節(jié)范圍內(nèi)。
舉例:
Start MOV R1,#10
ADR R4,start ;相當(dāng)于PC-10后賦值給R4
2. ADRL指令:中等范圍的地址讀取偽指令。
ADRL指令將基于PC相對(duì)偏移的地址值或基于相對(duì)偏移的地址值讀取到寄存器中,比ADR偽指令可讀取更大范圍的地址。在匯編編譯源程序時(shí),ADRL偽指令被編譯器替換成兩條合適的指令。若不能用兩條指令實(shí)現(xiàn)ADRL偽指令功能,則產(chǎn)生錯(cuò)誤,編譯失敗。
指令格式與ADR相同
非字對(duì)齊地址在64K字節(jié)范圍內(nèi);
字對(duì)齊地址在256K字節(jié)范圍內(nèi)。
舉例:
Start MOV R1,#10
ADR R4,start+6000 ;=>ADD R4,PC,#0xe800 ADD R4,R4,#0x254
3. LDR指令 大范圍的地址讀取偽指令
LDR偽指令用于加載32位的立即數(shù)或一個(gè)地址值到指定寄存器。
在匯編編譯源程序時(shí),LDR指令被編譯器替換成一條合適的指令,若加載的常數(shù)未超出MOV或MVN的范圍,則使用MOV或MVN指令代替該LDR偽指令,否則匯編器將常量放入字池(內(nèi)存),并使用一條程序相對(duì)偏移的LDR指令從文字池讀出常量。
指令格式:LDR {cond} register , = expr/label_expr
Expr 32位立即數(shù)
Label_expr 基于PC的地址表達(dá)式或外部表達(dá)式
舉例
LDR R0,=0x123987 ;加載32位立即數(shù)
LDR R0,=DATA_BUF+60 ;加載DATA_BUF地址+60
4. NOP指令
NOP指令產(chǎn)生所需的ARM無(wú)操作代碼。可以使用指令MOV R0,R0。NOP不能有條件使用。執(zhí)行和不執(zhí)行無(wú)操作指令是一樣的,因而不需要有條件執(zhí)行。ALU狀態(tài)不受NOP影響。
2) 符號(hào)定義( Symbol Definit年ion )偽指令
符號(hào)定義偽指令用于定義 ARM 匯編程序中的變量、對(duì)變量賦值以及定義寄存器的別名等操作。
常見的符號(hào)定義偽指令有如下幾種:
Ø 用于定義全局變量的 GBLA 、 GBLL 和 GBLS
Ø 用于定義局部變量的 LCLA 、 LCLL 和 LCLS
Ø 用于對(duì)變量賦值的 SETA 、 SETL 、 SETS
Ø 為通用寄存器列表定義名稱的 RLIST
1. GBLA、GBLL 和GBLS
語(yǔ)法格式:
GBLA ( GBLL 或 GBLS ) 全局變量名
GBLA 、 GBLL 和 GBLS 偽指令用于定義一個(gè) ARM 程序中的全局變量,并將其初始化。其中:
GBLA 偽指令用于定義一個(gè)全局的數(shù)字變量,并初始化為 0 ;
GBLL 偽指令用于定義一個(gè)全局的邏輯變量,并初始化為 F (假);
GBLS 偽指令用于定義一個(gè)全局的字符串變量,并初始化為空;
由于以上三條偽指令用于定義全局變量,因此在整個(gè)程序范圍內(nèi)變量名必須唯一。
使用示例:
GBLA Test1 ;定義一個(gè)全局的數(shù)字變量,變量名為 Test1
Test1 SETA 0xaa ;將該變量賦值為 0xaa
GBLL Test2 ;定義一個(gè)全局的邏輯變量,變量名為 Test2
Test2 SETL {TRUE} ;將該變量賦值為真
GBLS Test3 ;定義一個(gè)全局的字符串變量,變量名為 Test3
Test3 SETS “ Testing ” ;將該變量賦值為 “ Testing ”
2. LCLA、LCLL 和LCLS
語(yǔ)法格式:
LCLA ( LCLL 或 LCLS ) 局部變量名
LCLA 、 LCLL 和 LCLS 偽指令用于定義一個(gè) ARM 程序中的局部變量,并將 其初始化。其中:
LCLA 偽指令用于定義一個(gè)局部的數(shù)字變量,并初始化為 0 ;
LCLL 偽指令用于定義一個(gè)局部的邏輯變量,并初始化為 F (假);
LCLS 偽指令用于定義一個(gè)局部的字符串變量,并初始化為空;
以上三條偽指令用于聲明局部變量,在其作用范圍內(nèi)變量名必須唯一。
使用示例:
LCLA Test4 ;聲明一個(gè)局部的數(shù)字變量,變量名為 Test4
Test3 SETA 0xaa ;將該變量賦值為 0xaa
LCLL Test5 ;聲明一個(gè)局部的邏輯變量,變量名為 Test5
Test4 SETL {TRUE} ;將該變量賦值為真
LCLS Test6 ;定義一個(gè)局部的字符串變量,變量名為 Test6
Test6 SETS “ Testing ” ;將該變量賦值為 “ Testing ”
3. SETA、SETL 和SETS
語(yǔ)法格式:
變量名 SETA ( SETL 或 SETS ) 表達(dá)式
偽指令 SETA、SETL、SETS用于給一個(gè)已經(jīng)定義的全局變量或局部變量賦值。
SETA 偽指令用于給一個(gè)數(shù)學(xué)變量賦值;
SETL 偽指令用于給一個(gè)邏輯變量賦值;
SETS 偽指令用于給一個(gè)字符串變量賦值;
其中,變量名為已經(jīng)定義過的全局變量或局部變量,表達(dá)式為將要賦給變量的值。
使用示例:
LCLA Test3 ;聲明一個(gè)局部的數(shù)字變量,變量名為 Test3
Test3 SETA 0xaa ;將該變量賦值為 0xaa
LCLL Test4 ;聲明一個(gè)局部的邏輯變量,變量名為 Test4
Test4 SETL {TRUE} ;將該變量賦值為真
4. RLIST
語(yǔ)法格式:
名稱 RLIST { 寄存器列表 }
RLIST 偽指令可用于對(duì)一個(gè)通用寄存器列表定義名稱,使用該偽指令定義的名稱可在 ARM 指令 LDM/STM 中使用。在 LDM/STM 指令中,列表中的寄存器訪問次序?yàn)楦鶕?jù)寄存器的編號(hào)由低到高,而與列表中的寄存器排列次序無(wú)關(guān)。
使用示例:
RegList RLIST {R0-R5 , R8 , R10} ;將寄存器列表名稱定義為 RegList ,可在 ARM 指令 LDM/STM中通過該名稱訪問寄存器列表。
3) 數(shù)據(jù)定義( Data Definition )偽指令
數(shù)據(jù)定義偽指令一般用于為特定的數(shù)據(jù)分配存儲(chǔ)單元,同時(shí)可完成已分配存儲(chǔ)單元的初始化。
常見的數(shù)據(jù)定義偽指令有如下幾種:
— DCB 用于分配一片連續(xù)的字節(jié)存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。
— DCW (DCWU) 用于分配一片連續(xù)的半字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。
— DCD (DCDU) 用于分配一片連續(xù)的字存儲(chǔ)單元并用指定的數(shù)據(jù)初始化。
— DCFD (DCFDU)用于為雙精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用指 定的數(shù)據(jù)初始化。
— DCFS DCFSU) 用于為單精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用指 定的數(shù)據(jù)初始化。
— DCQ DCQU) 用于分配一片以 8 字節(jié)為單位的連續(xù)的存儲(chǔ)單元并用指定 的數(shù)據(jù)初始化。
— SPACE 用于分配一片連續(xù)的存儲(chǔ)單元
— MAP 用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表首地址
— FIELD 用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表的數(shù)據(jù)域
1. DCB
語(yǔ)法格式:
標(biāo)號(hào) DCB 表達(dá)式
DCB 偽指令用于分配一片連續(xù)的字節(jié)存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。其中,表達(dá)式可以為 0 ~ 255 的數(shù)字或字符串。 DCB 也可用 “ = ” 代替。
使用示例:
Str DCB “ This is a test ! ” ;分配一片連續(xù)的字節(jié)存儲(chǔ)單元并初始化。
2. DCW(或DCWU)
語(yǔ)法格式:
標(biāo)號(hào) DCW (或 DCWU ) 表達(dá)式
DCW (或 DCWU )偽指令用于分配一片連續(xù)的半字存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。
其中,表達(dá)式可以為程序標(biāo)號(hào)或數(shù)字表達(dá)式。
用 DCW 分配的字存儲(chǔ)單元是半字對(duì)齊的,而用 DCWU 分配的字存儲(chǔ)單元并不嚴(yán)格半字對(duì)齊。
使用示例:
DataTest DCW 1 , 2 , 3 ;分配一片連續(xù)的半字存儲(chǔ)單元并初始化。
3. DCD(或DCDU)
語(yǔ)法格式:
標(biāo)號(hào) DCD (或 DCDU )表達(dá)式
DCD(或 DCDU )偽指令用于分配一片連續(xù)的字存儲(chǔ)單元并用偽指令中指定的 表達(dá)式初始化。其中,表達(dá)式可以為程序標(biāo)號(hào)或數(shù)字表達(dá)式。DCD也可用 “ & ” 代替。
用 DCD 分配的字存儲(chǔ)單元是字對(duì)齊的,而用 DCDU 分配的字存儲(chǔ)單元并不嚴(yán)格 字對(duì)齊。
使用示例:
DataTest DCD 4 , 5 , 6 ;分配一片連續(xù)的字存儲(chǔ)單元并初始化。
4. DCFD(或DCFDU)
語(yǔ)法格式:
標(biāo)號(hào) DCFD (或 DCFDU ) 表達(dá)式
DCFD (或 DCFDU )偽指令用于為雙精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元 并用偽指令中指定的表達(dá)式初始化。每個(gè)雙精度的浮點(diǎn)數(shù)占據(jù)兩個(gè)字單元。用 DCFD 分配的字存儲(chǔ)單元是字對(duì)齊的,而用 DCFDU 分配的字存儲(chǔ)單元并不嚴(yán)格字對(duì)齊。
使用示例:
FDataTest DCFD 2E115 , -5E7 ;分配一片連續(xù)的字存儲(chǔ)單元并初始化為指定的雙精度數(shù)。
5. DCFS(或DCFSU)
語(yǔ)法格式:
標(biāo)號(hào) DCFS (或 DCFSU ) 表達(dá)式
DCFS (或 DCFSU )偽指令用于為單精度的浮點(diǎn)數(shù)分配一片連續(xù)的字存儲(chǔ)單元并用偽指令中指定的表達(dá)式初始化。每個(gè)單精度的浮點(diǎn)數(shù)占據(jù)一個(gè)字單元。用 DCFS 分配的字存儲(chǔ)單元是字對(duì)齊的,而用 DCFSU 分配的字存儲(chǔ)單元并不嚴(yán)格字對(duì)齊。
使用示例:
FDataTest DCFS 2E5 , -5E - 7 ;分配一片連續(xù)的字存儲(chǔ)單元并初始化為指定的單精度數(shù)。
6. DCQ(或DCQU)
語(yǔ)法格式:
標(biāo)號(hào) DCQ (或 DCQU )表達(dá)式
DCQ(或 DCQU )偽指令用于分配一片以 8 個(gè)字節(jié)為單位的連續(xù)存儲(chǔ)區(qū)域并用 偽指令中指定的表達(dá)式初始化。
用 DCQ 分配的存儲(chǔ)單元是字對(duì)齊的,而用 DCQU分配的存儲(chǔ)單元并不嚴(yán)格字齊
使用示例:
DataTest DCQ 100 ;分配一片連續(xù)的存儲(chǔ)單元并初始化為指定的值。
7. SPACE
語(yǔ)法格式:
標(biāo)號(hào) SPACE 表達(dá)式
SPACE 偽指令用于分配一片連續(xù)的存儲(chǔ)區(qū)域并初始化為 0 。其中,表達(dá)式為要分配的字節(jié)數(shù)。
SPACE 也可用 “ % ” 代替。
使用示例:
DataSpace SPACE 100 ;分配連續(xù) 100 字節(jié)的存儲(chǔ)單元并初始化為 0 。
8. MAP
語(yǔ)法格式:
MAP 表達(dá)式 { ,基址寄存器 }
MAP 偽指令用于定義一個(gè)結(jié)構(gòu)化的內(nèi)存表的首地址。MAP 也可用 “ ^ ” 代替。
表達(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á)式的值與基址寄存器的和。
MAP 偽指令通常與 FIELD 偽指令配合使用來(lái)定義結(jié)構(gòu)化的內(nèi)存表。
使用示例:
MAP 0x100 , R0 ;定義結(jié)構(gòu)化內(nèi)存表首地址的值為 0x100 + R0 。
9. FILED
語(yǔ)法格式:
標(biāo)號(hào) FIELD 表達(dá)式
FIELD 偽指令用于定義一個(gè)結(jié)構(gòu)化內(nèi)存表中的數(shù)據(jù)域。FILED也可用“ # ” 代替。
表達(dá)式的值為當(dāng)前數(shù)據(jù)域在內(nèi)存表中所占的字節(jié)數(shù)。
FIELD 偽指令常與 MAP 偽指令配合使用來(lái)定義結(jié)構(gòu)化的內(nèi)存表。 MAP 偽指令定義內(nèi)存表的首地址, FIELD 偽指令定義內(nèi)存表中的各個(gè)數(shù)據(jù)域,并可以為每個(gè)數(shù)據(jù)域指定一個(gè)標(biāo)號(hào)供其他的指令引用。
注意 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é),位置為 0x100
B FIELD 32 ;定義 B 的長(zhǎng)度為 32 字節(jié),位置為 0x110
S FIELD 256 ;定義 S 的長(zhǎng)度為 256 字節(jié),位置為 0x130
4) 匯編控制( Assembly Control )偽指令
匯編控制偽指令用于控制匯編程序的執(zhí)行流程,常用的匯編控制偽指令包括以下幾條:
— IF 、 ELSE 、 ENDIF
— WHILE 、 WEND
— MACRO 、 MEND
— MEXIT
1. IF、ELSE、ENDIF
語(yǔ)法格式:
IF 邏輯表達(dá)式
指令序列 1
ELSE
指令序列 2
ENDIF
IF 、 ELSE 、 ENDIF 偽指令能根據(jù)條件的成立與否決定是否執(zhí)行某個(gè)指令序列。當(dāng) IF 后面的邏輯表達(dá)式為真,則執(zhí)行指令序列 1 ,否則執(zhí)行指令序列 2 。其中, ELSE 及指令序列 2 可以沒有,此時(shí),當(dāng) IF 后面的邏輯表達(dá)式為真,則執(zhí)行指令序列 1 ,否則繼續(xù)執(zhí)行后面的指令。
IF 、 ELSE 、 ENDIF 偽指令可以嵌套使用。
使用示例:
GBLL Test ;聲明一個(gè)全局的邏輯變量,變量名為 Test……
IF Test = TRUE
指令序列 1
ELSE
指令序列 2
ENDIF
2. WHILE、WEND
語(yǔ)法格式:
WHILE 邏輯表達(dá)式
指令序列
WEND
WHILE 、 WEND 偽指令能根據(jù)條件的成立與否決定是否循環(huán)執(zhí)行某個(gè)指令序列。當(dāng) WHILE 后面的邏輯表達(dá)式為真,則執(zhí)行指令序列,該指令序列執(zhí)行完畢后,再判斷邏輯表達(dá)式的值,若為真則繼續(xù)執(zhí)行,一直到邏輯表達(dá)式的值為假。
WHILE 、 WEND 偽指令可以嵌套使用。
使用示例:
GBLA Counter ;聲明一個(gè)全局的數(shù)學(xué)變量,變量名為 Counter
Counter SETA 3 ;由變量Counter 控制循環(huán)次數(shù)
……
WHILE Counter < 10
指令序列
WEND
3. MACRO、MEND
語(yǔ)法格式:
MACRO
$ 標(biāo)號(hào) 宏名 $ 參數(shù) 1 , $ 參數(shù) 2 ,……
指令序列
MEND
MACRO 、 MEND 偽指令可以將一段代碼定義為一個(gè)整體,稱為宏指令,然后就可以在程序中通過宏指令多次調(diào)用該段代碼。其中, $ 標(biāo)號(hào)在宏指令被展開時(shí),標(biāo)號(hào)會(huì)被替換為用戶定義的符號(hào), 宏指令可以使用一個(gè)或多個(gè)參數(shù),當(dāng)宏指令被展開時(shí),這些參數(shù)被相應(yīng)的值替換。
宏指令的使用方式和功能與子程序有些相似,子程序可以提供模塊化的程序設(shè)計(jì)、節(jié)省存儲(chǔ)空間并提高運(yùn)行速度。但在使用子程序結(jié)構(gòu)時(shí)需要保護(hù)現(xiàn)場(chǎng),從而增加了系統(tǒng)的開銷,因此,在代碼較短且需要傳遞的參數(shù)較多時(shí),可以使用宏指令代替子程序。
包含在 MACRO 和 MEND 之間的指令序列稱為宏定義體,在宏定義體的第一行應(yīng)聲明宏的原型(包含宏名、所需的參數(shù)),然后就可以在匯編程序中通過宏名來(lái)調(diào)用該指令序列。在源程序被編譯時(shí),匯編器將宏調(diào)用展開,用宏定義中的指令序列代替程序中的宏調(diào)用,并將實(shí)際參數(shù)的值傳遞給宏定義中的形式參數(shù)。
MACRO 、 MEND 偽指令可以嵌套使用。
4. MEXIT
語(yǔ)法格式:
MEXIT
MEXIT 用于從宏定義中跳轉(zhuǎn)出去。
|