日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

汇编--LDR

發布時間:2023/12/15 综合教程 34 生活家
生活随笔 收集整理的這篇文章主要介紹了 汇编--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取出,即多取一次存儲器。

總結

以上是生活随笔為你收集整理的汇编--LDR的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。