0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于ARM編譯器版本5的工程遷移與適配-堆棧初始化以及總線異常問題

嵌入式那些事 ? 來源:嵌入式那些事 ? 2023-11-16 16:28 ? 次閱讀

1、Heap region was used, but no heap region was defined

工程中,我使用的是自己的分散加載文件,并且沒有定義ARM_LIB_STACKHEAP,ARM_LIB_STACK,ARM_LIB_HEAP這些符號,因為我自己要重新定義堆棧,就沒有使用這些符號,因此在C代碼中加入下述代碼:

//不使用ARM提供的堆函數(shù)
__asm(".global__use_no_heap");

但是編譯的時候還是報錯:Error: L6915E: Library reports error: Heap region was used, but no heap region was defined。最初我以為是分散加載文件的問題,后來查看分析,覺得分散加載沒問題,于是就換個方向思考。

因為我在C代碼中添加了不使用ARM提供的堆函數(shù)的聲明,然而ARM_LIB_STACKHEAP,ARM_LIB_STACK,ARM_LIB_HEAP這些符號會在ARM官方代碼的堆棧初始化函數(shù)中進(jìn)行使用,那么我只要分析出是誰在調(diào)用ARM官方的堆棧初始化函數(shù),就能夠解決這個問題了。

后來發(fā)現(xiàn)我的啟動代碼中少寫了一個函數(shù)__rt_entry,這個函數(shù)的作用就是一些初始化工作,當(dāng)然也就包括初始化堆棧了。這個函數(shù)在ARM官方庫中已經(jīng)實現(xiàn),由于我沒有自定義__rt_entry函數(shù),因此在啟動時會調(diào)用ARM官方的__rt_entry函數(shù),也就自然會調(diào)用ARM庫中的堆棧初始化函數(shù),在鏈接的過程中,當(dāng)發(fā)現(xiàn)分散加載文件沒有定義ARM_LIB_STACKHEAP,ARM_LIB_STACK,ARM_LIB_HEAP這些符號就報錯:Error: L6915E: Library reports error: Heap region was used, but no heap region was defined。

解決方法:自然是在我的啟動文件中自定義一個__rt_entry函數(shù),該函數(shù)會調(diào)用main()函數(shù)。

2、啟動過程中出現(xiàn)總線異常

在解決了上面第1個問題后,工程能夠成功的編譯和鏈接,但是程序運行時會出現(xiàn)總線錯誤,而且有時候代碼量減小又不會出現(xiàn)這個問題。最開始分析這個問題,花費了不少時間,也沒有摸著門道。后來也是在我組長的指點下才解決的。

首先分析下為什么這個總線異常問題有時會出現(xiàn),有時不會出現(xiàn)。這就得說說分散加載文件中加載域和執(zhí)行域中的一個標(biāo)志:NOCOMPRESS,AC6.12的armlink文檔中有對這個標(biāo)志的描述:

RW data compression is enabled by default. The NOCOMPRESS keyword enables you to specify that RW data in an execution region must not be compressed in the final image.

大致意思,默認(rèn)情況下RW數(shù)據(jù)會被壓縮放置在image中,只有在分散加載文件中相關(guān)的加載域與執(zhí)行域中使用標(biāo)志NOCOMPRESS時,才不會對RW數(shù)據(jù)進(jìn)行壓縮。

因為我沒有使用NOCOMPRESS標(biāo)志,因此會對RW數(shù)據(jù)進(jìn)行壓縮,因此在image中的RW數(shù)據(jù)會比實際的長度小一些。由于是我自己在啟動代碼中初始化data和bss,因此在將flash中的RW data拷貝到RAM中時,RW data的長度我仍然使用的是未壓縮的數(shù)據(jù)長度,該長度大于壓縮數(shù)據(jù)的長度,在將flash中的RW data拷貝到RAM中時,當(dāng)長度超過壓縮數(shù)據(jù)的長度,flash就會產(chǎn)生保護(hù),從而觸發(fā)一個總線異常,而且異常的地址存儲在BFAR中。

解決方法:這里有兩種解決方法。

方法1:最簡單的就是在分散加載文件中RW data相關(guān)的加載域和執(zhí)行域使用NOCOMPRESS標(biāo)志,當(dāng)然了這個方法的缺點就是產(chǎn)生的image文件會大一些。

方法2:可以在啟動代碼中配置好sp,然后就不用去對RW data和bss進(jìn)行處理(屏蔽掉自己寫的RW data和bss的處理代碼),這些工作ARM官方庫會自行處理的(壓縮的RW data在拷貝到RAM中時,ARM官方庫會進(jìn)行解壓縮的,這些ARM庫代碼在編譯和鏈接的時候會自動的加入到image中)。







審核編輯:劉清

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • ARM
    ARM
    +關(guān)注

    關(guān)注

    134

    文章

    8967

    瀏覽量

    365065
  • RAM
    RAM
    +關(guān)注

    關(guān)注

    8

    文章

    1344

    瀏覽量

    114218

原文標(biāo)題:基于ARM編譯器版本5的工程遷移與適配到ARM編譯器版本6.12 后續(xù)2 - 堆棧初始化以及總線異常問題

