一 ?u-boot的內(nèi)存分配
u-boot重定位到內(nèi)存以后,是有內(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