ARM中的---彙編指令

ARM中的---彙編指令

   

1. ".text"、".data"、".bss"

依次表示的是
“以下是代碼段”,
“以下是初始化數據段”,
“以下是未初始化數據段”。

2.".global"

定義一個全局符號,通常是爲ld使用。比如經常看到的

.global _start

3.".ascii"、".byte"、".short"、".int"、".long"、".word"、".quad"

定義一個字符串,併爲它分配空間
定義一個字節,併爲它分配空間,佔單字節,0x34
定義一個短整型,併爲它分配空間,佔雙字節,0x1234
定義一個整型,併爲它分配空間,佔四字節,0x12345678
定義一個長整型,併爲它分配空間,佔四字節,0x12345678
定義一個,併爲它分配空間,
定義一個,併爲它分配定義,佔八字節,...
比如
.long 0x22011110//BWSCON
.long 0x00000700//BANKCON0
...

4.".abort"

停止彙編

5.".align"

.align absexpr1,absexpr2

以某種對齊方式,在未使用的存儲區域填充值. 第一個值表示對齊方式,4, 8,16或32. 第二個表達式值表示填充的值

6.".if .else .endif"

.if

.else

.endif:支持條件預編譯

7.".include"

.include "file":包含指定的頭文件, 可以把一個彙編常量定義放在頭文件中

8.".comm"

.comm  symbol, length:

在bss段申請一段命名空間,該段空間的名稱叫symbol, 長度爲length. Ld連接器在連接會爲它留出空間

9.".equ"

.equ symbol, expression: 把某一個符號(symbol)定義成某一個值(expression).該指令並不分配空間,相當於C語言中的#define。例如

.equ aaa,0x20000000

10.".macro .endm"

.macro: 定義一段宏代碼,.macro表示代碼的開始,.endm表示代碼的結束,.exitm跳出宏, 示例如下:

.macro SHIFTLEFT a, b.if \b < 0

mov \a, \a, ASR #-\b

.exitm

.endif

mov \a, \a, LSL #\b

.endm

11.".req"

name .req register name: 爲寄存器定義一個別名

12.".code"

.code [16|32]: 指定指令代碼產生的長度, 16表示Thumb指令, 32表示ARM指令

13.".ltorg"

.ltorg: 表示當前往下的定義在歸於當前段,併爲之分配空間


二.帶下滑線的

1._start

彙編程序的缺省入口,但是可以更改,想要更改其他標誌,到相應的鏈接腳本中去用ENTRY指明其他入口標誌。標號可以直接認爲是地址。


三.不帶點的

1.mov--數據傳送指令

它的傳送指令只能是把一個寄存器的值(要能用立即數表示)賦給另一個寄存器,或者將一個常量賦給寄存器,將後邊的量賦給前邊的量,比如

mov r1,r2

mov r1,#4096

這個立即數是小於0xff(65535)的數,如果大於65535,則用ldr指令賦值

2.b、bl--相對跳轉指令

b只是跳轉,而bl除跳轉外還將返回地址(bl的下一條指令的地址)保存到lr寄存器中。其中跳轉範圍是當前指令的前後32M。

3.ldr、str--內存訪問指令

ldr指令即可能是大範圍的地址讀取僞指令,也可能是內存訪問指令,當它的第二個參數之前有“=”時,表示僞指令,否則是內存訪問指令,比如

ldr r0, =0x53000000//r0=0x53000000

str r1, [r2, #4]         // 將r1的數據保存到地址爲r2+4的內存單元中

str r1, [r2]                // 將r1的數據保存到地址爲r2的內存單元中

str r1, [r2], #4         // 將r1的數據保存到地址爲r2的內存單元中,然後r2=r2+4

ldr r1, [r2,#4]         // 將地址爲r2+4的內存單元數據讀取到r1

ldr r1, [r2]             // 將地址爲r2的內存單元數據讀取到r1中

ldr r1, [r2], #4         // 將地址爲r2的內存單元數據讀取到r1中,然後r2=r2+4

作爲變址方式有如下分類:


當ldr作爲大範圍地址讀取僞指令,LDR僞指令用於加載32們的立即數或一個地址值到指定寄存器。在彙編編譯源程序時,LDR僞指令被編譯器替換成一條合適的指令。若加載的常數未超出MOV或者MVN的範圍,剛使用MOV或MVN指令代替該LDR僞指令,否則彙編器將常量放入字池,並使用一 條程序相對偏移的LDR指令從文字池讀出常量。

作爲寄存器的尋址,寄存器的間接尋址的3種方式


4.adr--小範圍的地址讀取(只有兩個參數)

ADR指令將基於PC相對偏移的地址值讀取到寄存器中,在編譯源程序時ADR僞指令被編譯器替換成一條合適的指令。通常,編譯器用一條ADD指令或SUB指令來實現該ADR僞指令的功能,若不能用一條指令實現,剛產生錯誤,編譯失敗。比如

adr r0, delay//將標號delay的地址賦給r0

5.ldmia--多寄存器尋址

一條指令可以完成多個寄存器值的傳遞,可以完成最多16個通用寄存器值的傳遞。比如

ldmia r0, [r1, r2, r3, r4]

;r1 <- [r0]

;r2 <- [r0+4]

;r3 <- [r0+4*2]

;r4 <- [r0+4*3]

6.nop--空操作指令

什麼都不做,可以用作延時。

7.add,sub--加法減法指令

共三個參數,將後邊的兩個參數操作並賦給第一個參數,比如

add r1, r1, r0//r1=r1+r0

sub r1, r1, r2//r1=r1-r2

//這兩句的結果就是r1=r1+r0-r2

8.asr,lsl,lsr,ror,rrx,type rs

對於arm指令的基本格式如下:


對於operand2參數,如果能靈活使用,代碼效率會很高。


桶型一位寄存器操作



9.cmp--比較指令

cspr

10.eq,ne,ce等--指令的條件域

所有的ARM指令都可以條件執行:指令的執行與否取決於CPSR寄存器的N,Z,CandV
每一條ARM指令包含4位的條件碼位於指令的最高4位[31:28],條件碼共16種,每個條件碼可以用2個字符表示,這兩個字符可以添加在指令助記符的後邊和指令同時使用。
指令最高4位代表的條件域如下表

條件碼

助記符後綴

標誌

含義

0000

EQ

Z置位

相等

0001

NE

Z清零

不相等

0010

CS

C置位

無符號數大於或等於

0011

CC

C清零

無符號數小於

0100

MI

N置位

負數

0101

PL

N清零

正數或零

0110

VS

V置位

溢出

0111

VC

V清零

未溢出

1000

HI

C置位Z清零

無符號數大於

1001

LS

C清零Z置位

無符號數小於或等於

1010

GE

N等於V

帶符號數大於或等於

1011

LT

N不等於V

帶符號數小於

1100

GT

Z清零且(N等於V)

帶符號數大於

1101

LE

Z置位或(N不等於V)

帶符號數小於或等於

1110

AL

忽略

無條件執行

例如

cmp r0, r1

beq func//如果r1==r0,則跳轉到func標誌地址上去

cmp r0, r1

beq func//如果r1!=r0,則跳轉到func標誌地址上去

11.R13,R14寄存器--sp,lr

sp是堆棧指針,lr是鏈接地址寄存器。R13作爲sp,R14作爲lr。
對於鏈接地址寄存器,它的作用有兩個:
【1】當使用bl指令調用子程序時,系統會自動將 bl指令的下一條指令的地址存入lr中。
執行過程如下圖

①,程序A正常執行到BL Lable然後轉到程序B中去
②,系統將BL Lable的下一句NEXT的地址放到LR中去
③,執行完B的程序後,執行一個MOV PC,LR將NEXT的地址給到當前pc值
④,繼續執行程序A下面的語句

【2】當發生異常時,系統自動將異常的返回地址放入R14中(有些異常有一個小的固定的偏移量)。

12.邏輯運算指令--

;按位與

AND Rd, Rn                                 ; Rd &= Rn

AND.W Rd, Rn, #imm12           ; Rd = Rn & imm12

AND.W Rd, Rm, Rn                    ; Rd = Rm & Rn

 

;按位或

ORR Rd, Rn                                  ; Rd |= Rn

ORR.W Rd, Rn, #imm12            ; Rd = Rn | imm12

ORR.W Rd, Rm, Rn                     ; Rd = Rm | Rn

 

;按位清零

BIC Rd, Rn                                    ; Rd &= ~Rn

BIC.W Rd, Rn, #imm12             ; Rd = Rn & ~imm12

BIC.W Rd, Rm, Rn                      ; Rd = Rm & ~Rn

 

;按位或反

ORN.W Rd, Rn, #imm12          ; Rd = Rn | ~imm12

ORN.W Rd, Rm, Rn                   ; Rd = Rm | ~Rn

 

;按位異或

EOR Rd, Rn                                 ; Rd ^= Rn

EOR.W Rd, Rn, #imm12           ; Rd = Rn ^ imm12

EOR.W Rd, Rm, Rn                    ; Rd = Rm ^ Rn

 

;邏輯左移

LSL Rd, Rn, #imm5                  ; Rd = Rn<<imm5

LSL Rd, Rn                                 ; Rd <<= Rn

LSL.W Rd, Rm, Rn                   ; Rd = Rm<<Rn

 

;邏輯右移

LSR Rd, Rn, #imm5                ; Rd = Rn>>imm5

LSR Rd, Rn                               ; Rd >>= Rn

LSR.W Rd, Rm, Rn                  ; Rd = Rm>>Rn

 

;算術右移

ASR Rd, Rn, #imm5                 ; Rd = Rn>> imm5

ASR Rd, Rn                                ; Rd =>> Rn

ASR.W Rd, Rm, Rn                   ; Rd = Rm>>Rn

 

;循環右移

ROR Rd, Rn ;

ROR.W Rd, Rm, Rn ;


四.彙編中的註釋

(1) @ 表示註釋從當前位置到行尾的字符.

(2) # 註釋掉一整行.

(3) ; 新行分隔符.

五.彙編中的常數

(1)十進制數以非0數字開頭,如:123和9876;

(2)二進制數以0b開頭,其中字母也可以爲大寫; 

(3)八進制數以0開始,如:0456,0123;

(4)十六進制數以0x開頭,如:0xabcd,0X123f;

(5)字符串常量需要用引號括起來,中間也可以使用轉義字符,如: “You are welcome!/n”;

(6)當前地址以“.”表示,在彙編程序中可以使用這個符號代表當前指令的地址

(7)表達式:在彙編程序中的表達式可以使用常數或者數值, “-”表示取負數, “~”表示取補,“<>”表示不相等,其他的符號如:+、-、*、/、%、<、<<、>、>>、|、&、^、!、==、>=、<=、&&、||跟C語言中的用法相似。

六.彙編程序較好的格式




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