广州市建设企业网站平台wordpress破解版
一. uboot启动流程涉及函数
本文简单分析一下 save_boot_params_ret调用的函数:_main汇编函数。
本文继之前文章的学习,地址如下:
uboot启动流程-涉及s_init汇编函数_凌肖战的博客-CSDN博客
二. uboot启动流程涉及的 _main汇编函数
经过之前文章的分析,uboot启动流程的汇编函数调用关系:

下面来分析 _main 函数。
_main 函数定义在文件 arch/arm/lib/crt0.S 中,_main函数的前部分代码内容如下:
67 ENTRY(_main)
68 
69 /*
70 * Set up initial C runtime environment and call board_init_f(0).
71 */
72 
73 #if defined(CONFIG_SPL_BUILD) && defined(CONFIG_SPL_STACK)
74 ldr sp, =(CONFIG_SPL_STACK)
75 #else
76 ldr sp, =(CONFIG_SYS_INIT_SP_ADDR)
77 #endif
78 #if defined(CONFIG_CPU_V7M) /* v7M forbids using SP as BIC 
destination */
79 mov r3, sp
80 bic r3, r3, #7
81 mov sp, r3
82 #else
83 bic sp, sp, #7 /* 8-byte alignment for ABI compliance */
84 #endif
85 mov r0, sp
86 bl board_init_f_alloc_reserve 
 
 
 第  76  行,设置  sp  指针为  CONFIG_SYS_INIT_SP_ADDR ,也就是  sp  指向  0X0091FF00 。  
 
 第  83  行, sp  做  8  字节对齐。  
 
 第  85  行,读取  sp  到寄存器  r0  里面,此时  r0=0X0091FF00 。  
 
 
 第 86 行,调用 board_init_f_alloc_reserve  函数 ,此函数有一个参数,参数为 r0  中的值,也 就是  0X0091FF00 ,此函数定义在文件  common/init/board_init.c  中,内容如下: 
 
 
56 ulong board_init_f_alloc_reserve(ulong top)
57 {
58 /* Reserve early malloc arena */
59 #if defined(CONFIG_SYS_MALLOC_F)
60 top -= CONFIG_SYS_MALLOC_F_LEN;
61 #endif
62 /* LAST : reserve GD (rounded up to a multiple of 16 bytes) */
63 top = rounddown(top-sizeof(struct global_data), 16);
64
65 return top;
66 } 
 
 函数  board_init_f_alloc_reserve  主要是留出早期的  malloc  内存区域和  gd  内存区域,其中,CONFIG_SYS_MALLOC_F_LEN=0X400(  在文件  include/generated/autoconf.h  中定义  )  ,  sizeof(struct global_data)=248(GD_SIZE  值),完成以后的内存分布如下: 
 
 

 board_init_f_alloc_reserve  是有返回值的,返回值为新的  top  值,从上面的内存分布 可知,  此时, top=0X0091FA00 。  
 
 
 
 
继续分析_main 函数,如下是继以上代码:
87 mov sp, r0
88 /* set up gd here, outside any C code */
89 mov r9, r0
90 bl board_init_f_init_reserve 
 
 
 第  87  行,将  r0  写入到  sp  里面,r0 保存着 board_init_f_alloc_reserve 函数的返回值,所以这一句也就是设置 sp=0X0091FA00 。  
 
 第  89  行,将  r0  寄存器的值写到寄存器  r9  里面,因为  r9  寄存器存放着全局变量  gd  的地址,  在文件  arch/arm/include/asm/global_data.h  中。如下所示:   
 
 

 可以看出, uboot  中定义了一个指向  gd_t  的指针  gd , gd  存放在寄存器  r9  里面的,因此  gd  是个全局变量。 gd_t  是个结构体,在  include/asm-generic/global_data.h  里面有定义,  gd_ 定义如下: 
 
 
27 typedef struct global_data {
28 bd_t *bd;
29 unsigned long flags;
30 unsigned int baudrate;
31 unsigned long cpu_clk; /* CPU clock in Hz! */
32 unsigned long bus_clk;
33 /* We cannot bracket this with CONFIG_PCI due to mpc5xxx */
34 unsigned long pci_clk;
35 unsigned long mem_clk;
36 #if defined(CONFIG_LCD) || defined(CONFIG_VIDEO)
37 unsigned long fb_base; /* Base address of framebuffer mem */
38 #endif
......
121 #ifdef CONFIG_DM_VIDEO
122 ulong video_top; /* Top of video frame buffer area */
123 ulong video_bottom; /* Bottom of video frame buffer area */
124 #endif
125 } gd_t; 
 
 
 因此,_main函数的第 89 行代码就是设置  gd  所指向的位置,也就是  gd  指向  0X0091FA00 。  
 
 
 第  90  行调用函数  board_init_f_init_reserve ,此函数在文件common/init/board_init.c  中有定义,函数内容如下:  
 
