u-boot分析



一、U-BOOT的目錄結構 u-boot目錄下有18個子目錄,分別存放管理不通的源程序。
這些目錄中所要存放的文件有其規則,可以分成三類。
■第一類目錄與處理器體系結構或者開發板硬件直接相關;
■第二類目錄是一些通用的函數或者驅動程序;
■第三類目錄是u-boot的應用程序、工具或者文檔。
Board:和一些已有開發板相關的文件,比如Makefile和u-boot.lds等都和具體開發板的硬件和地址分配有關。
Common:與體系結構無關的文件,實現各種命令的C文件。
CPU:CPU相關文件,其中的子目錄都是以u-boot所支持的CPU爲名,比如有子目錄arm926ejs、mips、mpc8260和nios等,每個特定的子目錄中都包括cpu.c和interrupt.c和start.S。其中cpu.c初始化cpu、設置指令cache和數據cache等;interrupt.c設置系統的各種終端和異常,比如快速中斷,開關中斷、時鐘中斷、軟件中斷、預取中止和未定義指令等;start.S是u-boot啓動時執行的第一個文件,他主要是設置系統堆棧和工作髮式,爲進入C程序奠定基礎。
Disk:disk驅動的分區處理代碼、
Doc:文檔。
Drivers:通用設備驅動程序,比如各種網卡、支持CFI的flash、串口和USB總線等。
Dtt:數字溫度測量器或者傳感器的驅動
Examples:一些獨立運行的應用程序的例子。
Fs:支持文件系統的文件,u-boot現在支持cramfs、fat、fdos、jffs2、yaffs和registerfs。
Include:頭文件,還有對各種硬件平臺支持的會變文件,系統的配置文件和對文件系統支持的文件。
Net:與網絡有關的代碼,BOOTP協議、TFTP協議RARP協議和NFS文件系統的實現。
Lib_ppc:存放對PowerPC體系結構通用的文件,主要用於實現PowerPC平臺通用的函數,與PowerPC體系結構相關的代碼。
Lib_i386:存放對X86體系結構通用的文件,主要用於實現X86平臺通用的函數,與PowerPc體系結構相關的代碼。
Lib_arm:存放對ARM體系結構通用的文件,主要用於實現ARM平臺通用的函數,與ARM體系結構相關的代碼。
Lib_generic:通用的多功能函數實現。
Post:上電自檢。
Rtc: 實時時鐘驅動。
Tools:創建S-Record格式文件和U-BOOT p_w_picpaths的工具。

二、u-boot的編譯 u-boot的源碼是通過GCC和Makefile組織編譯的,頂層目錄下的Makefile首先可以設置板子的定義,然後遞歸地調用各級目錄下的Makefile,最後把編譯過的程序鏈接成u-boot的映像。 頂層目錄下的Makefile,它是負責U-Boot整體配置編譯。每一種開發板在Makefile都需要有板子配置的定義,如smdk2442定義如下: smdk2442_config: unconfig @./mkconfig $(@:_config=) arm arm920t smdk2442 執行配置U-Boot的命令make smdk2442_config,通過./mkconfig腳本生成include/config.mk的配置文件。文件內容是根據Makefile對板子的配置生成的。

配置環境和編譯過程如下所述,U-boot的編譯環境配置需要:cross-2.95.3.tar.bz2和s3c24x0_uboot_rel_0_0_1_061002.tar.bz2,將文件拷貝到/home/amoi/working/下,(chenpx@chenpx:/mnt/hgfs/share$ cp cross-2.95.3.tar.bz2 /home/amoi/working 和chenpx@chenpx:/mnt/hgfs/share$ cp s3c-u-boot-1.1.6.tar.bz2 /home/amoi/working), 然後對對文件進行解壓(chenpx@chenpx:/home/chenpx/working$ tar jxvf cross-2.95.3.tar.bz2和chenpx@chenpx:/home/chenpx/working$ tar jxvf s3c24x0_uboot_rel_0_0_1_061002.tar.bz2),在/usr/local/目錄下建立一個arm文件夾(mkdir –p /usr/local/arm (-p 是需要時創建上層目錄,如目錄早已存在則不當作錯誤))

