当前位置: 首页 > 工具软件 > ARMboot > 使用案例 >

uboot第2阶段分析 start_armboot

邹麻雀
2023-12-01

通过前面的分析,知道第2阶段代码主要是start_armboot,首先我们先介绍几个比较重要的结构体。这里主要是对start_armboot中的流程做具体的分析,我的uboot-2009-06.至于具体的代码分析将在后面的文章中介绍。

1.gd_t全局数据结构,主要保存了uboot的配置信息。include/asm-arm/global_data.h中定义:

typedef struct global_data {
 bd_t  *bd;//板子相关的结构体
 unsigned long flags;//

 unsigned long baudrate;//波特率
 unsigned long have_console; /* 串口调用*/
 unsigned long reloc_off; /* Relocation Offset 重定位偏移*/
 unsigned long env_addr; /* 环境变量基址*/
 unsigned long env_valid; /* Checksum of Environment valid? */
 unsigned long fb_base; /*帧缓冲基地址*/
#ifdef CONFIG_VFD
 unsigned char vfd_type; /* display type */
#endif
#if 0
 unsigned long cpu_clk; /* cpu频率!  */
 unsigned long bus_clk;/*总线频率*/
 phys_size_t ram_size; /* RAM大小设置,64M*/
 unsigned long reset_status; /* 重新设置状态地址*/
#endif
 void  **jt;  /* jump table 保存着些函数的入口地址,在common/Exports.c中进行填充*/
} gd_t;

#define DECLARE_GLOBAL_DATA_PTR     register volatile gd_t *gd asm ("r8")

这个声明告诉编译器使用寄存器r8来存储gd_t类型的指针gd,即这个定义声明了一个指针,并且指明了它的存储位置。
register表示变量放在机器的寄存器
volatile用于指定变量的值可以由外部过程异步修改

这个指针在board.c中初始化。

2.bd_t板子数据结构体,主要保存与板子相关的信息。定义在include/asm-arm/u-boot.h中

typedef struct bd_info {
    int   bi_baudrate; /* 串口频率*/
    unsigned long bi_ip_addr; /*IP地址*/
    struct environment_s        *bi_env;环境变量指针
    ulong         bi_arch_number; /* 板子ID*/
    ulong         bi_boot_params; /* 启动参数存放地址*/
    struct    /* RAM configuration */
    {
 ulong start;
 ulong size;
    }   bi_dram[CONFIG_NR_DRAM_BANKS];//SDRAM的大小,此处的CONFIG_NR_DRAM_BANKS在smdk2410.h中宏定义
} bd_t;

3.初始化函数队列

typedef int (init_fnc_t) (void);定义函数指针;

下面是函数指针数组

init_fnc_t *init_sequence[] = {
 cpu_init,  /* CPU初始化,定义在cpu/arm920t/cpu.c*/
#if defined(CONFIG_ARCH_CPU_INIT)
 arch_cpu_init,  /* 相关体系结构CPU初始化cpu/.../arch/cpu.c */
#endif
 board_init,  /* 相关板子初始化board/sbc2410x/sbc2410.c*/
 interrupt_init,  /* 中断初始化cpu/arm920t/ s3c24x0/interrupts.c*/
 env_init,  /* 环境变量初始化*/
 init_baudrate,  /* 初始化波特率*/
 serial_init,  /* 串口初始化drivers/serial/serial.c*/
 console_init_f,  /* stage 1 init of console ???不知何意*/
 display_banner,  /* say that we are here */
#if defined(CONFIG_DISPLAY_CPUINFO)
 print_cpuinfo,  /* 打印CPU信息 */
#endif
#if defined(CONFIG_DISPLAY_BOARDINFO)
 checkboard,  /* 显示板块信息*/
#endif
#if defined(CONFIG_HARD_I2C) || defined(CONFIG_SOFT_I2C)
 init_func_i2c,IIC初始化函数
#endif
 dram_init,  /* configure available RAM banks */
#if defined(CONFIG_CMD_PCI) || defined (CONFIG_PCI)
 arm_pci_init,
#endif
 display_dram_config,存储器件初始化
 NULL,

}

 

 

OK现在开始分析start_armboot,这里只分析流程。

 

 

void start_armboot (void)
{
 init_fnc_t **init_fnc_ptr;//定义二级函数指针
 char *s;
#if defined(CONFIG_VFD) || defined(CONFIG_LCD)//如果定义了Lcd或者VFD,则指定帧缓冲的地址
 unsigned long addr;
#endif

 /* Pointer is writable since we allocated a register for it */

 

/*定义了全局数据指针,见上1,_armboot_start的值位0x3ff80000,且CONFIG_SYS_MALLOC_LEN在smdk2410.h中定义,经过计算可知道config_sys_malloc_len大小为0x30000,sizeof(gd_t)=4+4*7+4=36=0x24.故gd所指向地址0x3ff80000-0x30000-0x24约为ox3ff50000*/
 gd = (gd_t*)(_armboot_start - CONFIG_SYS_MALLOC_LEN - sizeof(gd_t));
 /*下面这局话请参考内存屏障*/
 __asm__ __volatile__("": : :"memory");

 memset ((void*)gd, 0, sizeof (gd_t));
/*对板块信息指针初始化*/

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

/*设置当前uboot标志位已经重定位*/

gd->flags |= GD_FLG_RELOC;

 

monitor_flash_len = _bss_start - _armboot_start;//uboot镜像文件大小

/*初始化函数*/

 for (init_fnc_ptr = init_sequence; *init_fnc_ptr; ++init_fnc_ptr) {
  if ((*init_fnc_ptr)() != 0) {
   hang ();
  }
 }

 

 

//flash初始化以及打印其信息

display_flash_config (flash_init ());

 

//malloc内存空间清零

mem_malloc_init (_armboot_start - CONFIG_SYS_MALLOC_LEN);

 

//如果定义了nandflash,则对nandflash初始化

#if defined(CONFIG_CMD_NAND)
 puts ("NAND:  ");
 nand_init();  /* go init the NAND */
#endif

 

 

//环境初始化

env_relocate ();

 

 

serial_initialize();

 

//设置IP地址,以及设备初始化

gd->bd->bi_ip_addr = getenv_IPaddr ("ipaddr");

 devices_init (); /* get the devices list going. */

 jumptable_init ();//初始化跳转表,对gd中的jt(函数跳转表)数组进行初始化,其中保存着一些函数的入口地址

 

 


console_init_r (); /* fully init console as a device */

 

misc_init_r ();

 

 

//中断使能

enable_interrupts ();


//得到当前网卡物理地址

cs8900_get_enetaddr ();

//设置网卡

eth_initialize(gd->bd)

for (;;) {
  main_loop ();
 }

}

 类似资料: