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
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--比較指令
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中去【2】當發生異常時,系統自動將異常的返回地址放入R14中(有些異常有一個小的固定的偏移量)。
②,系統將BL Lable的下一句NEXT的地址放到LR中去
③,執行完B的程序後,執行一個MOV PC,LR將NEXT的地址給到當前pc值
④,繼續執行程序A下面的語句
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)十進制數以非0數字開頭,如:123和9876;
(2)二進制數以0b開頭,其中字母也可以爲大寫;
(3)八進制數以0開始,如:0456,0123;
(4)十六進制數以0x開頭,如:0xabcd,0X123f;
(5)字符串常量需要用引號括起來,中間也可以使用轉義字符,如: “You are welcome!/n”;
(6)當前地址以“.”表示,在彙編程序中可以使用這個符號代表當前指令的地址;
(7)表達式:在彙編程序中的表達式可以使用常數或者數值, “-”表示取負數, “~”表示取補,“<>”表示不相等,其他的符號如:+、-、*、/、%、<、<<、>、>>、|、&、^、!、==、>=、<=、&&、||跟C語言中的用法相似。