BootLoader与内核之间参数传递

发布者:dfdiqc最新更新时间:2018-02-16 来源: eefocus关键字:BootLoader  内核  参数传递 手机看文章 扫描二维码
随时随地手机看文章

    在嵌入式系统中,BootLoader 是用来初始化硬件,加载内核,传递参数。因为嵌入式系统的硬件环境各不相同,所以嵌入式系统的BootLoader 也各不相同,其中比较通用的是U-Boot,它支持不同的体系结构,如ARM,PowerPC,X86,MIPS 等。本文着重介BootLoader与内核之间参数传递这一基本功能。本文的硬件平台是基于AT91RM9200 处理器系统,软件平台是Linux-2.6.19.2 内核。内核映像文件为zImage。


    1. 系统硬件平台简介

    AT91RM9200 处理器,它是由Atmel 公司基于ARM920T 内核的微处理器,带有内存管理单元,CPU 时钟最高可达240MHz,它具有丰富的标准接口,EBI 接口,内部集成了静态存储控制器(SMC),SDRAM 控制器,BurST Flash 控制器。有关处理器的说明请参考AT91RM9200 的数据手册。本系统SDRAM(64MB)地址为:0x20000000, NorFlash(8MB)的地址为:0x10000000[1]。

    2. BootLoader 设计和实现

    内核源代码目录树下的documentatiON/arm/booting[2]文档规定了基于ARM 体系结构BootLoader 的基本功能。本系统BootLoader 除了完成这些基本的功能外,还结合自身硬件的特点加入了代码搬运等功能。

    BootLoader 的流程是:系统上电复位后,首先从NorFlash 开始运行(由处理器BMS 引脚连接决定),因为处理器此时的0 地址就是NorFlash 的首地址(0x10000000),BootLoader就是被烧写在这个位置,AT91RM9200 处理器能够映射的地址范围只有0x0000

    0000—0x001f ffff。 BootLoader 执行的第一步就是将自身代码从NorFlash 中搬运到处理器内部的RAM 中(0x00200000),然后将0 地址映射到内部RAM,并且跳转到内部RAM 的相应地址处继续执行。进入内部RAM 后才进入真正的硬件初始化阶段,这个阶段初始化的各种控制器都是内核所必须的,包括:PMC, EBI, SMC, SDRAM, USART 等。接着就是创建内核参数链表(Tagged list),创建完链表就是搬运事先烧写在NorFlash 中的内核映像和根文件系统映像到SDRAM,根据内核对BootLoader 的基本要求关闭中断,MMU 和数据Cache,并且配置r0=0, r1=0x0000 00fb 或者0x00000106(根据内核中linux/arch/arm/tools/mach-types[2]

    规定的机器编号),r2=0x20000100(BootLoader 传递给内核参数链表的物理地址),在ARM体系结构中,这个地址在同一种处理器的机器描述符(machine_desc)中都是默认的,所以在这里可以不指定。最后BootLoader 直接跳转到SDRAM 的内核处执行。

    3. 内核参数链表

    BootLoader 可以通过两种方法传递参数给内核, 一种是旧的参数结构方式(parameter_struct),主要是2.6 之前的内核使用的方式。另外一种就是现在的2.6 内核在用的参数链表 (tagged list) 方式。这些参数主要包括,系统的根设备标志,页面大小,内存的起始地址和大小,RAMDISK 的起始地址和大小,压缩的RAMDISK 根文件系统的起始地址和大小,内核命令参数等[3][4][5]。

    内核参数链表的格式和说明可以从内核源代码目录树中的 include/asm-arm/setup.h[2]中找到,参数链表必须以ATAG_CORE 开始,以ATAG_NONE 结束。这里的ATAG_CORE,ATAG_NONE 是各个参数的标记,本身是一个32 位值,例如:ATAG_CORE=0x54410001。

    其它的参数标记还包括: ATAG_MEM32 , ATAG_INITRD , ATAG_RAMDISK ,ATAG_COMDLINE 等。每个参数标记就代表一个参数结构体,由各个参数结构体构成了参数链表。参数结构体的定义如下:

    struct tag

    {

    struct tag_header HDR;

    union {

    struct tag_core core;

    struct tag_mem32 mem;

    struct tag_videotext videotext;

    struct tag_ramdisk ramdisk;

    struct tag_initrd initrd;

    struct tag_serialnr serialnr;

    struct tag_revision revision;

    struct tag_videolfb videolfb;

    struct tag_CMDline cmdline;

    struct tag_acorn acorn;

    struct tag_mEMClk mEMClk;

    } u;

    };

    参数结构体包括两个部分,一个是 tag_header 结构体,一个是u 联合体。

    tag_header 结构体的定义如下:

    struct tag_header

    {

    u32 size;

    u32 tag;

    };

    其中 size:表示整个tag 结构体的大小(用字的个数来表示,而不是字节的个数),等于tag_header 的大小加上u 联合体的大小,例如,参数结构体ATAG_CORE 的

    size=(sizeof(tag->tag_header)+sizeof(tag->u.core))>>2,一般通过函数 tag_size(struct * tag_xxx)来获得每个参数结构体的size。其中tag:表示整个tag 结构体的标记,如:ATAG_CORE等。

    联合体u 包括了所有可选择的内核参数类型,包括:tag_core, tag_mem32,tag_ramdisk等。参数结构体之间的遍历是通过函数tag_next(struct * tag)来实现的。本系统参数链表包括的结构体有: ATAG_CORE , ATAG_MEM, ATAG_RAMDISK, ATAG_INITRD32 ,ATAG_CMDLINE,ATAG_END。在整个参数链表中除了参数结构体ATAG_CORE 和ATAG_END 的位置固定以外,其他参数结构体的顺序是任意的。本BootLoader 所传递的参数链表如下:第一个内核参数结构体,标记为ATAG_CORE,参数类型为tag_core。每个参数类型的定义请参考源代码文件。

    tag_array 初始化为指向参数链表的第一个结构体的指针。

    tag_array->hdr.tag=ATAG_CORE;

    tag_array->hdr.size=tag_size(tag_core);

    tag_array->u.core.flags=1;

    tag_array->u.core.pagesize=4096;

    tag_array->u.core.rootdev=0x00100000;

    tag_array=tag_next(tag_array);

    tag_array->hdr.tag=ATAG_MEM;

    tag_array->hdr.size=tag_size(tag_mem32);

    tag_array->u.mem.size=0x04000000;

    tag_array->u.mem.start=0x20000000;

    tag_array=tag_next(tag_array);

    ……

    tag_array->hdr.tag=ATAG_NONE;

    tag_array->hdr.size=0;

    tag_array=tag_next(tag_array);

    最后将内核参数链表复制到内核默认的物理地址0x20000100 处。这样参数链表就建好了。

    4. 内核接收参数

    下面从基于ARM体系结构的zImage 映像启动来分析Linux 内核是怎样接收BootLoader传递过来的内核参数,zImage 启动过程如下图所示。

    (图有时间再画)

    在文件 arch/arm/boot/compressed/head.S[2]中 start 为zImage 的起始点,部分代码如下:

    start:

    mov r7, r1

    mov r8, r2

    …...

    mov r0, r4

    mov r3, r7

    bl decompress_kernel

    b call_kernel

    call_kernel:

    ……

    mov r0, #0

    mov r1, r7

    mov r2, r8

    mov pc, r4

    首先将BootLoader 传递过来的r1(机器编号)、r2(参数链表的物理地址)的值保存到r7、r8 中,再将r7 作为参数传递给解压函数decompress_kernel()。在解压函数中,再将r7 传递给全局变量__machine_arch_type。在跳到内核(vmlinux)入口之前再将r7,r8 还原到r1,r2 中。

    在文件 arch/arm/kernel/head.S[2]中,内核(vmlinux)入口的部分代码如下:

    stext:

    mrc p15, 0, r9, c0, c0

    bl __lookup_processor_type

    ………

    bl __lookup_machine_type

    首先从处理器内部特殊寄存器(CP15)中获得ARM 内核的类型,从处理器内核描述符(proc_info_list)表(__proc_info_begin—__proc_info_end)中查询有无此ARM 内核的类型,如果无就出错退出。处理器内核描述符定义在 include/asm-arm/procinfo.h[2]中,具体的函数实现在 arch/arm/mm/proc-xxx.S[2]中,在编译连接过程中将各种处理器内核描述符组合成表。接着从机器描述符(machine_desc)表(__mach_info_begin—__mach_info_end)中查询有无r1 寄存器指定的机器编号,如果没有就出错退出。机器编号mach_type_xxx 在arch/arm/tools/mach-types[2]文件中说明,每个机器描述符中包括一个唯一的机器编号,机器描述符的定义在 include/asm-arm/mach/arch.h[2]中,具体实现在 arch/arm/mach-xxxx[2]文件夹中,在编译连接过程中将基于同一种处理器的不同机器描述符组合成表。例如,基于AT91RM9200 处理器的各种机器描述符可以参考 arch/arm/mach-at91rm9200/board-xxx.c[2],机器编号为262 的机器描述符如下所示:

    MACHINE_START(AT91RM9200DK, "Atmel AT91RM9200-DK")

    /* Maintainer: SAN People/Atmel */

    .phys_io = AT91_BASE_SYS,

    .io_pg_offst = (AT91_VA_BASE_SYS >> 18) & 0xfffc,

    .boot_params = AT91_SDRAM_BASE + 0x100,

    .timer = &at91rm9200_timer,

    .map_io = dk_map_io,

    .init_IRQ = dk_init_irq,

    .init_machine = dk_board_init,

    MACHINE_END

    最后就是打开MMU,并跳转到 init/main.c[2]的start_kernel(初始化系统。在 init/main.c[2] 中,函数start_kernel()的部分代码如下:

    {

    ……

    setup_arch();

    ……

    }

    在 arch/arm/kernel/setup.c[2]中,函数setup_arch()的部分代码如下:

    {

    ……

    setup_processor();

    mdesc=setup_machine(machine_arch_type);

    ……

    parse_tags(tags);

    ……

    }

    setup_processor()函数从处理器内核描述符表中找到匹配的描述符,并初始化一些处理器变量。setup_machine()用机器编号(在解压函数decompress_kernel 中被赋值)作为参数返回机器描述符。从机器描述符中获得内核参数的物理地址,赋值给tags 变量。然后调用parse_tags()函数分析内核参数链表,把各个参数值传递给全局变量。这样内核就收到了BootLoader 传递的参数。

    5. 参数传递的验证和测试

    参数传递的结果可以通过内核启动的打印信息来验证。

    Machine: Atmel AT91RM9200-DK

    ……

    Kernel command line: console=ttyS0,115200 root=/dev/ram rw init=/linuxrc

    ……

    Memory: 64MB = 64MB total

    ……

    checking if image is initramfs...it isn''t (no cpio magIC); looks like an initrd

    Freeing initrd memory: 1024K

    ……

    RAMDISK: Compressed image found at bLOCk 0

    一个完备的BootLoader 是一个很复杂的工程,本文所介绍的只是嵌入式系统的BootLoaer 基本功能。任何一个BootLoader 都离不开这个基本功能,内核只有接收这些参数才能正确地启动,同时也为内核的移植和调试奠定了良好的基础。


关键字:BootLoader  内核  参数传递 引用地址:BootLoader与内核之间参数传递

上一篇:从单片机的应用发展到嵌入式操作系统的应用思考
下一篇:ARM伪指令地址读取:ADR ADRL LDR

推荐阅读最新更新时间:2024-03-16 15:55

ARM下的参数传递
之前在学习如何在C语言中嵌入汇编时有了解到C语言之前的参数调用是使用寄存器R0传递第一个参数,R1传递到 第二个..一直到R3传递第四个参数.但是实际上有时可能传递的参数非常多,超过8个,或是参数中有浮点数之类, 参数也会超过4个寄存器,对于超出的部份并不使用R4,而是使用堆栈的方式 华丽的分割线 对于ARM体系来说,不同语言撰写的函数之间相互调用(mix calls)遵循的是 ATPCS(ARM-Thumb Procedure Call Standard),ATPCS主要是定义了函数呼叫时参数的传递规则以及如何从函数返回,关于ATPCS的详细内容 可以查看ADS1.2 Online Boo
[单片机]
STM8 Bootloader和APP设计(都可以使用中断)
1、本Bootloader适用范围: STM8S 和STM8A 2、Bootloader和APP的标志位 在EEPROM中的0x4000地址存放一个标志位 Bootloader 的标志位: 0x11; APP的标志位: 0x39; 3、Bootloader 和APP的地址分配 Bootloader 的存储空间为: 0x8000~ 0x9FFF; APP 的存储空间为: 0xA000~0x27FFF; 在0x8000开始的前128个字节,放置着bootloader重定位过的向量表(bootloader的真正的向量表放置在另外的地方) 在0xA000开始的前128个字节,放置着application的
[单片机]
STM8 <font color='red'>Bootloader</font>和APP设计(都可以使用中断)
瑞昱半导体获得多项MIPS处理器内核授权
为数字消费、家庭网络、无线、通讯和商业应用提供业界标准处理器架构与内核的领导厂商美普思科技公司(MIPS Technologies, Inc)宣布,全球领先的 IC 设计公司瑞昱半导体公司(Realtek Semiconductor Corp.)已取得多项 MIPS32TM 处理器内核授权,再次肯定了对 MIPSTM 架构的长期承诺。瑞昱半导体将采用这些内核开发针对宽带、网络、数字家庭及其他多媒体应用的新一代 SoC。 该授权协议包括紧凑型 4KEcTM Pro 内核、高性能24KTM Pro 系列、多线程 34KTM Pro 系列、超标量 74KTM 系列、1004K™ 一致处理系统(Coherent Processi
[嵌入式]
S5PV210开发系列三_简易Bootloader的实现
Bootloader是嵌入式系统上电后第一段运行的代码。对于功能简单的处理器,可能并没有Bootloader的概念,但对于应用处理器,有不同的启动方式。不同的存储设备(Nand flash、sd/mmc、DDR2、SRAM等)。不同的操作系统等,往往须要一个Bootloader先初始化CPU和相关的硬件,建立内存空间映射,把内核或应用程序载入到对应的内存运行位置。最后调用内核或应用程序,释放CPU控制权,完毕整个Bootloader的流程。 笔者此处就S5PV210的Bootloader实现作一个简单的介绍。 1. Bootloader流程 Bootloader是严重依赖于详细硬件实现的,同样CPU内核架构。不同厂商生产的处理器其
[单片机]
S5PV210开发系列三_简易<font color='red'>Bootloader</font>的实现
STM8 自带 BootLoader 串口烧录程序
一、进入BootLoader模式 根据STM8的资料可以知道,进入BootLoader的方法只有两种: 空芯片(Flash首地址内容不是0x82或0xAC),上电后即可进入BootLoader模式。 OPTION配置参数地址 487Eh = 0x55 , 787Fh = 0xAA,复位后会进入BootLoader模式。 二、写入OPTION配置参数 通过Flash写入的方式将地址 487Eh 和 787Fh写入0x55和0xAA即可,通过MCU执行一次Flash写入。 1、使用库函数 此方法只适用于可以使用库函数的MCU,记得引用库的头文件 stm8s_flash.h,若MCU不能使用库函数可以使用寄存器方法。 /
[单片机]
STM8 自带 <font color='red'>BootLoader</font> 串口烧录程序
STM32复位来源 以及系统和内核复位区别
1STM32的复位和时钟控制 RCC:Reset and Clock Control 每一块STM32中都有这么一个RCC复位和时钟控制模块。 STM32的复位为三类:系统复位、电源复位和后备域复位。 系统复位: 1. NRST引脚上的低电平(外部复位) 2. 窗口看门狗计数终止(WWDG复位) 3. 独立看门狗计数终止(IWDG复位) 4. 软件复位(SW复位) 5. 低功耗管理复位 电源复位: 1. 上电/掉电复位(POR/PDR复位) 2. 从待机模式中返回 备份区域复位: 1. 软件复位,备份区域复位可由设置备份域控制寄存器(RCC_BDCR)中的BDRST位产生。 2. 在VDD和VBAT两者掉电的前提下, VDD或VB
[单片机]
STM32复位来源 以及系统和<font color='red'>内核</font>复位区别
linux内核移植s3c2410,准备工作,继续
vmlinux在链接时,使用的文件是在vmlinux-all中定义的所有文件,链接俄脚本是在arch/arm/kernel/vmlinux.lds中定义该如何 链接这些文件。 总结上面: 真个的 linux的源码是通过Makefile来实现项目的 管理的,顶层的Makefile定义了那些文件夹 中的 内容 被编译进内核,各级 的子目录下的Makefile决定该目录下的那些文件被编译进内核。 对于编译选项的设置的话,存在全局的,局部的和仅对一个文件 起作用的选项。 3.内核的Kconfig文件的 分析 在linux内核目录下,如果直接使用make menuconfig的话,出现的配置界面就是使用的是Kbuild来实现的。最终通过ma
[单片机]
Nucleus RTOS支持新Diamond CPU内核
Mentor Graphics公司所属Accelerated Technology公司的Nucleus RTOS现已支持Tensilica公司所有6个Diamond Standard处理器核心。 对处理器核心的支持,使嵌入系统开发人员可以在非专用软件环境下使用Diamond Standard处理器,以支持包括消费电子和网络管理系统的广泛应用。
[嵌入式]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved