從0學(xué)ARM——什么是位置無關(guān)碼?
二、怎么實(shí)現(xiàn)位置無關(guān)碼?
1. 什么是《編譯地址》?什么是《運(yùn)行地址》?
「編譯地址:」
32位的處理器,它的每一條指令是4個字節(jié),以4個字節(jié)存儲順序,進(jìn)行順序執(zhí)行,CPU是順序執(zhí)行的,只要沒發(fā)生什么跳轉(zhuǎn),它會順序進(jìn)行執(zhí)行, 編譯器會對每一條指令分配一個編譯地址,這是編譯器分配的,在編譯過程中分配的地址,我們稱之為編譯地址。
「運(yùn)行地址:」
是指程序指令真正運(yùn)行的地址,是由用戶指定的,用戶將運(yùn)行地址燒錄到哪里,哪里就是運(yùn)行的地址。比如有一個指令的編譯地址是0x40008000,實(shí)際運(yùn)行的地址是0x40008000,如果用戶將指令燒到0x60000000上,那么這條指令的運(yùn)行地址就是0x60000000。
當(dāng)編譯地址和運(yùn)行地址不同的時候會出現(xiàn)什么結(jié)果?結(jié)果是不能跳轉(zhuǎn),編譯后會產(chǎn)生跳轉(zhuǎn)地址,如果實(shí)際地址和編譯后產(chǎn)生的地址不相等,那么就不能跳轉(zhuǎn)。
「C語言編譯地址:」
都希望把編譯地址和實(shí)際運(yùn)行地址放在一起的,但是匯編代碼因?yàn)椴恍枰鯟語言到匯編的轉(zhuǎn)換,可以直接的去寫地址,所以直接寫的就是他的運(yùn)行地址,這就是為什么任何bootloader剛開始會有一段匯編代碼,因?yàn)槠鹗即a編譯地址和實(shí)際地址不相等,這段代碼和匯編無關(guān),跳轉(zhuǎn)用的運(yùn)行地址。
2. 舉例
實(shí)現(xiàn)位置無關(guān)碼主要考慮以下兩個方面:
1. 位置無關(guān)的函數(shù)跳轉(zhuǎn)
2. 位置無關(guān)的常量訪問
下面我們通過兩個例子詳細(xì)講解。
代碼
編譯代碼使用的連接文件「map.lds」如下:
OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
OUTPUT_FORMAT("elf32-arm", "elf32-arm", "elf32-arm")
OUTPUT_ARCH(arm)
ENTRY(_start)
SECTIONS
{
. = 0x40008000;
. = ALIGN(4);
.text :
{
gcd.o(.text)
*(.text)
}
. = ALIGN(4);
.rodata :
{ *(.rodata) }
. = ALIGN(4);
.data :
{ *(.data) }
. = ALIGN(4);
.bss :
{ *(.bss) }
}
如文件map.lds所示:「0x40008000」就是鏈接地址,
其他源文件如下:「gcd.s」
.text
.global _start
_start:
ldr sp,=0x70000000 get stack top pointer
bl func
ldr pc,=func
b main
func:
mv pc,lr
「main.c」
* main.c
*
* Created on: 2020-12-12
* Author: 一口Linux
int aaaa=0;
int main(void)
{
aaaa = 0x11;
while(1);
return 0;
}
「Makefile」
TARGET=gcd
TARGETC=main
all:
arm-none-linux-gnueabi-gcc -O1 -g -c -o $(TARGETC).o $(TARGETC).c
arm-none-linux-gnueabi-gcc -O1 -g -c -o $(TARGET).o $(TARGET).s
arm-none-linux-gnueabi-gcc -O1 -g -S -o $(TARGETC).s $(TARGETC).c
arm-none-linux-gnueabi-ld $(TARGETC).o $(TARGET).o -Tmap.lds -o $(TARGET).elf
arm-none-linux-gnueabi-objcopy -O binary -S $(TARGET).elf $(TARGET).bin
arm-none-linux-gnueabi-objdump -D $(TARGET).elf > $(TARGET).dis
clean:
rm -rf *.o *.elf *.dis *.bin
反匯編文件「gcd.dis」
如上圖所示:
_start對應(yīng)的鏈接地址是0x400080009行 bl func對應(yīng)的指令10行 ldr pc,=pc對應(yīng)的指令func的鏈接地址0x40008010全局變量aaaa對應(yīng)的內(nèi)存位于bss段0x4000802c19行 aaaa = 0x11 賦值語句對應(yīng)的機(jī)器碼

發(fā)表評論
請輸入評論內(nèi)容...
請輸入評論/評論長度6~500個字
最新活動更多
-
即日-9.1立即下載>> 【限時下載】ADI中國三十周年感恩回饋助力企業(yè)升級!
-
10月23日立即報名>> Works With 開發(fā)者大會深圳站
-
11月27日立即報名>> 【工程師系列】汽車電子技術(shù)在線大會
-
精彩回顧立即查看>> 【在線研討會】解析安森美(onsemi)高精度與超低功耗CGM系統(tǒng)解決方案
-
精彩回顧立即查看>> 【在線會議】CAE優(yōu)化設(shè)計:醫(yī)療器械設(shè)計的應(yīng)用案例與方案解析
-
精彩回顧立即查看>> 《2024智能制造產(chǎn)業(yè)高端化、智能化、綠色化發(fā)展藍(lán)皮書》
推薦專題
- 1 傳魏建軍與賈躍亭合作,長城汽車出海美國
- 2 黃仁勛:與雷軍長期合作,共探AI智駕
- 3 阿里首位程序員,“掃地僧”多隆已離職
- 4 DeepSeek R2加持,中國AI與芯片產(chǎn)業(yè)迎來新一輪協(xié)同進(jìn)化
- 5 六大國產(chǎn)大模型,誰是最強(qiáng)“金融分析師”?|錦緞評測
- 6 2025年第一支10倍股,來了!
- 7 募資39.85億元!寒武紀(jì)押注大模型芯片與軟件平臺
- 8 國內(nèi)免費(fèi)版Deep Research上線,秘塔AI深度研究嘗試重塑知識工作范式
- 9 清庫存?曝英偉達(dá)H20供應(yīng)有限,且沒有復(fù)產(chǎn)計劃
- 10 清華跑出具身智能獨(dú)角獸:給機(jī)器人安上眼睛和大腦,融資近20億