將cross-2.95.3.tar.bz2解壓出來的移動到/usr/local/arm/下(mv 2.95.3 /usr/local/arm/) 移動後添加環境變量export PATH=$PATH:/usr/local/2.95.3/bin/ 修改s3c24x0_uboot_dev中的makefile,修改CROSS_COMPILE = /usr/local/arm/2.95.3/bin/arm-linux-其他的用#註釋掉。 接下來就是加載配置: 最後進行編譯:make,最終在s3c24x0_uboot-dev目錄下生成u-boot、u-boot.bin、u-boot.map、2 u-boot.srec四個文件。

三、u-boot系統啓動流程 大多數bootloader都分爲stage1和stage2兩部分,u-boot也不例外。依賴於CPU體系結構的代碼(如設備初始化代碼等)通常都放在stage1且可以用彙編語言來實現,而stage2則通常用C語言來實現,這樣可以實現複雜的功能,而且有更好的可讀性和移植性。 1、Stage1 start.S代碼結構 u-boot的stage1代碼通常放在start.S文件中,他用彙編語言寫成,其主要代碼部分如下:

(1)定義入口。由於一個可執行的Image必須有一個入口點,並且只能有一個全局入口,通常這個入口放在ROM(Flash)的0x0地址,因此,必須通知編譯器以使其知道這個入口,該工作可通過修改連接器腳本來完成。
(2)設置異常向量(Exception Vector)。
(3)設置CPU的速度、時鐘頻率及終端控制寄存器。
(4)初始化內存控制器。
(5)將ROM中的程序複製到RAM中。
(6)初始化堆棧。
(7)轉到RAM中執行,該工作可使用指令ldr pc來完成。

2、Stage2 C語言代碼部分 lib_arm/board.c中的start arm boot是C語言開始的函數也是整個啓動代碼中C語言的主函數,同時還是整個u-boot(armboot)的主函數,該函數只要完成如下操作:
(1)調用一系列的初始化函數。
(2)初始化Flash設備。
(3)初始化系統內存分配函數。
(4)如果目標系統擁有NAND設備,則初始化NAND設備。
(5)如果目標系統有顯示設備,則初始化該類設備。
(6)初始化相關網絡設備,填寫IP、MAC地址等。
(7)進去命令循環(即整個boot的工作循環),接受用戶從串口輸入的命令,然後進行相應的工作。

3、U-Boot的啓動順序
主要順序如下圖所示

                     函數順序             初始化順序
                              圖爲 U-Boot順序
下面就根據代碼進行解釋:
/*********************** 中斷向量 ***********************/
.globl _start                      //u-boot啓動入口
_start: b    reset             //復位向量並且跳轉到reset
ldr pc, _undefined_instruction
ldr pc, _software_interrupt
ldr pc, _prefetch_abort
ldr pc, _data_abort
ldr pc, _not_used
ldr pc, _irq                   //中斷向量
ldr pc, _fiq                   //中斷向量
b   sleep_setting          //跳轉到sleep_setting
並通過下段代碼拷貝到內存裏
relocate:                             //把uboot重新定位到RAM
adr r0, _start                // r0 是代碼的當前位置
ldr r2, _armboot_start             //r2 是armboot的開始地址
ldr r3, _armboot_end             //r3 是armboot的結束地址
sub r2, r3, r2                   // r2得到armboot的大小
ldr r1, _TEXT_BASE          // r1 得到目標地址  
add r2, r0, r2                       // r2 得到源結束地址
copy_loop:                             //重新定位代碼
ldmia r0!, {r3-r10}                //從源地址[r0]中複製
stmia r1!, {r3-r10}                //複製到目標地址[r1]
cmp   r0, r2                      //複製數據塊直到源數據末尾地址[r2]
ble copy_loop
系統上電或reset後,cpu的PC一般都指向0x0地址,在0x0地址上的指令是
reset:                               //復位啓動子程序
/******** 設置CPU爲SVC32模式***********/
mrs r0,cpsr                       //將CPSR狀態寄存器讀取,保存到R0中
bic r0,r0,#0x1f
orr r0,r0,#0xd3
msr cpsr,r0
                  //將R0寫入狀態寄存器中
