ARM匯編指令的一些總結(jié)
比較有用的是MOV
還是通過(guò)具體匯編代碼來(lái)學(xué)習(xí)吧。
MOV沒(méi)有什么好說(shuō)的,只要掌握幾個(gè)尋址方式就可以了,而且ARM的尋址方式比386的簡(jiǎn)單很多。立即數(shù)尋址方式,立即數(shù)要求以“#”作前綴,對(duì)于十六進(jìn)制的數(shù),還要求在#后面加上0x或者&。0x大家很好理解。有一次我碰到了&ff這個(gè)數(shù),現(xiàn)在才明白跟0xff是一樣的。
STR是比較重要的指令了,跟它對(duì)應(yīng)的是LDR。ARM指令集是加載/
比如STR R0, [R1],意思是R0-> [R1],它把源寄存器寫(xiě)在前面,跟MOV、LDR都相反。
LDR應(yīng)該是非常常見(jiàn)了。LDR就是把數(shù)據(jù)從存儲(chǔ)器傳輸?shù)郊拇嫫魃?。而且有個(gè)偽指令也是LDR,因此我有個(gè)百思不得其解的問(wèn)題??催@段代碼:
movr1, #GPIO_CTL_BASE
對(duì)于當(dāng)中的ldr那句,我就不明白了,如果你把=去掉,是不能通過(guò)編譯的。我查了一些資料,個(gè)人感覺(jué)知道了原因:這個(gè)=應(yīng)該表示LDR不是ARM指令,而是偽指令。作為偽指令的時(shí)候,LDR的格式如下:
它的作用是把一個(gè)32位的地址或者常量調(diào)入寄存器。嗬嗬,那大家可能會(huì)問(wèn),
“MOV r2,#0x55aa”也可以啊。應(yīng)該是這樣的。不過(guò),LDR是偽指令啊,也就是說(shuō)編譯時(shí)編譯器會(huì)處理它的。怎么處理的呢?——規(guī)則如下:如果該數(shù)字常量在MOV指令范圍內(nèi),匯編器會(huì)把這個(gè)指令作為MOV。如果不在MOV范圍中,匯編器把該常量放在程序后面,用LDR來(lái)讀取,PC和該常量的偏移量不能超過(guò)4KB。
這么一說(shuō),雖然似懂非懂,但是能夠解釋這個(gè)語(yǔ)句了。
然后說(shuō)一下跳轉(zhuǎn)指令。ARM有兩種跳轉(zhuǎn)方式。
(1)mov pc <跳轉(zhuǎn)地址〉
(2)通過(guò)B
B是最簡(jiǎn)單的跳轉(zhuǎn)指令。要注意的是,跳轉(zhuǎn)指令的實(shí)際值不是絕對(duì)地址,而是相對(duì)地址——是相對(duì)當(dāng)前PC值的一個(gè)偏移量,它的值由匯編器計(jì)算得出。
BL非常常用。它在跳轉(zhuǎn)之前會(huì)在寄存器LR(R14)中保存PC的當(dāng)前內(nèi)容。BL的經(jīng)典用法如下:
最后提一下Thumb指令。ARM體系結(jié)構(gòu)還支持16位的Thumb指令集。Thumb指令集是ARM指令集的子集,它保留了32位代碼優(yōu)勢(shì)的同時(shí)還大大節(jié)省了存儲(chǔ)空間。由于Thumb指令集的長(zhǎng)度只有16位,所以它的指令比較多。它和ARM各有自己的應(yīng)用場(chǎng)合。對(duì)于系統(tǒng)性能有較高要求,應(yīng)使用32位存儲(chǔ)系統(tǒng)和ARM指令集;對(duì)于系統(tǒng)成本和功耗有較高要求,應(yīng)使用16位存儲(chǔ)系統(tǒng)和ARM指令集。
評(píng)論