ldr和adr在使用标号表达式作为操作数的区别
生活随笔
收集整理的這篇文章主要介紹了
ldr和adr在使用标号表达式作为操作数的区别
小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.
ARM匯編有l(wèi)dr指令以及l(fā)dr、adr偽指令,他門都可以將標號表達式作為操作數(shù),下面通過分析一段代碼以及對應的反匯編結(jié)果來說明它們的區(qū)別。?
ldr ? ? r0, _start adr ? ? r0, _start ldr ? ? r0, =_start _start: b ?_start 編譯的時候設置 RO 為 0x30000000(好像有問題),下面是反匯編的結(jié)果:
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在此是一條指令,把內(nèi)存地址 _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 指令來實現(xiàn)該ADR 偽指令的功能,若不能用一條指令實現(xiàn),則產(chǎn)生錯誤, 編譯失敗. r0的值為((標號_start 的地址與此指令的距離差)+(此指令的地址))。在此例中被匯編成:add r0, pc, #0。該代碼可以在和標號相對位置不變的情況下移動(也就是說整段代碼從flash中拷貝到ram中依然可以正常運行); 假如這段代碼在 0x30000000 運行,那么 adr r0, _start 得到 r0 = 0x3000000c;如果在地址 0 運行,就是 0x0000000c 了。 通過這一點可以判斷程序在什么地方運行。U-boot中那段relocate代碼就是通過adr實現(xiàn)判斷當前程序是在RAM中還是flash中。
3.ldr ? ? r0, =_start ?:將指定標號的值賦給r0 ldr在此是一條偽指令,_start(即:label-expr)是一個相對程序的或外部的表達式。匯編程序?qū)⑾鄬Τ绦虻臉颂柋磉_式 label-expr 的值放在一個文字池中,并生成一個相對程序的 LDR 指令來從文字池中裝載該值,在此例中生成的指令為:ldr r0, [pc, #0],對應文字池中的地址以及值為:0x00000010: 3000000c。如果 label-expr 是一個外部表達式,或者未包含于當前段內(nèi),則匯編程序在目標文件中放置一個鏈接程序重定位命令。鏈接程序在鏈接時生成地址。 因此取得的是標號 _start 的絕對地址,這個絕對地址(運行地址)是在連接的時候確定的。它要占用 2 個 32bit 的空間,一條是指令,另一條是文字池中存放_start 的絕對地址。因此可以看出,不管這段代碼將來在什么地方運行,它的結(jié)果都是 r0 = 0x3000000c。由于ldr r0, =_start取得的是_start的絕對地址,這句代碼可以在_start標號的絕對位置不變的情況下移動;如果使用寄存器pc在程序中可以實現(xiàn)絕對轉(zhuǎn)移。(1.絕對地址;2.標號對應的值)
舉例:
GPFCON ? ? ?EQU ?0x56000050
ldr ? r0,=GPFCON
GPFCON ? ? ?:標號
0x56000050 ? ? ?:標號的值 創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎
ldr ? ? r0, _start adr ? ? r0, _start ldr ? ? r0, =_start _start: b ?_start 編譯的時候設置 RO 為 0x30000000(好像有問題),下面是反匯編的結(jié)果:
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在此是一條指令,把內(nèi)存地址 _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 指令來實現(xiàn)該ADR 偽指令的功能,若不能用一條指令實現(xiàn),則產(chǎn)生錯誤, 編譯失敗. r0的值為((標號_start 的地址與此指令的距離差)+(此指令的地址))。在此例中被匯編成:add r0, pc, #0。該代碼可以在和標號相對位置不變的情況下移動(也就是說整段代碼從flash中拷貝到ram中依然可以正常運行); 假如這段代碼在 0x30000000 運行,那么 adr r0, _start 得到 r0 = 0x3000000c;如果在地址 0 運行,就是 0x0000000c 了。 通過這一點可以判斷程序在什么地方運行。U-boot中那段relocate代碼就是通過adr實現(xiàn)判斷當前程序是在RAM中還是flash中。
3.ldr ? ? r0, =_start ?:將指定標號的值賦給r0 ldr在此是一條偽指令,_start(即:label-expr)是一個相對程序的或外部的表達式。匯編程序?qū)⑾鄬Τ绦虻臉颂柋磉_式 label-expr 的值放在一個文字池中,并生成一個相對程序的 LDR 指令來從文字池中裝載該值,在此例中生成的指令為:ldr r0, [pc, #0],對應文字池中的地址以及值為:0x00000010: 3000000c。如果 label-expr 是一個外部表達式,或者未包含于當前段內(nèi),則匯編程序在目標文件中放置一個鏈接程序重定位命令。鏈接程序在鏈接時生成地址。 因此取得的是標號 _start 的絕對地址,這個絕對地址(運行地址)是在連接的時候確定的。它要占用 2 個 32bit 的空間,一條是指令,另一條是文字池中存放_start 的絕對地址。因此可以看出,不管這段代碼將來在什么地方運行,它的結(jié)果都是 r0 = 0x3000000c。由于ldr r0, =_start取得的是_start的絕對地址,這句代碼可以在_start標號的絕對位置不變的情況下移動;如果使用寄存器pc在程序中可以實現(xiàn)絕對轉(zhuǎn)移。(1.絕對地址;2.標號對應的值)
舉例:
GPFCON ? ? ?EQU ?0x56000050
ldr ? r0,=GPFCON
GPFCON ? ? ?:標號
0x56000050 ? ? ?:標號的值 創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎
總結(jié)
以上是生活随笔為你收集整理的ldr和adr在使用标号表达式作为操作数的区别的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 农行漂亮升级妈妈信用卡额度是多少?怎么提
- 下一篇: 交通银行信用卡分期付款怎么办理