關(guān)于u-boot的內(nèi)存分配問題

一 ?u-boot的內(nèi)存分配

u-boot重定位到內(nèi)存以后,是有內(nèi)存空間分配的,如圖所示

u-boot的內(nèi)存分配

首先需要注意的是,_start,即u-boot的起始點(diǎn),在u-boot重定位的時(shí)候其實(shí)是叫_armboot_star,在start.s中

relocate: /* relocate U-Boot to RAM ? ? */

adr r0, _start /* r0 <- current position of code? */

ldr r1, _TEXT_BASE /* test if we run from flash or RAM */

cmp? ? r0, r1? ? ? ? ? ? ? ? ? /* don't reloc during debug? ? ? ? */

beq? ? stack_setup

ldr r2, _armboot_start

ldr r3, _bss_start

sub r2, r3, r2 /* r2 <- size of armboot? ? ? ? ? ? */

add r2, r0, r2 /* r2 <- source end address? ? ? ? */

將_armboot_start開始的部分到_bss_start都復(fù)制到_TEXT_BASE的基址位置,這個(gè)_TEXT_BASE在reset的地方定義了.記著_armboot_start是U-boot開始的地方,start_armboot是第二階段代碼入口,不要搞錯(cuò)了!

.word TEXT_BASE

.globl _armboot_start

_armboot_start:

.word _start

另外一個(gè)需要注意的是GBL_DATA_SIZE就是gd_t和bd_t的空間大小

第三點(diǎn)需要注意的是,地址0開始的其實(shí)是中斷向量表,真正的u-boot是reset開始

二 u-boot的內(nèi)存分配代碼和start_armboot的內(nèi)存分配代碼部分區(qū)別

u-boot中有設(shè)置堆棧的代碼,如下

stack_setup:

ldr r0, _TEXT_BASE /* upper 128 KiB: relocated uboot? */

sub r0, r0, #CFG_MALLOC_LEN /* malloc area? ? ? ? ? ? ? ? ? ? ? */

sub r0, r0, #CFG_GBL_DATA_SIZE /* bdinfo? ? ? ? ? ? ? ? ? ? ? ? */

#ifdef CONFIG_USE_IRQ

sub r0, r0, #(CONFIG_STACKSIZE_IRQ+CONFIG_STACKSIZE_FIQ)

#endif

sub sp, r0, #12 /* leave 3 words for abort-stack? ? */

這里目的是為了設(shè)置棧指針,雖然減去了堆,中斷,全局變量的空間,但這里只是預(yù)留了空間,但是指針并沒有指向這里,所以還不能用。在第二階段才真正安排了空間。

void start_armboot (void)

{

gd = (gd_t*)(_armboot_start - CFG_MALLOC_LEN - sizeof(gd_t));

memset ((void*)gd, 0, sizeof (gd_t));

gd->bd = (bd_t*)((char*)gd - sizeof(bd_t));

memset (gd->bd, 0, sizeof (bd_t));

三 ?環(huán)境變量與u-boot的參數(shù)區(qū)別

? ? 環(huán)境變量存了波特率,自啟動(dòng)延遲時(shí)間,IP地址,MAC地址,tftp地址等信息,而u-boot的struct tag參數(shù)存放的是內(nèi)存地址,大?。ˋTAG_MEM),命令參數(shù)的存放地(ATAG_COMLINE)等的信息。

? ? 環(huán)境變量存放在哪?有兩種情況。它可以是內(nèi)嵌與u-boot中,u-boot重定位的時(shí)候也將它復(fù)制到了內(nèi)存中。也可以是存放在堆區(qū),在common/env_common.c中:

#ifdef ENV_IS_EMBEDDED

* The environment buffer is embedded with the text segment,

* just relocate the environment pointer

env_ptr = (env_t *)((ulong)env_ptr + gd->reloc_off);

DEBUGF ("%s[%d] embedded ENV at %p\n", __FUNCTION__,__LINE__,env_ptr);

#else

* We must allocate a buffer for the environment

env_ptr = (env_t *)malloc (CFG_ENV_SIZE);

DEBUGF ("%s[%d] malloced ENV at %p\n", __FUNCTION__,__LINE__,env_ptr);

#endif

如果定義了ENV_IS_EMBEDDED就內(nèi)嵌與u-boot中,否則就定義在堆區(qū)!

? ? U-Boot與Linux內(nèi)核的交互是單向的,U-Boot將各類參數(shù)傳遞給內(nèi)核。由于他們不能同時(shí)運(yùn)行,傳遞辦法只能有一個(gè)個(gè):U-Boot將參數(shù)放在某個(gè)約定的地方之后,在啟動(dòng)內(nèi)核,內(nèi)核啟動(dòng)后從這個(gè)地方獲得參數(shù)。

? ? 所以需要告訴內(nèi)核,u-boot將參數(shù)存放的地方,這個(gè)地址就是用gd_t->bd_t結(jié)構(gòu)體存放的。那這個(gè)地方是在內(nèi)存的哪個(gè)地方?在 board_init函數(shù)中:

//傳給Kernel的參數(shù)=(struct tag *)型的bd->bi_boot_params

//bd->bi_boot_params在board_init函數(shù)中初始化如對于at91rm9200,初始化在at91rm9200dk.c的board_init中進(jìn)行:bd->bi_boot_params=PHYS_SDRAM + 0x100;

//這個(gè)地址就是所有taglist的首地址

}

? ? 總結(jié)就是:為了實(shí)現(xiàn)u-boot與linux內(nèi)核通信,u-boot將自己的一些參數(shù)(這些參數(shù)就是存在u-boot的.rodata,.data等的數(shù)據(jù)吧)打包,打包成struct tag的數(shù)據(jù)結(jié)構(gòu),然后放到內(nèi)存的PHYS_SDRAM + 0x100;這個(gè)地方,并且把這個(gè)地址存在了bd_t結(jié)構(gòu)體中,即告訴linux內(nèi)核。

/*

* The environment buffer is embedded with the text segment,

* just relocate the environment pointer

*/

env_ptr = (env_t *)((ulong)env_ptr + gd->reloc_off);

DEBUGF ("%s[%d] embedded ENV at %p\n", __FUNCTION__,__LINE__,env_ptr);

#else

/*

* We must allocate a buffer for the environment

*/

env_ptr = (env_t *)malloc (CFG_ENV_SIZE);

DEBUGF ("%s[%d] malloced ENV at %p\n", __FUNCTION__,__LINE__,env_ptr);

#endif

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容