一、链接脚本
OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
OUTPUT_ARCH(arm)
ENTRY(_start) //指定输出可执行文件的起始代码段为_start
SECTIONS
{
. = 0x00000000;
. = ALIGN(4);
.text :
{
*(.__image_copy_start)
*(.vectors)
arch/arm/cpu/armv7/start.o (.text*)
*(.text*)
}
. = ALIGN(4);
.rodata : { *(SORT_BY_ALIGNMENT(SORT_BY_NAME(.rodata*))) }
. = ALIGN(4);
.data : {
*(.data*)
}
. = ALIGN(4);
. = .;
. = ALIGN(4);
.u_boot_list : {
KEEP(*(SORT(.u_boot_list*)));
}
. = ALIGN(4);
.image_copy_end :
{
*(.__image_copy_end)
}
.rel_dyn_start :
{
*(.__rel_dyn_start)
}
.rel.dyn : {
*(.rel*)
}
.rel_dyn_end :
{
*(.__rel_dyn_end)
}
.end :
{
*(.__end)
}
_image_binary_end = .;
. = ALIGN(4096);
.mmutable : {
*(.mmutable)
}
.bss_start __rel_dyn_start (OVERLAY) : {
KEEP(*(.__bss_start));
__bss_base = .;
}
.bss __bss_base (OVERLAY) : {
*(.bss*)
. = ALIGN(4);
__bss_limit = .;
}
.bss_end __bss_limit (OVERLAY) : {
KEEP(*(.__bss_end));
}
.dynsym _image_binary_end : { *(.dynsym) }
.dynbss : { *(.dynbss) }
.dynstr : { *(.dynstr*) }
.dynamic : { *(.dynamic*) }
.plt : { *(.plt*) }
.interp : { *(.interp*) }
.gnu.hash : { *(.gnu.hash) }
.gnu : { *(.gnu*) }
.ARM.exidx : { *(.ARM.exidx*) }
.gnu.linkonce.armexidx : { *(.gnu.linkonce.armexidx.*) }
}
整个程序入口为_start,位于arch/arm/lib/vectors.S中。
从0地址开始存放 .text 代码段
.rodate 只读数据段
.data 数据段
.u_boot_list
.image_copy_end
.rel_dyn_start
……
二、uboot的入口就是在_start
//arm/arm/lib/vector.S
b reset
//arch/arm.cpu/armv7/start.S
save_boot_params_ret //关中断
// 在CP15 VBAR寄存器中设置矢量地址
bl cpu_init_cp15 //无效icache,dcache等
bl cpu_init_crit
b lowlevel_init //跳转至arch\arm\cpu\armv7\lowlevel_init.S
bl s_init //跳转至在arch\arm\cpu\armv7\mx6\soc.c
pop {ip, pc}
bl _main
三、此时将跳转到第二阶段_main在arch\arm\lib\crt0.S中定义
ENTRY(_main) //crt0.S
bl board_init_f //跳转至common/board_f.c
zero_global_data();
initcall_run_list(init_sequence_f)
//回调init_sequence_f