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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

什么是分散加載文件?何時進行分散加載

工程師鄧生 ? 來源:wenzi嵌入式軟件 ? 作者:wenzid ? 2022-09-14 10:38 ? 次閱讀

什么是分散加載文件

分散加載文件(scatter file)是一個文本文件,它的作用是可以用于描述 ARM 鏈接器生成映像文件所需要的信息。

如果不使用 scatter file 文件來指定,那么 ARM 鏈接器會按照默認的方式來生成映像文件,但是對于某些應用場景來說,我們希望能夠將一些數據放在指定的位置,這個時候,分散加載文件就發(fā)揮其作用了。

何時進行分散加載

在之前的一篇文章MCU 是如何從上電復位運行到 main 函數的?中詳細敘述了MCU運行到 main 函數之前所做的操作。簡而言之,主要做了如下三個工作:

堆棧以及堆的初始化

定位中斷向量表

調用 Reset Handler

下圖列出了ARM Cortex M4系列芯片的一個啟動流程,廠商不一樣,會存在細微的差別。

8195e6be-3371-11ed-ba43-dac502259ad0.png

由上述啟動流程可以看到,分散加載操作是在 __main() 函數內部完成的,緊接著,就運行 C 語言運行環(huán)境初始化 & C Library 的初始化。

分散加載的原理

在理解分散加載的原理之前,需要明白以下幾個概念:

Code: 為程序代碼部分

RO-Data: 表示程序定義的常量及const型數據

RW-Data:表示已經初始化的靜態(tài)變量,變量有初值

ZI-Data: 表示未初始化的靜態(tài)變量,變量無初值

除此之外,因為分散加載的機制是將不同代碼放在不同的存儲空間,因此還需要了解代碼的映像文件的基本概念。ARM 映像文件其實就是源文件經編譯器生成的目標文件 .obj(object file)和相應的 C/C++ 運行時庫( Runtime Library )經過連接器的處理后,生成的 axf 格式的映像文件,它可以直接燒錄到目標設備的 ROM 中直接運行或加載后運行。映像文件的組成如下所示:

82163daa-3371-11ed-ba43-dac502259ad0.png

映像文件的組成

由上圖可以知道,映像文件由域(區(qū))、輸出段(節(jié))和輸入段(節(jié))的層次結構組成:

輸入段:輸入段包含代碼、初始化數據,或描述未初始化的或在映像執(zhí)行之前必須設定為 0 的內存片段。這些特性通過 RO 、 RW 和 ZI 這樣的屬性來表示。

輸出段:一個輸出段由若干個具有相同 RO 、 RW 或 ZI 屬性的相鄰輸入段組成。輸出段的屬性與組成它的輸入段的屬性相同 。

域:一個域由一個、兩個或者三個相鄰的輸出段組成。區(qū)中的輸出段根據其屬性排序。首先是 RO 輸出段,然后是 RW 輸出段,最后是 ZI 輸出段。域通常映射到物理內存設備,如 ROM 、 RAM 或外圍設備。

ARM 映像文件各組成部分在存儲系統中的地址有兩種:

裝載域

運行域

在一個簡單的嵌入式計算機系統中,存儲器一般分為ROM和RAM。鏈接器生成的映像被分為“Read-Only”段和“Read-Write”段(包含已初始化數據和未初始化數據)。通常來說,在程序下載的時候,他們會被下載到ROM上,而在程序開始執(zhí)行的時候,Read-Write段會從ROM被Copy到RAM,下面就是這個加載過程的示意圖。

823334b4-3371-11ed-ba43-dac502259ad0.png

裝載域和運行域示意圖

以上只是一個簡單的例子,但是在比較復雜的嵌入式系統中,其存儲器往往還包括ROM,SRAM,DRAM,FLASH等等,這個時候就需要分散加載文件了。

分散加載的語法

分散加載文件主要由一個加載時域(區(qū))和多個運行時域(區(qū))組成,其大致結構如下圖所示:

825a0b70-3371-11ed-ba43-dac502259ad0.png

本次先介紹一種簡單的情況,一個Cortex M3系列的微控制器有Flash和RAM資源如下所示:

Flash基址:0x00000000,大小256KB;

RAM基址:0x10000000,大小32KB

