免费视频淫片aa毛片_日韩高清在线亚洲专区vr_日韩大片免费观看视频播放_亚洲欧美国产精品完整版

打開APP
userphoto
未登錄

開通VIP,暢享免費電子書等14項超值服

開通VIP
關于adr指令的理解
之前在閱讀arm的匯編代碼時,碰到了adr指令,查arm的指令手冊,只說該指令是采用相對地址的,但這個相對地址應該怎么理解,卻沒有具體說明。之后在網上以adr指令為關鍵字進行搜索,也沒有找到進一步的知識。結果,今天在搜索android資料的時候,意外的發(fā)現了adr指令與ldr指令的不同,一下子解決了心中的問題。以adr指令與ldr指令對比作為關鍵字,甚至可以搜到好幾篇文章,實在是...... 竟然困擾了自己那么長時間。
   將兩篇轉來,作為備忘吧。
一、adr和ldr的區(qū)別

同學們在學習ARM指令時,多數都會對adr和ldr這兩個命令產生疑惑,那他們究竟有什么區(qū)別呢?

其實這兩個都是偽指令:adr是小范圍的地址讀取偽指令,ldr是大范圍的讀取地址偽指令??蓪嶋H上adr是將基于PC相對偏移的地址值或基于寄存器相對地址值讀取的為指令,而ldr用于加載32為立即數或一個地址到指定的寄存器中。到這兒就會看到其中的區(qū)別了。如果在程序中想加載某個函數或者某個在聯接時候指定的地址時請使用adr,例如在lds中需要重新定位的地址。當加載32為的立即數或外部地址時請用ldr。

我給大家先舉個例子:

AREA test,CODE,READONLY
        ENTRY

ldr r0,_start
        adr r0,_start
        ldr r0,=_start
        nop

        
_start
        nop
        END

這段代碼并無實際意義,只是為了方便說明。我們反匯編一下看看:

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 的地方把值讀入。執(zhí)行這個后,r0 = 0xe1a00000

adr         r0, _start

取得 _start 的地址到 r0,但是請看反編譯的結果,它是與位置無關的。其實取得的時相對的位置。例如這段代碼在 0x00000000 運行,那么 adr r0, _start 得到 r0 = 0x00000010;

ldr          r0, =_start

這個取得標號 _start 的絕對地址。這個絕對地址是在 link 的時候確定的??瓷先ミ@只是一個指令,但是它要占用 2 個 32bit 的空間,一條是指令,另一條是 _start 的數據(因為在編譯的時候不能確定 _start 的值,而且也不能用 mov 指令來給 r0 賦一個 32bit 的常量,所以需要多出一個空間存放 _start 的真正數據,在這里就是 0x0000000c)。

因此可以看出,這個是絕對的尋址,不管這段代碼在什么地方運行,它的結果都是 r0 = 0x0000000c。



本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/linweig/archive/2010/03/24/5411655.aspx

二、ldr和adr在使用標號表達式作為操作數的區(qū)別
http://blog.sina.com.cn/s/blog_4b5210840100c80i.html
 http://blog.sina.com.cn/s/blog_4b5210840100c80i.html


ARM匯編有l(wèi)dr指令以及l(fā)dr、adr偽指令,他門都可以將標號表達式作為操作數,下面通過分析一段代碼以及對應的反匯編結果來說明它們的區(qū)別。 

        ldr     r0, _start
        adr     r0, _start
        ldr     r0, =_start
_start:
        b  _start
