admin 管理员组文章数量: 1184232
bootm命令浅析
与boom命令实现有关的源文件主要为 common/cmd_bootm.c 和lib_arm/armlinux.c。下面就这两个文件 中的关键代码段进行 分析。
common/cmd_bootm.c(前面数字为Source Insight 中显示的行号):
168 if (argc < 2) {
判断运行bootm时是否指定了程序加载地址,若没有则使用默认的加载地址,load_addr在cmd_bootm.c中是这样定义的:
ulong load_addr = CFG_LOAD_ADDR;
183 memmove (&header, (char *)addr, sizeof(image_header_t));
185 if (ntohl(hdr->ih_magic) != IH_MAGIC) {
#ifdef __I386__
#endif
判断文件头中的幻数是否为IH_MAGIC,所以如果不是u-boot镜像格式,会输出提示信息”Bad Magic Number”
204 data = (ulong)&header;
比对u-boot image 文件头的CRC32校验和。
229 data = addr + sizeof(image_header_t);
比对u-boot image 数据部分的校验和。
245 #if defined(__PPC__)
#elif defined(__ARM__)
…
#else
# error Unknown CPU type
#endif
这部条件编译指令检验检验image header 中的arch类型是否是否正确。
275 switch (hdr->ih_type) {
判断image的类型,这里只列出了Kernel Image类型的代码,直接break,:)
switch (hdr->ih_comp) {
这段代码比较重要,它根据image所采用的压缩类型,将image解压到hdr->ih_load指向的地址,这个ih_load就是在mkimage中的-a选项指定的地址,这下明白了,-a选项指定的是内核解压后的地址。
412 switch (hdr->ih_os) {
#ifdef CONFIG_SILENT_CONSOLE
#endif
#ifdef DEBUG
#endif
既然内核已经解压完了,接下来就改启动Linux内核了,这里有用到了image header中的另一个字段ih_os:指明操作系统的类型,我只列出Linux操作系统类型的处理,可以看到它把控制权传递给了 do_bootm_linux 这个do_bootm_linux 对于不同的ARCH有不同的实现,而ARM的实现就是在 lib_arm/armlinux.c中。
armlinux.c : do_bootm_linux
83 ulong initrd_start, initrd_end;
initrd的起始地址和结束地址
85 void (*theKernel)(int zero, int arch, uint params);
Linux 内核的入口参数,zero = 0,arch为平台编号,params为传递给内核的参数在内存中的地址
89 #ifdef CONFIG_CMDLINE_TAG
#endif
如果在include/configs/<board name>.h定义了CONFIG_CMDLINE_TAG则将bootargs环境变量传递给内核。所以如果发现无法向内核传递参数,应该检查一下CONFIG_CMDLINE_TAG是否定义。
93 theKernel = (void (*)(int, int, uint))ntohl(hdr->ih_ep);
hdr为指向image header的指针,hdr->ih_ep就是我们用mkimage创建image时-e选项的参数:内核的入口地址。
从98到207行为处理bootm命令传递的initrd参数。包括判断bootm是否传递了initrd参数,检验initrd的有效性(幻数,校验和等)
下面这一段代码在内核启动前,向内核传递参数(以Tag标记的形式),这段代码依赖于很多的宏定义,比如CONFIG_CMDLINE_TAG等,它们通常定义在include/configs/<board name>.h中,所以如果要想u-boot给内核传递特定的标记,则必须定义相应的宏。传递标记是通过setup_<tag name>_tag函数完成的,具体可以参考cmd_boot.m里面的实现。
#if defined (CONFIG_SETUP_MEMORY_TAGS) || \
#ifdef CONFIG_SERIAL_TAG
#endif
#ifdef CONFIG_REVISION_TAG
#endif
#ifdef CONFIG_SETUP_MEMORY_TAGS
#endif
#ifdef CONFIG_CMDLINE_TAG
#endif
#ifdef CONFIG_INITRD_TAG
#endif
#if defined (CONFIG_VFD) || defined (CONFIG_LCD)
#endif
#endif
#ifdef CONFIG_USB_DEVICE
#endif
275 theKernel (0, bd->bi_arch_number, bd->bi_boot_params);
经过一系列的准备,终于到了启动内核的时候了,这里bd->bi_arch_number和bd->bi_boot_params在具体开发板的board_init函数里面初始化,比如对于smdk2410的board_init:
106 gd->bd->bi_arch_number = MACH_TYPE_SMDK2410;
107 gd->bd->bi_boot_params = 0x30000100;
smdk2410把启动参数放在了0x30000100开始的地方。上面提到的Tag的传递,也是放到这个地址。这点可以从armlinux.c中的setup_start_tag函数看出
static void setup_start_tag (bd_t *bd)
{
}
上面函数的第一句表明第一个param的地址为bd->bi_boot_params,到这了我们明白,u-boot向内核传递启动参数由一系列在include/configs/<board name>.h中的宏控制,启动参数传递的地址在board_init中初始化。
版权声明:本文标题:U-Boot Bootm命令:轻松实现内核启动与参数传递 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.roclinux.cn/b/1773420532a3561741.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论