那么分散加載文件可以這么寫:

LR_IROM10x000000000x00040000;定義一個加載域,域基址為0x00000000,大小為0x00040000
{;對應著實際的Flash的大小
ER_IROM10x000000000x00040000;定義一個運行域,第一個運行域必須和加載域起始地址相同
{
*.o(RESER,+First);將RESET最先加載到本域的起始地址,即RESET的起始地址為0
 .ANY(+RO);加載所有匹配目標文件的只讀屬性數據,包含:Code,RW-Data
}

RW_IRAM10x100000000x00008000;定義一個運行時域,域基址為0x10000000,域大小為0x00008000
{
*(+RW+ZI);加載所有匹配目標文件的RW-Data,ZI-Data
}
}




審核編輯:劉清

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

    關注

    146

    文章

    16888

    瀏覽量

    349929
  • ARM
    ARM
    +關注

    關注

    134

    文章

    9027

    瀏覽量

    366492
  • ROM
    ROM
    +關注

    關注

    4

    文章

    562

    瀏覽量

    85623
  • Cortex-M4
    +關注

    關注

    6

    文章

    89

    瀏覽量

    46494
  • 上電復位
    +關注

    關注

    1

    文章

    39

    瀏覽量

    15771

原文標題:keil分散加載文件淺析

