<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > DSP編程技巧---詳解cmd文件

DSP編程技巧---詳解cmd文件

作者: 時(shí)間:2016-12-21 來(lái)源:網(wǎng)絡(luò ) 收藏
cmd文件是編譯完成之后鏈接各個(gè)目標文件時(shí),用來(lái)指示各個(gè)數據、符號等是如何劃分到各個(gè)段,以及每個(gè)段所使用的存儲空間的。許多筒子對cmd文件有畏難情緒,不容易理解各個(gè)段的含義,特別是在程序編譯沒(méi)有問(wèn)題,但是在鏈接生成可執行的.out遇到錯誤時(shí)更容易手足無(wú)措,所以我們就來(lái)詳細解讀一下cmd文件的具體含義。

C28x的編譯器把存儲空間劃分為兩個(gè)部分進(jìn)行管理,包括:

本文引用地址:http://dyxdggzs.com/article/201612/332355.htm

1. 程序存儲空間:包含可執行的代碼,初始化的記錄和switch-case使用的表。

2. 數據存儲空間:包含外部變量,靜態(tài)變量以及系統的棧;一般情況下,各個(gè)寄存器對應的存儲空間也歸類(lèi)在數據空間里。

為了方便管理,不同種類(lèi)的代碼、變量等往往又被分別分配到不同的段(section)之中,然后對存儲空間的劃分就變成了對段的地址分配問(wèn)題了。例如,在下面的代碼中,就規定了.text這個(gè)段會(huì )存放在RAM中Page0下面的RAML1中,RAML1的起始地址是0x009000,長(cháng)度是0x001000。

MEMORY

{

/* 省略不在此顯示的代碼 */

PAGE 0 :

RAML1 : origin = 0x009000, length = 0x001000

RAML2 : origin = 0x00A000, length = 0x001000

/* 省略不在此顯示的代碼 */

SECTIONS

{

/* 省略不在此顯示的代碼 */

.text : > RAML1, PAGE = 0

/* 省略不在此顯示的代碼 */

一般情況下,我們的代碼不會(huì )大到無(wú)法存儲,但是也有可能因為代碼特別多導致無(wú)法存儲,產(chǎn)生.text的實(shí)際大小是size xxx,但是RAML1的size只有yyy這樣的鏈接錯誤,以至于無(wú)法生成輸出文件。此時(shí)我們可以把上面對應的RAML1的長(cháng)度,即length增大,使得.text段所分配的地址空間變多。但是RAML1地址空間擴大之后,擠占了RAML2的空間,導致地址重疊,此時(shí)RAML2的起始位置要后移,其長(cháng)度也要相應地縮減,才能不產(chǎn)生地址覆蓋錯誤;修改之后可以為:

RAML1 : origin = 0x009000, length = 0x001500

RAML2 : origin = 0x00A500, length = 0x000500

還有一個(gè)解決方法則是把.text給分配到其它更長(cháng)的地址空間里去;如果沒(méi)有現成的地址范圍比較長(cháng)的段,也可以合并現有的段,修改方法比如把RAML2刪除,把它的地址全部合并到RAML1中去,而.text還是分配在RAML1,就沒(méi)有問(wèn)題了。刪除RAML2的時(shí)候要注意,它在沒(méi)有被任何段使用的情況下才能操作,否則編譯、鏈接的時(shí)候又提示其它的段找不到對應的存儲單元了。

下面我們就解釋一下各個(gè)段的含義:

一.初始化的段

其中包含了數據和可執行代碼,通常情況下是只讀的。它們包括:

1 .cinit和.pinit

包含了初始化變量和常量所用的表格,是只讀的。

C28x .cinit被限制在16bit范圍內,即低64K范圍。

2 .const

包含了字符串常量、字符串文字、選擇表以及使用const關(guān)鍵字定義(但是不包括volatile類(lèi)型,并假設使用小內存模型)的只讀型變量。

3 .econst

包含了字符串常量,以及使用far關(guān)鍵字定義的全局變量和靜態(tài)變量。

4 .switch

存放switch-case指令所使用的選擇表。

5 .text

通常是只讀的,包含所有可執行的代碼,以及編譯器編譯產(chǎn)生的常量。

二.無(wú)初始化的段

無(wú)初始化的段雖然不會(huì )被初始化,但是仍然需要在存儲單元(一般是RAM)中保留相關(guān)的地址空間。它們包括:

1 .bss

為全局和靜態(tài)變量保留存儲空間。在啟動(dòng)或者程序加載的時(shí)候,C/C++的啟動(dòng)程序會(huì )把.cinit段中的數據(一般存放在ROM中)復制到.bss段中。

2 .ebss

為far關(guān)鍵字定義(僅適用于C代碼)的全局和靜態(tài)變量保留存儲空間。在啟動(dòng)或者程序加載的時(shí)候,C/C++的啟動(dòng)程序會(huì )把.cinit段中的數據(一般存放在ROM中)復制到.ebss段中。

3 .stack

默認情況下,棧(stack)保存在.stack段中(參考boot.asm),這個(gè)段用來(lái)為棧保留存儲空間。棧(stack)的作用主要有:

1) 保留存儲空間用于存儲傳遞給函數的參數;

2) 為局部變量分配相關(guān)的地址空間;

3) 保存處理器的狀態(tài);

4) 保存函數的返回地址;

5) 保存某些臨時(shí)變量的值。

需要注意的是,.stack段只能使用低64K地址的數據存儲單元,因為CPU的SP寄存器是16位的,它無(wú)法讀取超過(guò)64K的地址范圍。此外,編譯器無(wú)法檢查棧的溢出錯誤(除非我們自己編寫(xiě)某些代碼來(lái)檢測),這將導致錯誤的輸出結果,所以要為棧分配一個(gè)相對較大的存儲空間,它的默認值是1K字。改變棧的大小的操作可以通過(guò)編譯器選項--stack_size來(lái)完成。

4 .sysmem

為動(dòng)態(tài)內存分配保留存儲空間,從而為malloc,calloc,realloc和 new等動(dòng)態(tài)內存分配程序服務(wù)。如果這幾個(gè)動(dòng)態(tài)內存管理函數沒(méi)有在C/C++代碼中用到的話(huà),則不需要創(chuàng )建.sysmem段。

此外,我們經(jīng)常提到“堆棧”,在這里我們只講了棧,那堆(heap)是干啥的呢?堆就是是用來(lái)做動(dòng)態(tài)內存分配的,因為在DSP上RAM資源仍然是相對寶貴的,所以堆占用的存儲空間不能無(wú)限擴展,對于near關(guān)鍵字修飾的堆,其占用的地址空間最大只能到32K字;對于far關(guān)鍵字修飾的堆,它使用的存儲空間由編譯器自動(dòng)設置,默認只有1K字。

5 .esysmem

為far malloc函數分配動(dòng)態(tài)存儲空間。如果沒(méi)有用到這個(gè)函數,則編譯器不會(huì )自動(dòng)創(chuàng )建.esysmem段。

對于匯編器,它會(huì )自動(dòng)創(chuàng )建.text, .bss和.data三個(gè)段。我們可以使用#pragma CODE_SECTION和#pragma DATA_SECTION來(lái)創(chuàng )建更多的段。

默認情況下,各個(gè)段所分配的存儲空間配置如下(可根據需要進(jìn)行更改):

最后,以一個(gè)ADC寄存器對應的內存地址分配的例子,來(lái)看看完成的cmd文件是如何完成的(事實(shí)上所有寄存器的內存地址分配在TI的外設和頭文件包中已經(jīng)幫我們做好了,這里是個(gè)演示)。

首先,在使用寄存器(或者自定義的變量)的頭文件或者源程序里,為寄存器(或者自定義的變量)指定一個(gè)自定義的段:

#ifdef __cplusplus

#pragma DATA_SECTION("AdcRegsFile")

#else

#pragma DATA_SECTION(AdcRegs,"AdcRegsFile");#endif

volatile struct ADC_REGS AdcRegs; //使得結構體被分配在指定的段中

然后,在cmd文件中,在SECTIONS下把AdcRegsFile這個(gè)段分配到ADC這塊內存區域中,并在MEMORY中定義ADC這塊內存區域的起始位置和長(cháng)度。

MEMORY

{

PAGE 0: /* Program Memory */

/* 省略不相關(guān)內容的顯示 */

PAGE 1: /* Data Memory */

/* 省略不相關(guān)內容的顯示 */

ADC : origin = 0x007100, length = 0x000020 /* ADC registers */

/* 省略不相關(guān)內容的顯示 */

}

SECTIONS

{

/* 省略不相關(guān)內容的顯示 */

AdcRegsFile : > ADC, PAGE = 1

/* 省略不相關(guān)內容的顯示 */

}

以上是一個(gè)自定義段并制定內存區域的完整例子。如果不需要這樣的自定義,則可以不去管它,使用現有的,比如某個(gè)例子中可以使用的cmd文件就可以了。



關(guān)鍵詞: DSP編程技

評論


技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>