/************** 關閉看門狗 ******************/
ldr    r0, =pWTCON
mov     r1, #0x0
str    r1, [r0]
/************** 關閉所有中斷 *****************/
mov r1, #0xffffffff
ldr r0, =INTMSK
str r1, [r0]
ldr r2, =0x7ff
ldr r0, =INTSUBMSK
str r2, [r0]
/************** 初始化系統時鐘 *****************/
ldr r0, =LOCKTIME
ldr     r1, =0xffffff
str     r1, [r0]
clear_bss:
       ldr    r0, _bss_start           //找到bss的起始地址
       add    r0, r0, #4              //從bss的第一個字開始
       ldr    r1, _bss_end           // bss末尾地址
       mov    r2, #0x00000000    //清零  
clbss_l:str        r2, [r0]             // bss段空間地址清零循環
       add     r0, r0, #4
       cmp     r0, r1
       bne    clbss_l
/***************** 關鍵的初始化子程序 ************************/
/ * cpu初始化關鍵寄存器
* 設置重要寄存器
* 設置內存時鐘
* /
cpu_init_crit:
/** flush v4 I/D caches*/
mov r0, #0
mcr p15, 0, r0, c7, c7, 0 /* flush v3/v4 cache */
mcr p15, 0, r0, c8, c7, 0 /* flush v4 TLB */
/************* disable MMU stuff and caches ****************/
mrc p15, 0, r0, c1, c0, 0
bic r0, r0, #0x00002300 @ clear bits 13, 9:8 (--V- --RS)
bic r0, r0, #0x00000087 @ clear bits 7, 2:0 (B--- -CAM)
orr r0, r0, #0x00000002 @ set bit 2 (A) Align
orr r0, r0, #0x00001000 @ set bit 12 (I) I-Cache
mcr p15, 0, r0, c1, c0, 0
/******* 在重新定位前,我們要設置RAM的時間,因爲內存時鐘依賴開發板硬件的,你將會找到board目錄底下的memsetup.S。**************/
mov ip, lr
#ifndef CONFIG_S3C2440A_JTAG_BOOT
bl memsetup        //調用memsetup子程序(在board/smdk2442memsetup.S)
#endif
mov lr, ip
mov pc, lr                      //子程序返回

memsetup:
/**************** 初始化內存 **************/
       mov     r1, #MEM_CTL_BASE
       adrl r2, mem_cfg_val
       add     r3, r1, #52
1:    ldr     r4, [r2], #4
       str     r4, [r1], #4
       cmp     r1, r3
       bne     1b
