匯編--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取出,即多取一次存儲器。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM