用IAR EWARM開發(fā)嵌入式系統(tǒng)時目標代碼的鏈接與定位
1 數(shù)據段和代碼段的作用
本文引用地址:http://m.butianyuan.cn/article/80514.htmIAR C/C-+編譯器是一種具有世界先進水平的標準C/C-+編譯器,支持符合ANSI C標準的C或C++編程語言。源程序經編譯后生成包含數(shù)據或代碼存儲器邏輯映像的數(shù)據段和代碼段。每個段都有一個段名和一個表示其存儲器空間的段類型。段類型CODE表示ROM中的執(zhí)行代碼,段類型CONST表示ROM中的數(shù)據,段類型DATA表示RAM中的數(shù)據。段名可與段類型相同,但其意義不一樣,實際使用時不能混淆。表l所列為IAR C/C++編譯器所使用的各種段、段類型及其讀/寫屬性說明。
![](http://editerupload.eepw.com.cn/200809/aaaf23d363bd2e99bea873faf77c74ab.jpg)
1.1 數(shù)據段
數(shù)據位于DATA段中,包括靜態(tài)(statIC)存儲器、堆棧(stack)、堆(heap)以及已定位的數(shù)據。DATA段可以帶有后綴。例如,DATA_C用于常數(shù)數(shù)據,包括文字字符串;DATA_Z用于無初值或用0初值聲明的靜態(tài)和全局變量。
全局變量或已聲明的靜態(tài)變量保存在靜態(tài)存儲器空間。已聲明的靜態(tài)變量有:初值為O或非0的變量、采用“@”或“#pragma”操作符定位了的變量、被聲明為“const”因而可在ROM中保存的變量,以及采用關鍵字“__no_init”定義不允許被初始化的變量等。
堆棧用于為函數(shù)保存局部變量及其他臨時數(shù)據,是由堆棧指針寄存器SP指向的一段連續(xù)存儲器。作為堆棧的數(shù)據段稱為“CSTACK”。初始化模塊cstartup將堆棧指針初始化為指向CSTACK段的尾部。堆棧容量在很大程度上取決于具體程序操作細節(jié)。如果給定的堆棧容量太小,則會使堆棧中的數(shù)據發(fā)生覆蓋而導致程序出錯;如果給定的堆棧容量太大,則會浪費RAM空間。ARM核處理器支持5種異常工作模式,每種模式都有自己的堆棧,用戶應在啟動代碼中分別初始化各個堆棧指針,并在鏈接器命令文件中進行段定位。
堆用于保存動態(tài)分配數(shù)據。作為堆的數(shù)據段稱為“HEAP”,它儀在使用動態(tài)存儲器分配時才被包含到應用系統(tǒng)中。與CSTACK段類似,HEAP段容量的大小需要視具體應用而定。使用標準輸入/輸出庫時,應將HEAP容量設置為滿足對標準輸入/輸出進行緩沖的要求,通常為512字節(jié)。
明確指定了地址的變量將被定位在DATA_AC段或DATA_AN段。DATA_AC段用于初始化為常數(shù)的數(shù)據,DATA_AN段用于聲明為“__no_init”的變量。
1.2 代碼段
代碼段包括啟動代碼、普通代碼和異常向最。
啟動代碼位于ICODE段,包括系統(tǒng)啟動(cstartup)、運行初始化(cmain)和系統(tǒng)終止(cexit)等代碼。ICODE段必須被定位在一段連續(xù)的存儲器空間,鏈接器命令文件中不能采用-P命令選項來定位ICODE段。啟動代碼通過復位向量調用。
普通代碼位于CODE段,其中保存普通函數(shù)的執(zhí)行代碼。CODE段可以帶有后綴,如CODE_I段保存由CODE_ID段初始化并在RAM中執(zhí)行的代碼。與編譯器對已初始化變量的處理類似,代碼運行時將ROM存儲器中初始化時段的內容復制到RAM存儲器中再執(zhí)行。普通代碼段還與符號及調試信息有關。
異常向量位于INTVEC段。如果在異常向量處使用跳轉到異常句柄的指令(如B指令),則異常句柄必須位于跳轉指令能夠到達的范圍之內,使用PC加載指令(如LDR PC指令)則不存在這個問題。
2 段在存儲器中的定位
IAR C編譯器所生成的段需要通過XLINK鏈接器根據鏈接命令文件中一系列命令選項的規(guī)定在存儲器中進行定位,才能保證目標代碼的正常運行。鏈接命令文件是一種文本文件,以“.xcl”為擴展名,其中包含各種XLINK命令選項。
![](http://editerupload.eepw.com.cn/200809/a917c0c8204cfc5649ff65396e1f1c31.jpg)
3 鏈接命令文件應用舉例
鏈接器命令文件的作用足通知XLLINK鏈接器,根據文件中給定的命令選項在存儲器中進行各種段定位;由于ARM核處理器的種類繁多,各處理器的存儲器配置也不盡相同,因此鏈接器命令文件通常需要根據具體應用系統(tǒng)硬件設計進行定制,以保證代碼和數(shù)據在給定的存儲器范圍之內不發(fā)生越界而導致錯誤。
下面以Philips公司的LPC2148 ARM核處理器芯片為例,說明如何定制XLINK鏈接命令文件。LPC2148具有32 KB片內SRAM和512 KB片內Flash,其存儲器地址如下。
片內Flash:0x00000000~Ox0007FFFF。
片內SRAM:0x40000000~0x413007FFF。
針對LPC2148所具有的存儲器地址范圍,可分別定制,在片內SRAM中或在片內Flash中,調試應用程序的XLINK鏈接命令文件。
(1)在片內SRAM中調試應用程序的鏈接命令文件
![](http://editerupload.eepw.com.cn/200809/ad0a0907722e76252af7c3a570d8d8c4.jpg)
![](http://editerupload.eepw.com.cn/200809/352bf6b4c4f35fe9d49c105f137a2d59.jpg)
(2)在片內Flash中調試應用程序的鏈接命令文件
對上述鏈接命令文件進行適當修改后可以用于在LPC2148片內Flash中調試應用程序,主要是重新規(guī)定代碼段和數(shù)據段的存儲器地址,有時還要重新定義堆棧(stack)和堆(heap)的長度。下面僅列出修改后的部分,其他相同部分略去:
![](http://editerupload.eepw.com.cn/200809/e01e0e32e1a11ae03dc6a73e1f4f8372.jpg)
4 結論
采用IAR EWARM集成環(huán)境開發(fā)ARM嵌入式系統(tǒng),需要通過鏈接命令文件通知XLINK鏈接器如何對C編譯器生成的代碼和數(shù)據段進行鏈接和定位。用戶需要熟悉所使用ARM核處理器的SRAM和Flash存儲器配置,并根據實際可用地址空間來確定采用哪些XLINK命令選項。只有采用合適命令選項對代碼和數(shù)據段進行正確定位,生成可靠的執(zhí)行代碼,最后將執(zhí)行代碼寫入到Flash中,才能成功完成ARM嵌入式系統(tǒng)設計;否則,即使編寫的C源程序再優(yōu)化,也起不到其應有的作用。
c++相關文章:c++教程
存儲器相關文章:存儲器原理
評論