DSP編程技巧之19---詳解cmd文件
cmd文件是編譯完成之后鏈接各個目標(biāo)文件時,用來指示各個數(shù)據(jù)、符號等是如何劃分到各個段,以及每個段所使用的存儲空間的。許多筒子對cmd文件有畏難情緒,不容易理解各個段的含義,特別是在程序編譯沒有問題,但是在鏈接生成可執(zhí)行的.out遇到錯誤時更容易手足無措,所以我們就來詳細(xì)解讀一下cmd文件的具體含義。
本文引用地址:http://www.ex-cimer.com/article/256732.htmC28x的編譯器把存儲空間劃分為兩個部分進(jìn)行管理,包括:
1. 程序存儲空間:包含可執(zhí)行的代碼,初始化的記錄和switch-case使用的表。
2. 數(shù)據(jù)存儲空間:包含外部變量,靜態(tài)變量以及系統(tǒng)的棧;一般情況下,各個寄存器對應(yīng)的存儲空間也歸類在數(shù)據(jù)空間里。
為了方便管理,不同種類的代碼、變量等往往又被分別分配到不同的段(section)之中,然后對存儲空間的劃分就變成了對段的地址分配問題了。例如,在下面的代碼中,就規(guī)定了.text這個段會存放在RAM中Page0下面的RAML1中,RAML1的起始地址是0x009000,長度是0x001000。
MEMORY
{
/* 省略不在此顯示的代碼 */
PAGE 0 :
RAML1 : origin = 0x009000, length = 0x001000
RAML2 : origin = 0x00A000, length = 0x001000
/* 省略不在此顯示的代碼 */
?。?/p>
SECTIONS
{
/* 省略不在此顯示的代碼 */
.text : > RAML1, PAGE = 0
/* 省略不在此顯示的代碼 */
}
一般情況下,我們的代碼不會大到無法存儲,但是也有可能因為代碼特別多導(dǎo)致無法存儲,產(chǎn)生.text的實際大小是size xxx,但是RAML1的size只有yyy這樣的鏈接錯誤,以至于無法生成輸出文件。此時我們可以把上面對應(yīng)的RAML1的長度,即length增大,使得.text段所分配的地址空間變多。但是RAML1地址空間擴(kuò)大之后,擠占了RAML2的空間,導(dǎo)致地址重疊,此時RAML2的起始位置要后移,其長度也要相應(yīng)地縮減,才能不產(chǎn)生地址覆蓋錯誤;修改之后可以為:
RAML1 : origin = 0x009000, length = 0x001500
RAML2 : origin = 0x00A500, length = 0x000500
還有一個解決方法則是把.text給分配到其它更長的地址空間里去;如果沒有現(xiàn)成的地址范圍比較長的段,也可以合并現(xiàn)有的段,修改方法比如把RAML2刪除,把它的地址全部合并到RAML1中去,而.text還是分配在RAML1,就沒有問題了。刪除RAML2的時候要注意,它在沒有被任何段使用的情況下才能操作,否則編譯、鏈接的時候又提示其它的段找不到對應(yīng)的存儲單元了。
下面我們就解釋一下各個段的含義:
一.初始化的段
其中包含了數(shù)據(jù)和可執(zhí)行代碼,通常情況下是只讀的。它們包括:
1 .cinit和.pinit
包含了初始化變量和常量所用的表格,是只讀的。
C28x .cinit被限制在16bit范圍內(nèi),即低64K范圍。
2 .const
包含了字符串常量、字符串文字、選擇表以及使用const關(guān)鍵字定義(但是不包括volatile類型,并假設(shè)使用小內(nèi)存模型)的只讀型變量。
3 .econst
包含了字符串常量,以及使用far關(guān)鍵字定義的全局變量和靜態(tài)變量。
4 .switch
存放switch-case指令所使用的選擇表。
5 .text
通常是只讀的,包含所有可執(zhí)行的代碼,以及編譯器編譯產(chǎn)生的常量。
二.無初始化的段
無初始化的段雖然不會被初始化,但是仍然需要在存儲單元(一般是RAM)中保留相關(guān)的地址空間。它們包括:
1 .bss
為全局和靜態(tài)變量保留存儲空間。在啟動或者程序加載的時候,C/C++的啟動程序會把.cinit段中的數(shù)據(jù)(一般存放在ROM中)復(fù)制到.bss段中。
2 .ebss
為far關(guān)鍵字定義(僅適用于C代碼)的全局和靜態(tài)變量保留存儲空間。在啟動或者程序加載的時候,C/C++的啟動程序會把.cinit段中的數(shù)據(jù)(一般存放在ROM中)復(fù)制到.ebss段中。
3 .stack
默認(rèn)情況下,棧(stack)保存在.stack段中(參考boot.asm),這個段用來為棧保留存儲空間。棧(stack)的作用主要有:
1) 保留存儲空間用于存儲傳遞給函數(shù)的參數(shù);
2) 為局部變量分配相關(guān)的地址空間;
3) 保存處理器的狀態(tài);
4) 保存函數(shù)的返回地址;
5) 保存某些臨時變量的值。
需要注意的是,.stack段只能使用低64K地址的數(shù)據(jù)存儲單元,因為CPU的SP寄存器是16位的,它無法讀取超過64K的地址范圍。此外,編譯器無法檢查棧的溢出錯誤(除非我們自己編寫某些代碼來檢測),這將導(dǎo)致錯誤的輸出結(jié)果,所以要為棧分配一個相對較大的存儲空間,它的默認(rèn)值是1K字。改變棧的大小的操作可以通過編譯器選項--stack_size來完成。
c++相關(guān)文章:c++教程
評論