<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > ARM指令LDR和ADR的一些區別

ARM指令LDR和ADR的一些區別

作者: 時(shí)間:2016-11-27 來(lái)源:網(wǎng)絡(luò ) 收藏
之前在閱讀 arm的匯編代碼時(shí),碰到了adr指令,查arm的指令手冊,只說(shuō)該指令是采用相對地址的,但這個(gè)相對地址應該怎么理解,卻沒(méi)有具體說(shuō)明。之后在網(wǎng)上以 adr指令為關(guān)鍵字進(jìn)行搜索,也沒(méi)有找到進(jìn)一步的知識。結果,今天在搜索android資料的時(shí)候,意外的發(fā)現了adr指令與ldr指令的不同,一下子解 決了心中的問(wèn)題。以adr指令與ldr指令對比作為關(guān)鍵字,甚至可以搜到好幾篇文章,實(shí)在是...... 竟然困擾了自己那么長(cháng)時(shí)間。
將兩篇轉來(lái),作為備忘吧。
一、adr和ldr的區別
同學(xué)們在學(xué)習ARM指令時(shí),多數都會(huì )對adr和ldr這兩個(gè)命令產(chǎn)生疑惑,那他們究竟有什么區別呢?
其實(shí)這兩個(gè)都是偽指 令:adr是小范圍的地址讀取偽指令,ldr是大范圍的讀取地址偽指令??蓪?shí)際上adr是將基于PC相對偏移的地址值或基于寄存器相對地址值讀取的為指 令,而ldr用于加載32為立即數或一個(gè)地址到指定的寄存器中。到這兒就會(huì )看到其中的區別了。如果在程序中想加載某個(gè)函數或者某個(gè)在聯(lián)接時(shí)候指定的地址時(shí) 請使用adr,例如在lds中需要重新定位的地址。當加載32為的立即數或外部地址時(shí)請用ldr。
我給大家先舉個(gè)例子:
AREA test,CODE,READONLY
ENTRY
ldr r0,_start
adr r0,_start
ldr r0,=_start
nop
_start
nop
END
這段代碼并無(wú)實(shí)際意義,只是為了方便說(shuō)明。我們反匯編一下看看:
4: ldr r0,_start
0x00000000 E59F0008 LDR R0,[PC,#0x0008]
5: adr r0,_start
0x00000004 E28F0004 ADD R0,PC,#0x00000004
6: ldr r0,=_start
0x00000008 E59F0004 LDR R0,[PC,#0x0004]
7: nop
8:
9:
10: _start
0x0000000C E1A00000 NOP
11: nop
ldr r0, _start
從內存地址 _start 的地方把值讀入。執行這個(gè)后,r0 = 0xe1a00000
adr r0, _start
取得 _start 的地址到 r0,但是請看反編譯的結果,它是與位置無(wú)關(guān)的。其實(shí)取得的時(shí)相對的位置。例如這段代碼在 0x00000000 運行,那么 adr r0, _start 得到 r0 = 0x00000010;
ldr r0, =_start
這個(gè)取得標號 _start 的絕對地址。這個(gè)絕對地址是在 link 的時(shí)候確定的??瓷先ミ@只是一個(gè)指令,但是它要占用 2 個(gè) 32bit 的空間,一條是指令,另一條是 _start 的數據(因為在編譯的時(shí)候不能確定 _start 的值,而且也不能用 mov 指令來(lái)給 r0 賦一個(gè) 32bit 的常量,所以需要多出一個(gè)空間存放 _start 的真正數據,在這里就是 0x0000000c)。
因此可以看出,這個(gè)是絕對的尋址,不管這段代碼在什么地方運行,它的結果都是 r0 = 0x0000000c。
本文來(lái)自CSDN博客,轉載請標明出處:http://blog.csdn.net/linweig/archive/2010/03/24/5411655.aspx
二、ldr和adr在使用標號表達式作為操作數的區別
http://blog.sina.com.cn/s/blog_4b5210840100c80i.html
http://blog.sina.com.cn/s/blog_4b5210840100c80i.html
ARM匯編有ldr指令以及l(fā)dr、adr偽指令,他門(mén)都可以將標號表達式作為操作數,下面通過(guò)分析一段代碼以及對應的反匯編結果來(lái)說(shuō)明它們的區別。
ldr r0, _start
adr r0, _start
ldr r0, =_start
_start:
b _start
編譯的時(shí)候設置 RO 為 0x30000000(好像有問(wèn)題),下面是反匯編的結果:
0x00000000: e59f0004 ldr r0, [pc, #4] ; 0xc
0x00000004: e28f0000 add r0, pc, #0 ; 0x0
0x00000008: e59f0000 ldr r0, [pc, #0] ; 0x10
0x0000000c: eafffffe b 0xc
0x00000010: 3000000c andcc r0, r0, ip ;注這條指令是不在上面指令中的任何一條
1.ldr r0, _start :讀取指定地址中的值
ldr在此是一條指令,把內存地址 _start 位置中的值讀入r0。(_start為指針之意,讀取指針的值)
在這里_start是一個(gè)標號(是一個(gè)相對程序的表達式),匯編程序計算相對于 PC 的偏移量,并生成相對于 PC的前索引指令:ldr r0, [pc, #4]。執行指令后,r0 = 0xeafffffe。
可以在和_start標號的相對位置不變的情況下移動(dòng)( 也就是說(shuō)整段代碼從flash中拷貝到ram中依然可以正常運行)。
2.a(chǎn)dr r0, _start :將指定地址賦到r0中
ADR是小范圍的地址讀取偽指令.ADR 指令將基于PC 相對偏移的地址值讀取到寄存器中.在匯編編譯源程序時(shí),ADR 偽指令被編譯器替換成一條合適的指令.通常,編譯器用一條
ADD 指令或SUB 指令來(lái)實(shí)現該ADR 偽指令的功能,若不能用一條指令實(shí)現,則產(chǎn)生錯誤,
編譯失敗.
r0的值為((標號_start 的地址與此指令的距離差)+(此指令的地址))。在此例中被匯編成:add r0, pc, #0。該代碼可以在和標號相對位置不變的情況下移動(dòng)(也就是說(shuō)整段代碼從flash中拷貝到ram中依然可以正常運行);
假如這段代碼在 0x30000000 運行,那么 adr r0, _start 得到 r0 = 0x3000000c;如果在地址 0 運行,就是 0x0000000c 了。
通過(guò)這一點(diǎn)可以判斷程序在什么地方運行。U-boot中那段relocate代碼就是通過(guò)adr實(shí)現判斷當前程序是在RAM中還是flash中。
3.ldr r0, =_start :將指定標號的值賦給r0
ldr在此是一條偽指令,_start(即:label-expr)是一個(gè)相對程序的或外部的表達式。匯編程序將相對程序的標號表達式 label-expr 的值放在一個(gè)文字池中,并生成一個(gè)相對程序的 LDR 指令來(lái)從文字池中裝載該值,在此例中生成的指令為:ldr r0, [pc, #0],對應文字池中的地址以及值為:0x00000010: 3000000c。如果 label-expr 是一個(gè)外部表達式,或者未包含于當前段內,則匯編程序在目標文件中放置一個(gè)鏈接程序重定位命令。鏈接程序在鏈接時(shí)生成地址。
因此取得的是標號 _start 的絕對地址,這個(gè)絕對地址(運行地址)是在連接的時(shí)候確定的。它要占用 2 個(gè) 32bit 的空間,一條是指令,另一條是文字池中存放_start 的絕對地址。因此可以看出,不管這段代碼將來(lái)在什么地方運行,它的結果都是 r0 = 0x3000000c。由于ldr r0, =_start取得的是_start的絕對地址,這句代碼可以在_start標號的絕對位置不變的情況下移動(dòng);如果使用寄存器pc在程序中可以實(shí)現絕對轉 移。(1.絕對地址;2.標號對應的值)
舉例:
GPFCON EQU 0x56000050
ldr r0,=GPFCON
GPFCON :標號
0x56000050 :標號的值
http://blog.chinaunix.net/u2/72383/showart_1071068.html
ldr的確是個(gè)復雜的指令,現總結一下:
首先要判斷我們用的是ldr arm指令還是偽指令。 當我們用的是arm指令時(shí),它的作用不是向寄存器里加載立即數,而是將某個(gè)地址里的內容加載到寄存器。而偽指令ldr的作用就是向寄存器里加載立即數。
(1) ldr偽指令
ldr偽指令的格式是 ldr Rn, =expr
其中,expr是要加載到Rn中的內容,一般可以是立即數或者label。
如果expr可以用8bit數據向右移偶數位得到,那么這條偽指令就被編譯器翻譯成mov指令。具體的移位情況可以去查閱資料。反之如果立即數很大,超 過(guò)了12bit的表示范疇,那么就不能用一條mov指令了,畢竟arm指令最大只有32bit的空間可用(RISC的arm所有的指令長(cháng)度是一致的,效率 較高,當然我們并不關(guān)心16bit的thumb指令)。如果不能用一條32bit的指令乘下來(lái),那么就只能另辟蹊徑了,新開(kāi)一段緩沖,將立即數expr放 到里面,然后將其地址(暫時(shí)標記為addr)拿來(lái)使用:
ldr Rn, addr
xxx (xxx就是expr)
xxx
由于編譯器一般來(lái)說(shuō)新安排的存儲這個(gè)立即數expr的緩沖的位置是在相應代碼的附近(這個(gè)應該可以控制,好像是使用.ltorg偽指令)。我們從addr地址加載數據到Rn不就可以了。
(2)ldr arm 指令
就是將一個(gè)地址的內容加載到寄存器。不能用mov,因為arm里的mov只是在寄存器之間傳輸數據,不支持在寄出器和memory之間傳遞數據。因此就 出現了ldr/str指令。如ldr Rn, addr,注意這里的addr的值也是有限制的。這個(gè)label應該距離當前指令的距離不超過(guò)4k。因為我們知道label在具體使用的時(shí)候應該是被翻譯 成了相對偏移,如果這個(gè)label長(cháng)度不超過(guò)12bit,那么就不應超過(guò)4k,我們可以這樣做:
ldr pc, _start_armboot
_start_armboot: .word arm_startboot
這樣label _start_armboot就在指令下方,因此肯定是合法的。


關(guān)鍵詞: ARM指令LDRAD

評論


技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>