文章出處:【微信號:嵌入式那些事,微信公眾號:嵌入式那些事】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    字符型、指針型等變量該如何初始化

    在敲代碼的時候,我們會給變量一個初始值,以防止因為編譯器的原因造成變量初始值的不確定性。對于數(shù)值類型的變量往往初始化為0,但對于其他類型的變量,如字符型、指針型等變量等該如何
    發(fā)表于 09-23 11:50 ?2118次閱讀

    Keil修改ARM編譯器及配置方法

    Keil MDK自 V5.36 版本之后,默認(rèn)就不帶 Arm Compiler V5版本編譯器。如果需要使用 V
    發(fā)表于 09-19 10:41 ?3658次閱讀
    Keil修改<b class='flag-5'>ARM</b><b class='flag-5'>編譯器</b>及配置方法

    ARM堆棧學(xué)習(xí)筆記

    R13,在用戶應(yīng)用程序的初始化部分,一般都要初始化每種模式下的R13,使其指向該運行模式的??臻g,這樣,當(dāng)程序的運行進(jìn)入異常模式時,可以將需要保護(hù)的寄存放入R13所指向的
    發(fā)表于 06-15 11:50

    學(xué)習(xí)ARM過程中的堆棧初始化詳解

    指令強制性的要 求使用R13作為堆棧指針。由于處理的每種運行模式均有自己獨立的物理寄存R13,在用戶應(yīng)用程序的初始化部分,一般都要初始化
    發(fā)表于 04-20 14:11

    S32K146 ECC初始化,為什么不初始化堆棧空間?

    S32K146 ECC初始化,為什么不初始化堆棧空間?
    發(fā)表于 04-20 12:55

    ARM編譯器for Embedded Version 6.20移植和兼容性指南

    Arm?編譯器嵌入式遷移和兼容性指南為從舊版本Arm編譯器
    發(fā)表于 08-10 07:17

    Arm編譯器遷移和兼容性指南

    Arm?編譯器遷移和兼容性指南為從舊版本Arm編譯器遷移
    發(fā)表于 08-10 06:57

    用于嵌入式FUSA的ARM編譯器移植和兼容性指南

    《用于Embedded FUSA的ARM?編譯器遷移與兼容性指南》為從舊版本ARM編譯器
    發(fā)表于 08-29 07:02

    ICC AVR編譯器的安裝與使用

    ICCAVR編譯器的安裝、運行、破解、使用 用ICCAVR編譯器產(chǎn)生初始化程序和程序框架
    發(fā)表于 07-09 18:06 ?258次下載

    avr初始化代碼生成器中文版免費下載

    avr初始化代碼生成器,中文界面,適用于ICCAVR和WINAVR(GCC)編譯器,是工程師的實用軟件。
    發(fā)表于 12-12 16:35 ?0次下載
    avr<b class='flag-5'>初始化</b>代碼生成器中文版免費下載

    關(guān)于KEIL ARM編譯器的使用介紹

    KEIL ARM編譯器的使用
    的頭像 發(fā)表于 07-10 10:50 ?6198次閱讀

    EE-88:使用21xx編譯器初始化C語言中的變量

    EE-88:使用21xx編譯器初始化C語言中的變量
    發(fā)表于 05-19 21:08 ?1次下載
    EE-88:使用21xx<b class='flag-5'>編譯器</b><b class='flag-5'>初始化</b>C語言中的變量

    什么是指定初始化?

    按照這種固定的順序,我們可以依次給 a[0] 和 a[8] 賦值。因為沒有對 a[9] 賦值,所以編譯器會將 a[9] 默認(rèn)設(shè)置為0。當(dāng)數(shù)組長度比較小時,使用這種方式初始化比較方便。
    的頭像 發(fā)表于 02-17 09:32 ?743次閱讀

    使用ARMClang V6版本編譯器出現(xiàn)錯誤的解決方法

    很多STM32開發(fā)者使用ARM mdk IDE進(jìn)行開發(fā),我們知道ARM MDK IDE早已推出ARMClang V6版本編譯器了。不過,當(dāng)我們把在V
    的頭像 發(fā)表于 06-07 12:32 ?5029次閱讀
    使用ARMClang V6<b class='flag-5'>版本</b><b class='flag-5'>編譯器</b>出現(xiàn)錯誤的解決方法

    基于ARM編譯器版本5工程遷移適配ARM編譯器版本6.12

    AC5和AC6的主要差異是AC6使用armclang代替了armcc,因此在AC6中就沒有armcc這個編譯工具了。并且armclang的編譯參數(shù)相對于之前的armcc的編譯參數(shù)也有許
    的頭像 發(fā)表于 11-12 11:17 ?8152次閱讀
    基于<b class='flag-5'>ARM</b><b class='flag-5'>編譯器</b><b class='flag-5'>版本</b><b class='flag-5'>5</b>的<b class='flag-5'>工程</b><b class='flag-5'>遷移</b>與<b class='flag-5'>適配</b>到<b class='flag-5'>ARM</b><b class='flag-5'>編譯器</b><b class='flag-5'>版本</b>6.12