110 void board_init_f_init_reserve(ulong base)
111 {
112 struct global_data *gd_ptr;
113 #ifndef _USE_MEMCPY
114 int *ptr;
115 #endif
116
117 /*
118 * clear GD entirely and set it up.
119 * Use gd_ptr, as gd may not be properly set yet.
120 */
121
122 gd_ptr = (struct global_data *)base;
123 /* zero the area */
124 #ifdef _USE_MEMCPY
125 memset(gd_ptr, '\0', sizeof(*gd));
126 #else
127 for (ptr = (int *)gd_ptr; ptr < (int *)(gd_ptr + 1); )
128 *ptr++ = 0;
129 #endif
130 /* set GD unless architecture did it already */
131 #if !defined(CONFIG_ARM)
132 arch_setup_gd(gd_ptr);
133 #endif
134 /* next alloc will be higher by one GD plus 16-byte alignment */
135 base += roundup(sizeof(struct global_data), 16);
136
137 /*
138 * record early malloc arena start.
139 * Use gd as it is now properly set for all architectures.
140 */
141
142 #if defined(CONFIG_SYS_MALLOC_F)
143 /* go down one 'early malloc arena' */
144 gd->malloc_base = base;
145 /* next alloc will be higher by one 'early malloc arena' size */
146 base += CONFIG_SYS_MALLOC_F_LEN;
147 #endif
148 } 
 
 可以看出,此函数用于初始化  gd ,其实就是清零处理。另外,此函数还设置了  gd->malloc_base  为  gd  基地址 +gd  大小 =0X0091FA00+248=0X0091FAF8 ,在做  16  字节对齐,最  终 gd->malloc_base=0X0091FB00 ,这个也就是  early malloc  的起始地址。  
 
 
 
继续分析_main 函数,如下是继以上代码:
92 mov r0, #0
93 bl board_init_f
94 
95 #if ! defined(CONFIG_SPL_BUILD)
96 
97 /*
98 * Set up intermediate environment (new sp and gd) and call
99 * relocate_code(addr_moni). Trick here is that we'll return
100 * 'here' but relocated.
101 */
102
103 ldr sp, [r9, #GD_START_ADDR_SP] /* sp = gd->start_addr_sp */
104 #if defined(CONFIG_CPU_V7M) /* v7M forbids using SP as BIC 
destination */
105 mov r3, sp
106 bic r3, r3, #7
107 mov sp, r3
108 #else
109 bic sp, sp, #7 /* 8-byte alignment for ABI compliance */
110 #endif
111 ldr r9, [r9, #GD_BD] /* r9 = gd->bd */
112 sub r9, r9, #GD_SIZE /* new GD is below bd */ 
 
 
 第  92  行设置  R0  为  0 。  
 
 第  93  行,调用  board_init_f  函数,此函数定义在文件  common/board_f.c  中!主要用来初始化  DDR ,定时器,完成代码拷贝等等,此函数我们后面在详细的分析。  
 
 
 第 103 行,重新设置环境(sp 和 gd)、获取 gd->start_addr_sp 的值赋给 sp,在函数 board_init_f 中会初始化 gd 的所有成员变量,其中 gd->start_addr_sp=0X9EF44E90, 所以这里相当于设置  
 
 sp=gd->start_addr_sp=0X9EF44E90。0X9EF44E90 是 DDR 中的地址,说明新的 sp 和 gd 将会存 放到 DDR 中,而不是内部的 RAM 了。 
 
 
 第 109 行,sp 做 8 字节对齐。  
 
 
 第 111 行,获取 gd->bd 的地址赋给 r9,此时 r9 存放的是老的 gd,这里通过获取 gd->bd 的地址来计算出新的 gd 的位置。GD_BD=0。 
 
 
 第 112 行,新的 gd 在 bd 下面,所以 r9 减去 gd 的大小就是新的 gd 的位置,获取到新的 gd 的位置以后赋值给 r9。  
 
 
 
 
 
下一篇文章继续分析 _main函数。