編譯的時候設置 RO 為 0x30000000(好像有問題),下面是反匯編的結果:

   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是一個標號(是一個相對程序的表達式),匯編程序計算相對于 PC 的偏移量,并生成相對于 PC的前索引指令:ldr r0, [pc, #4]。執(zhí)行指令后,r0 = 0xeafffffe。
   可以在和_start標號的相對位置不變的情況下移動( 也就是說整段代碼從flash中拷貝到ram中依然可以正常運行)。

2.adr     r0, _start  :將指定地址賦到r0中
   ADR是小范圍的地址讀取偽指令.ADR 指令將基于PC 相對偏移的地址值讀取到寄存器中.在匯編編譯源程序時,ADR 偽指令被編譯器替換成一條合適的指令.通常,編譯器用一條
ADD 指令或SUB 指令來實現該ADR 偽指令的功能,若不能用一條指令實現,則產生錯誤,
編譯失敗.
    r0的值為((標號_start 的地址與此指令的距離差)+(此指令的地址))。在此例中被匯編成:add r0, pc, #0。該代碼可以在和標號相對位置不變的情況下移動(也就是說整段代碼從flash中拷貝到ram中依然可以正常運行);
    假如這段代碼在 0x30000000 運行,那么 adr r0, _start 得到 r0 = 0x3000000c;如果在地址 0 運行,就是 0x0000000c 了。
    通過這一點可以判斷程序在什么地方運行。U-boot中那段relocate代碼就是通過adr實現判斷當前程序是在RAM中還是flash中。

  3.ldr     r0, =_start  :將指定標號的值賦給r0
   ldr在此是一條偽指令,_start(即:label-expr)是一個相對程序的或外部的表達式。匯編程序將相對程序的標號表達式 label-expr 的值放在一個文字池中,并生成一個相對程序的 LDR 指令來從文字池中裝載該值,在此例中生成的指令為:ldr r0, [pc, #0],對應文字池中的地址以及值為:0x00000010: 3000000c。如果 label-expr 是一個外部表達式,或者未包含于當前段內,則匯編程序在目標文件中放置一個鏈接程序重定位命令。鏈接程序在鏈接時生成地址。
    因此取得的是標號 _start 的絕對地址,這個絕對地址(運行地址)是在連接的時候確定的。它要占用 2 個 32bit 的空間,一條是指令,另一條是文字池中存放_start 的絕對地址。因此可以看出,不管這段代碼將來在什么地方運行,它的結果都是 r0 = 0x3000000c。由于ldr r0, =_start取得的是_start的絕對地址,這句代碼可以在_start標號的絕對位置不變的情況下移動;如果使用寄存器pc在程序中可以實現絕對轉移。(1.絕對地址;2.標號對應的值)

舉例:

GPFCON      EQU  0x56000050

ldr   r0,=GPFCON

GPFCON      :標號

0x56000050      :標號的值


http://blog.chinaunix.net/u2/72383/showart_1071068.html

ldr的確是個復雜的指令,現總結一下: 
    首先要判斷我們用的是ldr arm指令還是偽指令。 當我們用的是arm指令時,它的作用不是向寄存器里加載立即數,而是將某個地址里的內容加載到寄存器。而偽指令ldr的作用就是向寄存器里加載立即數。
    (1) ldr偽指令
    ldr偽指令的格式是 ldr Rn, =expr
    其中,expr是要加載到Rn中的內容,一般可以是立即數或者label。
    如果expr可以用8bit數據向右移偶數位得到,那么這條偽指令就被編譯器翻譯成mov指令。具體的移位情況可以去查閱資料。反之如果立即數很大,超過了12bit的表示范疇,那么就不能用一條mov指令了,畢竟arm指令最大只有32bit的空間可用(RISC的arm所有的指令長度是一致的,效率較高,當然我們并不關心16bit的thumb指令)。如果不能用一條32bit的指令乘下來,那么就只能另辟蹊徑了,新開一段緩沖,將立即數expr放到里面,然后將其地址(暫時標記為addr)拿來使用:
    ldr Rn, addr
    xxx (xxx就是expr)
    xxx

    由于編譯器一般來說新安排的存儲這個立即數expr的緩沖的位置是在相應代碼的附近(這個應該可以控制,好像是使用.ltorg偽指令)。我們從addr地址加載數據到Rn不就可以了。

    (2)ldr arm 指令
    就是將一個地址的內容加載到寄存器。不能用mov,因為arm里的mov只是在寄存器之間傳輸數據,不支持在寄出器和memory之間傳遞數據。因此就出現了ldr/str指令。如ldr Rn, addr,注意這里的addr的值也是有限制的。這個label應該距離當前指令的距離不超過4k。因為我們知道label在具體使用的時候應該是被翻譯成了相對偏移,如果這個label長度不超過12bit,那么就不應超過4k,我們可以這樣做:
ldr pc, _start_armboot
_start_armboot: .word arm_startboot
這樣label _start_armboot就在指令下方,因此肯定是合法的。



本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/wenlifu71022/archive/2009/05/06/4152522.aspx
本站僅提供存儲服務,所有內容均由用戶發(fā)布,如發(fā)現有害或侵權內容,請點擊舉報
打開APP,閱讀全文并永久保存 查看更多類似文章
猜你喜歡
類似文章
【z】ldr和adr在使用標號表達式作為操作數的區(qū)別
arm 匯編指令 積累
ARM匯編偽指令
【博文連載】ARM編譯器(一)ARM匯編與ARM GNU匯編
[ARM]ldr和?adr/adrl?偽指令的區(qū)別
ARM匯編語言(4) 指令、偽操作、偽指令學習
更多類似文章 >>
生活服務
分享 收藏 導長圖 關注 下載文章
綁定賬號成功
后續(xù)可登錄賬號暢享VIP特權!
如果VIP功能使用有故障,
可點擊這里聯系客服!

聯系客服