汇编--LDR
轉載:https://my.oschina.net/zengsai/blog/23733
ARM LDR 偽指令的格式:
LDR Rn, =expr
如果name是立即數的話
LDR R0,=0X123;//將0X123存入R0
如果name時個標識符
LDR R0,=NAME;//將NAME的地址存入R0
LDR R0, =0x3FF5000 ; 偽指令: 把 0x3FF5000 直接賦值給 R0,相當于 R0=0x3FF5000。
LDR R0, 0x3FF5000 ; 存儲訪問指令: 把以 0x3FF5000 為地址的存儲單元中的數據賦值給 R0, 相當于 R0=[0x3FF5000]。
附1 《ARM中LDR偽指令與LDR加載指令》:
ARM指令集中,LDR通常都是作加載指令的,但是它也可以作偽指令。
ARM是RISC結構,數據從內存到CPU之間的移動只能通過L/S指令來完成,也就是ldr/str指令。
比如想把數據從內存中某處讀取到寄存器中,只能使用ldr 加載指令
比如:
ldr r0, 0x12345678
就是把0x12345678這個地址中的值存放到r0中。
而mov不能干這個活,mov只能在寄存器之間移動數據,或者把立即數移動到寄存器中,這個和x86這種CISC架構的芯片區別最大的地方。
x86中沒有ldr這種指令,因為x86的mov指令可以將數據從內存中移動到寄存器中。
雖然ldr偽指令和ARM的ldr指令很像,但是作用不太一樣。ldr偽指令可以在立即數前加上=,以表示把一個地址寫到某寄存器中,比如:
ldr r0, =0x12345678
這樣,就把0x12345678這個地址寫到r0中了。所以,ldr偽指令和mov是比較相似的。只不過mov指令限制了立即數的長度為8位,也就是不能超過512。而ldr偽指令沒有這個限制。如果使用ldr偽指令時,后面跟的立即數沒有超過8位,那么在實際匯編的時候該ldr偽指令是被轉換為 mov指令的。
ldr偽指令和ldr指令不是一個同東西。
附2《ARM 偽指令之地址讀取 》:
1、ADR偽指令---小范圍的地址讀取
ADR偽指令將基于PC相對偏移的地址值或基于寄存器相對偏移的地址值讀取到寄存器中。在匯編編譯器編譯源程序時,ADR偽指令被編譯器替換成一條合適的指令。通常,編譯器用一條ADD指令或SUB指令來實現該ADR偽指令的功能,若不能用一條指令實現,則產生錯誤,編譯失敗。
ADR偽指令格式 :ADR{cond}register, expr
地址表達式expr的取值范圍:
當地址值是字節對齊時,其取指范圍為:+255~255B;
當地址值是字對齊時,其取指范圍為:-1020~1020B;
2、ADRL偽指令----中等范圍的地址讀取
ADRL偽指令將基于PC相對偏移的地址值或基于寄存器相對偏移的地址值讀取到寄存器中,比ADR偽指令可以讀取更大范圍的地址。在匯編編譯器編譯源程序時,ADRL偽指令被編譯器替換成兩條合適的指令。若不能用兩條指令實現,則產生錯誤,編譯失敗。
ADRL偽指令格式:ADRL{cond}register, expr
地址表達式expr的取值范圍:
當地址值是字節對齊時,其取指范圍為:-64K~64K;
當地址值是字對齊時,其取指范圍為:-256K~256K;
3、LDR偽指令-----大范圍的地址讀取
LDR偽指令用于加載32位的立即數或一個地址值到指定寄存器。在匯編編譯源程序時,LDR偽指令被編譯器替換成一條合適的指令。若加載的常數未超出MOV或MVN的范圍,則使用MOV或MVN指令代替該LDR偽指令,否則匯編器將常量放入文字池,并使用一條程序相對偏移的LDR指令從文字池讀出常量。
轉載:https://www.cnblogs.com/hnrainll/archive/2011/06/14/2080241.html
ARM指令集中,LDR通常都是作加載指令的,但是它也可以作偽指令。
LDR偽指令的形式是“LDR Rn,=expr”。下面舉一個例子來說明它的用法。
COUNT EQU 0x40003100
……
LDR R1,=COUNT
MOV R0,#0
STR R0,[R1]
COUNT是我們定義的一個變量,地址為0x40003100。這中定義方法在匯編語言中是很常見的,如果使用過單片機的話,應該都熟悉這種用法。
LDR R1,=COUNT是將COUNT這個變量的地址,也就是0x40003100放到R1中。
MOV R0,#0是將立即數0放到R0中。最后一句STR R0,[R1]是一個典型的存儲指令,將R0中的值放到以R1中的值為地址的存儲單元去。實際就是將0放到地址為0x40003100的存儲單元中去。可 見這三條指令是為了完成對變量COUNT賦值。用三條指令來完成對一個變量的賦值,看起來有點不太舒服。這可能跟ARM的采用RISC有關。
下面還有一個例子
;將COUNT的值賦給R0
LDR R1,=COUNT
LDR R0,[R1]
LDR R1,=COUNT這條偽指令,是怎樣完成將COUNT的地址賦給R1,有興趣的可以看它編譯后的結果。這條指令實際上會編譯成一條LDR指令和一條DCD偽指令。
LDR 的兩種用法
1)LDR pc, =MyHandleIRQ 表示將MyHandleIRQ符號放入pc寄存器中
2)LDR PC,MyHandleIRQ 表示將讀取存儲器中MyHandleIRQ符號所表示的地址中的值,及需要多讀一次存儲器。
在代碼中:
start:
ldr pc,=MyHandleReset @jump to HandleReset
ldr pc,=MyHandleUndef @jump to HandleUndef
ldr pc,=MyHandleSWI @jump to HandleSWI
ldr pc,=MyHandleIabort @jump to HandleIabort
ldr pc,=MyHandleDabort @jump to HandleDabort
nop
ldr pc,=MyHandleIRQ @jump to HandleIRQ<=之前出錯的一行
ldr pc,=MyHandleFIQ @jump to HandleFIQ
@MyHandleIRQ: .word OS_CPU_IRQ_ISR
MyHandleIRQ:
sub lr, lr, #4 @ to calculate the return address
stmdb sp!, {r0-r12,lr}
ldr lr, =int_return @ restore the return address
ldr pc, =int_handle @ call for the interrupt handler
在“之前出錯的一行”處,如果改成“ldr pc,MyHandleIRQ”當中斷來臨時,無法進行中斷處理。
另一種情況是正確的,注意體會:
start:
ldr pc,=MyHandleReset @jump to HandleReset
ldr pc,=MyHandleUndef @jump to HandleUndef
ldr pc,=MyHandleSWI @jump to HandleSWI
ldr pc,=MyHandleIabort @jump to HandleIabort
ldr pc,=MyHandleDabort @jump to HandleDabort
nop
ldr pc,MyHandleIRQ @jump to HandleIRQ<=之前出錯的一行
ldr pc,=MyHandleFIQ @jump to HandleFIQ
MyHandleIRQ: .word OS_CPU_IRQ_ISR
@MyHandleIRQ:
@ sub lr, lr, #4 @ to calculate the return address
@ stmdb sp!, {r0-r12,lr}
@ ldr lr, =int_return @ restore the return address
@ ldr pc, =int_handle @ call for the interrupt handler
因為當中斷來臨時,還需要去MyHandleIRQ處把OS_CPU_IRQ_ISR取出,即多取一次存儲器。
總結
- 上一篇: 联想 2023 款小新 Air 14 笔
- 下一篇: 有些00后存款是90后2倍,90后怎么看