/*********** 跳轉到原來進來的下一個指令(start.S文件裏) ***************/  
mov     pc, lr                 //子程序返回
/****************** 建立堆棧 *******************/
ldr r0, _armboot_end             //armboot_end重定位
add r0, r0, #CONFIG_STACKSIZE //向下配置堆棧空間
sub sp, r0, #12                //爲abort-stack預留個3字
/**************** 跳轉到C代碼去 **************/
ldr pc, _start_armboot           //跳轉到start_armboot函數入口,start_armboot
字保存函數入口指針
_start_armboot: .word start_armboot //start_armboot函數在lib_arm/board.c中實現
從此進入第二階段C語言代碼部分
/**************** 異常處理程序 *******************/
.align   5
undefined_instruction:             //未定義指令
get_bad_stack
bad_save_user_regs
bl   do_undefined_instruction
.align 5
software_interrupt:                //軟件中斷
get_bad_stack
bad_save_user_regs
bl   do_software_interrupt
.align 5
prefetch_abort:                   //預取異常中止
get_bad_stack
bad_save_user_regs
bl   do_prefetch_abort
.align 5
data_abort:                          //數據異常中止
get_bad_stack
bad_save_user_regs
bl   do_data_abort
.align 5
not_used:                         //未利用
get_bad_stack
bad_save_user_regs
bl   do_not_used
.align 5
irq:                                   //中斷請求
get_irq_stack
irq_save_user_regs
bl   do_irq
irq_restore_user_regs
.align 5
fiq:                                   //快速中斷請求
get_fiq_stack
/* someone ought to write a more effiction fiq_save_user_regs */
irq_save_user_regs
bl   do_fiq
irq_restore_user_regs
sleep_setting:                         //休眠設置
@ prepare the SDRAM self-refresh mode
ldr r0, =0x48000024 @ REFRESH Register
ldr r1, [r0]
orr r1, r1,#(1bd = &bd_data;
memset (gd->bd, 0, sizeof (bd_t));
monitor_flash_len = _armboot_end_data - _armboot_start;
/*** 調用執行init_sequence數組按順序執行初始化 ***/
for (init_fnc_ptr = init_sequence; *init_fnc_ptr; ++init_fnc_ptr) {
   if ((*init_fnc_ptr)() != 0) {
hang ();
   }
}
#if 0
/**************** 配置可用的flash單元 *************/
size = flash_init ();          //初始化flash
display_flash_config (size);    //顯示flash的大小
/******** _arm_boot在armboot.lds鏈接腳本中定義 ********/
#endif
#ifdef CONFIG_VFD
#   ifndef PAGE_SIZE
#   define PAGE_SIZE 4096
#   endif
/*********** 爲VFD顯示預留內存(整個頁面)   **********/
/******** armboot_real_end在board-specific鏈接腳本中定義********/
addr = (_armboot_real_end + (PAGE_SIZE - 1)) & ~(PAGE_SIZE - 1);
size = vfd_setmem (addr);
gd->fb_base = addr;
/******* 進入下一個界面 ********/
addr += size;
addr = (addr + (PAGE_SIZE - 1)) & ~(PAGE_SIZE - 1);
mem_malloc_init (addr);
#else
/********   armboot_real_end 在board-specific鏈接腳本中定義 *******/
mem_malloc_init (_armboot_real_end);
#endif /* CONFIG_VFD */
#if (CONFIG_COMMANDS & CFG_CMD_NAND)
puts ("NAND:");
nand_init();   /* NAND初始化 */
#endif
#ifdef CONFIG_HAS_DATAFLASH
AT91F_DataflashInit();
dataflash_print_info();
#endif
/********* 初始化環境 **********/
env_relocate ();
/*********** 配置環境變量,重新定位 **********/
#ifdef CONFIG_VFD
/* must do this after the framebuffer is allocated */
drv_vfd_init();
#endif
/* 從環境中得到IP地址 */
bd_data.bi_ip_addr = getenv_IPaddr ("ipaddr");
/*以太網接口MAC地址*/
{
   int i;
   ulong reg;
   char *s, *e;
   uchar tmp[64];
   i = getenv_r ("ethaddr", tmp, sizeof (tmp));
   s = (i > 0) ? tmp : NULL;
   for (reg = 0; reg bd->bi_enetaddr);
#endif
#ifdef CONFIG_DRIVER_LAN91C96
if (getenv ("ethaddr")) {
   smc_set_mac_addr(gd->bd->bi_enetaddr);
}
/* eth_hw_init(); */
#endif /* CONFIG_DRIVER_LAN91C96 */
/* 通過環境變量初始化*/
if ((s = getenv ("loadaddr")) != NULL) {
   load_addr = simple_strtoul (s, NULL, 16);
}
#if (CONFIG_COMMANDS & CFG_CMD_NET)
if ((s = getenv ("bootfile")) != NULL) {
   copy_filename (BootFile, s, sizeof (BootFile));
}
#endif /* CFG_CMD_NET */
#ifdef BOARD_POST_INIT
board_post_init ();
#endif
/* main_loop() 總是試圖自動啓動,循環不斷執行*/
for (;;) {
   main_loop (); /*主循環函數處理執行用戶命令—common/main.c
}
/* NOTREACHED - no way out of command loop except booting */
}


 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章