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