section 結(jié)構(gòu)
SECTIONS {
...
secname start BLOCK(align) (NOLOAD) : AT ( ldadr )
??{ contents } >region :phdr =fill
...
}
secname:段名
contents:決定哪些內(nèi)容存放在此段
start:本段的連接地址(實際運行地址)
AT(ldadr):存儲地址(加載的地址)
//例子U-Boot.lds代碼(根據(jù)上面的section的介紹,雖能大體看懂,但是還是有些許疑惑)
SECTIONS
{
????. = 0x00000000;??????// ?????此處對應(yīng)section結(jié)構(gòu)中哪個標(biāo)識,我覺得應(yīng)該是存儲地址吧?? 但卻沒有 AT 標(biāo)識????
????. = ALIGN(4);????????//此處應(yīng)該是4字節(jié)對齊的意思,???? 但對應(yīng)section結(jié)構(gòu)中的哪個標(biāo)志不是很明白
????.text??????:????????????//此處應(yīng)該是secname 段名
?? {
???????? cpu/arm920t/start.o
????????(.text)??????????//大括號,應(yīng)該為contents段,指示該段存放的內(nèi)容
????????*(.text)
?? }
?? . = ALIGN(4);????????????????????//以下類似
?? .rodata : { *(.rodata) }
?? . = ALIGN(4);
?? .data : { *(.data) }
?? . = ALIGN(4);
?? .got : { *(.got) }
?? . = .;
?? __u_boot_cmd_start = .;
?? .u_boot_cmd : { *(.u_boot_cmd) }
?? __u_boot_cmd_end = .;
?? . = ALIGN(4);
?? __bss_start = .;
?? .bss : { *(.bss) }
?? _end = .;
}
問題1,二進(jìn)制文件不包含BSS段,那把BSS段放在哪?
答:修改有1000個全局變量,難道要BIN里要存1000個0嗎?在鏈接腳本里把BSS段組織在一起,記下它的起始地址、結(jié)束地址,重定位后把這塊內(nèi)存清0即可
問題2:全局變量不初始化的話默認(rèn)初始化為零,干嘛還要手動清零?
答:因為它是在BSS段的
bss段:
BSS段(bsssegment)通常是指用來存放程序中未初始化的全局變量的一塊內(nèi)存區(qū)域。BSS是英文BlockStarted by Symbol的簡稱。BSS段屬于靜態(tài)內(nèi)存分配。
data段:
數(shù)據(jù)段(datasegment)通常是指用來存放程序中已初始化的全局變量的一塊內(nèi)存區(qū)域。數(shù)據(jù)段屬于靜態(tài)內(nèi)存分配。
text段:
代碼段(codesegment/textsegment)通常是指用來存放程序執(zhí)行代碼的一塊內(nèi)存區(qū)域。這部分區(qū)域的大小在程序運行前就已經(jīng)確定,并且內(nèi)存區(qū)域通常屬于只讀,某些架構(gòu)也允許代碼段為可寫,即允許修改程序。在代碼段中,也有可能包含一些只讀的常數(shù)變量,例如字符串常量等。
rodata段:
存放C中的字符串和#define定義的常量
heap堆:
堆是用于存放進(jìn)程運行中被動態(tài)分配的內(nèi)存段,它的大小并不固定,可動態(tài)擴(kuò)張或縮減。當(dāng)進(jìn)程調(diào)用malloc等函數(shù)分配內(nèi)存時,新分配的內(nèi)存就被動態(tài)添加到堆上(堆被擴(kuò)張);當(dāng)利用free等函數(shù)釋放內(nèi)存時,被釋放的內(nèi)存從堆中被剔除(堆被縮減)
stack棧:
是用戶存放程序臨時創(chuàng)建的局部變量,也就是說我們函數(shù)括弧“{}”中定義的變量(但不包括static聲明的變量,static意味著在數(shù)據(jù)段中存放變量)。除此以外,在函數(shù)被調(diào)用時,其參數(shù)也會被壓入發(fā)起調(diào)用的進(jìn)程棧中,并且待到調(diào)用結(jié)束后,函數(shù)的返回值也會被存放回棧中。由于棧的先進(jìn)先出特點,所以棧特別方便用來保存/恢復(fù)調(diào)用現(xiàn)場。從這個意義上講,我們可以把堆棧看成一個寄存、交換臨時數(shù)據(jù)的內(nèi)存區(qū)。
常量段:
常量段一般包含編譯器產(chǎn)生的數(shù)據(jù)(與只讀段包含用戶定義的只讀數(shù)據(jù)不同)。比如說由一個語句a=2+3編譯器把2+3編譯期就算出5,存成常量5在常量段中
一般情況下,一個程序本質(zhì)上都是由 bss段、data段、text段三個組成的——本概念是當(dāng)前的計算機(jī)程序設(shè)計中是很重要的一個基本概念。而且在嵌入式系統(tǒng)的設(shè)計中也非常重要,牽涉到嵌入式系統(tǒng)運行時的內(nèi)存大小分配,存儲單元占用空間大小的問題。
在采用段式內(nèi)存管理的架構(gòu)中(比如intel的80x86系統(tǒng)),bss段(Block Started by Symbol segment)通常是指用來存放程序中未初始化的全局變量的一塊內(nèi)存區(qū)域,一般在初始化時bss 段部分將會清零(bss段屬于靜態(tài)內(nèi)存分配,即程序一開始就將其清零了)。
比如,在C語言程序編譯完成之后,已初始化的全局變量保存在.data 段中,未初始化的全局變量保存在.bss 段中。???
l??????????text和data段都在可執(zhí)行文件中(在嵌入式系統(tǒng)里一般是固化在鏡像文件中),由系統(tǒng)從可執(zhí)行文件中加載;
l????????? 而bss段不在可執(zhí)行文件中,由系統(tǒng)初始化。
編譯兩個小程序如下:
程序1:
int ar[30000];
void main()
{
......
}
程序2:
int ar[300000] =? {1, 2, 3, 4, 5, 6 };
void main()
{
......
}
發(fā)現(xiàn)程序2編譯之后所得的.exe文件比程序1的要大得多。 為什么?
區(qū)別很明顯,一個位于.bss段,而另一個位于.data段,兩者的區(qū)別在于:
l????????? 全局的未初始化變量存在于.bss段中,具體體現(xiàn)為一個占位符;全局的已初始化變量存于.data段中;
l????????? 而函數(shù)內(nèi)的自動變量都在棧上分配空間。
l????????? .bss是不占用.exe文件空間的,其內(nèi)容由操作系統(tǒng)初始化(清零);
l??????????而.data卻需要占用,其內(nèi)容由程序初始化,因此造成了上述情況。
注意:
l????????? bss段(未手動初始化的數(shù)據(jù))并不給該段的數(shù)據(jù)分配空間,只是記錄數(shù)據(jù)所需空間的大小。
l????????? data(已手動初始化的數(shù)據(jù))段則為數(shù)據(jù)分配空間,數(shù)據(jù)保存在目標(biāo)文件中。
l????????? DATA段包含經(jīng)過初始化的全局變量以及它們的值。
l????????? BSS段的大小從可執(zhí)行文件中得到,然后鏈接器得到這個大小的內(nèi)存塊,緊跟在數(shù)據(jù)段后面。當(dāng)這個內(nèi)存區(qū)進(jìn)入程序的地址空間后全部清零。包含DATA和BSS段的整個區(qū)段此時通常稱為數(shù)據(jù)區(qū)。
?
評論
查看更多