分散加載文件淺析
先發(fā)個(gè)鬧騷:昨晚腦殘的半夜2點(diǎn)刷機(jī),可是忘了備份,所以手機(jī)里現(xiàn)在是空空如也。fuck !!! ? fucking....
好了,下面開始正文。
? ? ? 官方描述:通過使用分散加載機(jī)制,可以為鏈接器指定映像的內(nèi)存映射。分散加載為您提供了對(duì)映像組建分組和位置的全面控制。分散加載可以用于簡(jiǎn)單映像,但它通常僅用于具有復(fù)雜內(nèi)存映射的的映像,即多個(gè)區(qū)在加載和執(zhí)行是分散在內(nèi)存映射中。
? ? ?個(gè)人理解:就是通過分散加載文件來自己管理代碼的內(nèi)存分布。
? ? ? 我們知道映像是由區(qū)和輸出節(jié)組成的。映像中的每個(gè)區(qū)可以包含不同的加載和執(zhí)行地址,要構(gòu)建映像的內(nèi)存映射,鏈接器必須具有描述如何將輸入節(jié)劃分到輸出節(jié)和區(qū)中的分組信息,和描述區(qū)位于內(nèi)存映射中的地址和位置信息。那么我們就知道分散加載文件的組成肯定有加載域,執(zhí)行域和輸入節(jié)三個(gè)必要的部分。
上面的三個(gè)名詞就不用解釋了,看字面意思也可以理解,下面直接上例子。
? ? ? 三個(gè)組件的結(jié)構(gòu)方式大同小異:
? ? ? ? ?名稱基址(偏移量)(特性的屬性)最大大小{
? ? ? ? ? ? ? 指定執(zhí)行區(qū)(輸入節(jié))的名稱,地址(屬性)和內(nèi)容
? ? ? ? ?}
? ps:括弧內(nèi)部的為可選擇的選項(xiàng)
? ? ? ? ?首先講一下數(shù)據(jù)的三種基本屬性:+RO為只讀數(shù)據(jù), +RW 為讀寫數(shù)據(jù) +ZI指未初始化的或者映像執(zhí)行前必須設(shè)為0的內(nèi)存片段。
? ? ? ? ?有許多ARM庫節(jié)必須放置在根區(qū)中,例如__main.o,__scatter*.o,__dc*.o和*Region$$Table等,鏈接器可以使用InRoot$$Section自動(dòng)放置所有節(jié),而不會(huì)影響到將來的使用,也就是初始入口點(diǎn),初始入口點(diǎn)必須滿足兩個(gè)條件,一個(gè)是映像入口點(diǎn)必須始終在執(zhí)行區(qū)內(nèi),二執(zhí)行區(qū)必須是非重疊的,而且必須是根執(zhí)行區(qū),所謂根區(qū)就是加載地址與執(zhí)行地址相同的區(qū)。
? ? ? ? ?關(guān)于.ANY:指通過使用特殊模塊選擇器模式.ANY,可以將輸入節(jié)分配給執(zhí)行區(qū),而無需考慮其父模塊??梢允褂?ANY以任意分配方式填充執(zhí)行區(qū)。
? ? ? ? ?接下來就是屬性了,前面我們用到了一個(gè)偽屬性FIRST,還有一個(gè)偽屬性LAST,表示其加載的順序,自我理解有點(diǎn)優(yōu)先級(jí)的意思。下面我就列舉幾個(gè)我們常用的屬性,這東西就跟字典一樣,大概知道有這么個(gè)東西就行,到用的時(shí)候知道在哪里可以查到就可以了。
? ? ? ? ?EMPTY只適用于執(zhí)行區(qū),如果在加載區(qū)中定義了適用了該屬性,鏈接器將生成警告并忽略該屬性,表示在執(zhí)行區(qū)中保留一個(gè)給定長(zhǎng)度的空白內(nèi)存塊,通常供堆棧使用。不能將任何字節(jié)放置在該屬性的區(qū)中。
? ? ? ? ?例如:ARM_LIB_HEAP 0x10008000 EMPTY 0x1000{
? ? ? ? ? ?}
? ? ? ? ? ? ? ? ? ?ARM_LIB_STACK 0x10009000 EMPTY 0x1000{
? ? ? ? ? ? }
? ? ? ? ? OVERLAY用于具有重疊的地址范圍的節(jié)。將為具有該屬性且基址偏移為+0的連續(xù)執(zhí)行區(qū)指定相同的基址。
? ? ? ? ? UNINIT用于創(chuàng)建包含未初始化的數(shù)據(jù)或內(nèi)存映射的I/O的執(zhí)行區(qū)。
? ? ? ? ? 其實(shí)更多的是我們自己去指定一個(gè)屬性,例如我們可以這樣定義一個(gè)數(shù)組:
? ? ? ? ? ? ? ? ? ? ? ? volatile uint8_t i2c_tx_buff[MAX_SIZE] __attribute__((section("i2c_ex_buff")));
? ? ? ? ? 那么我們就可以在執(zhí)行域中這樣寫
? ? ? ? ? ? ? ? EW_IRAM2 0x10000000 0x00008000 {
? ? ? ? ? ? ? ? ? ? ? ? ? ?.ANY(+RW +ZI)
? ? ? ? ? ? ? ? ? ? ? ? ? ?*.o(i2c_tx_buff)
? ? ? ? ? ? ? ? ?}
? ? ? ? ? 當(dāng)然一個(gè)工程可以對(duì)應(yīng)多個(gè)加載域,我們可以通過加載域去剝離工程中的某個(gè)源文件,實(shí)現(xiàn)對(duì)單個(gè)文件的加密等等。
? ? ? ? ?對(duì)于分散加載文件還有一些專門針對(duì)的函數(shù)可以使用,匯編語言程序可以導(dǎo)入這些符號(hào)地址并將其用作可重定位的地址,或者從c或者c++源代碼中將其作為extern符號(hào)進(jìn)行引用,說白了就是用特定的函數(shù)獲取分散加載文件中的一些參數(shù)。注意:僅當(dāng)代碼引用鏈接器定義的符號(hào)時(shí),才會(huì)生成這些符號(hào)。
? ? ? ? ?與執(zhí)行域有關(guān)的函數(shù)有:ImageBase(region_name),ImageLength(region_name),ImageLimit(region_name).它們對(duì)應(yīng)的鏈接器定義的符號(hào)值為Image$$region_name$$Base,Image$$region_name$$Length,Image$$region_name$$Limit.意思分別為求出基址,求出長(zhǎng)度,求出大小。與加載域相關(guān)的只需要把Image修改為L(zhǎng)oad即可。
就先寫這些吧。若有不足,歡迎指正。。。