文章出處:【微信號:wenzi嵌入式軟件,微信公眾號:wenzi嵌入式軟件】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    如何優(yōu)化EPS文件以提高加載速度

    在圖形設計和排版領域,EPS文件因其高兼容性和高質量輸出而廣受歡迎。然而,EPS文件往往體積較大,加載速度慢,這在處理大型項目或需要快速迭代時成為了一個瓶頸。 一、了解EPS文件 EP
    的頭像 發(fā)表于 10-30 14:32 ?115次閱讀

    AG32 下的分散加載

    配置,達成這樣的目的。 分散加載在gcc 下的限制: 不能插入到代碼段,只能放到正常編譯的 bin 后邊(參后詳述)。 在默認情況下,AG32編譯時使用的連接配置文件
    發(fā)表于 09-20 15:26

    AWR294x主引導加載程序和輔助引導加載程序

    電子發(fā)燒友網站提供《AWR294x主引導加載程序和輔助引導加載程序.pdf》資料免費下載
    發(fā)表于 09-06 09:47 ?0次下載
    AWR294x主引導<b class='flag-5'>加載</b>程序和輔助引導<b class='flag-5'>加載</b>程序

    【GD32 MCU 入門教程】七、分散加載說明

    分散加載說明以GD32F103ZE為例,分別用Keil、IAR和Embedded Builder工具實現:將函數放置某個地址、將常量放置某個地址、將函數放在RAM中運行的三種效果。 1、將
    的頭像 發(fā)表于 08-27 09:19 ?384次閱讀
    【GD32 MCU 入門教程】七、<b class='flag-5'>分散</b><b class='flag-5'>加載</b>說明

    如何用加載分散法將軟件中部分變量從內部RAM轉移到外部RAM?

    如何用加載分散法將軟件中部分變量從內部RAM轉移到外部RAM, 加載分散文件怎么設置?堆和棧需要設置嗎?
    發(fā)表于 05-10 07:52

    深入分析一下MDK的分散加載文件

    ARM C 庫提供了函數 __user_setup_stackheap() 的多個實現,并且可以根據分散文件中提供的信息自動選擇正確的一種。
    發(fā)表于 04-28 14:21 ?698次閱讀
    深入分析一下MDK的<b class='flag-5'>分散</b><b class='flag-5'>加載文件</b>

    stm32分散加載文件導致程序死機怎么解決?

    單片機 stm32f401ccu6 flash大小 16kb+16kb+16kb+16kb+64kb+128kb = 256kb 為了使用留出參數空間,將代碼空間分為三段,其中第二段留空供參數使用 當使用以下配置時可以正常運行,使用FAL命令擦寫參數段是可以的 LR_IROM1 0x08000000 0x000040000 { ; load region size_region ER_IROM1 0x08000000 0x00010000 { ; load address = execution address *.o (RESET, +First) *(InRoot$$Sections) .ANY (+RO) } ER_IROM2 0x08020000 FIXED 0x00040000 { ; load address = execution address .ANY (+RO) } RW_IRAM1 0x20000000 0x00010000 { ; RW data .ANY (+RW +ZI) } } 但是將第二段空間改為0x00020000時程序編譯正常,但是燒錄死機 ER_IROM2 0x08020000 FIXED 0x00020000 { ; load address = execution address 以下是錯誤代碼 \\ | / - RT -Thread Operating System / | \\5.0.2 build Mar 12 2024 19:24:05 2006 - 2022 Copyright by RT-Thread team psr: 0x80000000 r00: 0x0803f330 r01: 0x0803f11c r02: 0x00000400 r03: 0x00000000 r04: 0xdeadbeef r05: 0xdeadbeef r06: 0xdeadbeef r07: 0xdeadbeef r08: 0xdeadbeef r09: 0xdeadbeef r10: 0xdeadbeef r11: 0xdeadbeef r12: 0x20001fd0 lr: 0x0802cdeb pc: 0x0803f330 hard fault on thread: main rt_thread_ threadpristatusspstack size max used left tickerror ---------- -------- ---------- ---------- -------------------------- --- 0x20001f78 ulog_asy30ready0x00000044 0x00000400 06%0x00000014 OK 0x20000ae8 tidle0 31ready0x00000044 0x00000100 26%0x00000020 OK 0x200016b8 main10running 0x00000044 0x00000800 18%0x00000013 OK usage fault: SCB_CFSR_UFSR:0x02 INVSTATEml
    發(fā)表于 03-26 06:31

    STM32L552VET6配置SDMMCH和文件系統,加載文件系統掛載存儲卡會返回FR_NOT_READY如何解決?

    STM32L552VET6配置SDMMCH和文件系統,不加載文件系統可以正常操作存儲卡,但是加載文件系統掛載存儲卡返回FR_NOT_READY,該如何解決。
    發(fā)表于 03-08 07:30

    使用分散加載將部分程序放到RAM,RAM掉電后數據就沒有了,如何復原?

    我使用分散加載將部分程序放到RAM,RAM掉電后數據就沒有了,重新上電后,芯片是如何將RAM區(qū)程序復原的呢。
    發(fā)表于 03-06 07:01

    怎么在rtthread studio中使用分散加載功能,工程里也找不到sct文件?

    怎么在rtthread studio中使用分散加載功能,工程里也找不到sct文件
    發(fā)表于 02-23 08:28

    STM32H750IBK6如何將指定代碼存放在外部QSPI FLASH里?

    如題,使用的STM32H750IBK6,內部只有128KB的FLASH,太小了。MDK使用的是分散加載文件來指定代碼存放內存。同理我嘗試修改了studio中的鏈接腳本,可無論如何都無法將指定代碼存放在外部QSPI FLASH里,是我鏈接腳本問題嗎?
    發(fā)表于 02-23 06:02

    如何加載ELF燒錄文件?

    請問在如下的界面中,如何加載ELF燒錄文件?
    發(fā)表于 02-01 08:25

    如何在LabVIEW中清晰加載圖片呢?

    有開發(fā)者提出,在使用LabVIEW開發(fā)圖片加載顯示程序時,為什么明明看著很清晰的圖片,LabVIEW加載顯示后就變得粗糙,線條不流暢。
    的頭像 發(fā)表于 12-20 09:08 ?1542次閱讀
    如何在LabVIEW中清晰<b class='flag-5'>加載</b>圖片呢?

    Keil分散加載文件淺析

    ARM 映像文件其實就是源文件經編譯器生成的目標文件,一般是bin文件或者hex文件,可以直接燒錄到ROM中執(zhí)行(一般是內部FLASH),這
    的頭像 發(fā)表于 11-17 10:00 ?2362次閱讀
    Keil<b class='flag-5'>分散</b><b class='flag-5'>加載文件</b>淺析

    在MATLAB中如何保存和加載消息

    到MAT文件中。 save ( 'posedata.mat' , 'posedata' ) 在將文件加載回工作空間之前,清除posedata變量。 clear posedata 現在可以通過調用load
    的頭像 發(fā)表于 11-15 15:17 ?369次閱讀