xloader
[1],先看一下整个系统的结构(软件是灵魂,硬件是驱体,再强大的灵魂力若没有躯体终将是游魂野鬼,再强壮的驱体若没有灵魂终将是植物人)
结构 | 作用 | 备注 |
硬件 | 一切软件的载体 | |
xloader | 引导uboot | |
uboot | 启动加载或下载linux kernel | |
linux kernel | OS,承载android | |
android | OS,承载APP |
[2],为什么需要xloader?
cpu上电后会自动加载一小段程序到内部ram中运行,内部的ram资源很小,一般只有几十k的空间,比如我现在用的cpu就只有32k的空间。uboot功能很强,具有初始化,交互操作,下载和引导linux的作用,因此体积上会超过cpu内部的ram大小,所以我们需要更小的xloader来为我们加载引导uboot。xloader一般只作一些最最核心的硬件初始化,比如cpu时钟,外部内存,flash和usb/SD/uart的初始化工作,然后就加载uboot,把更深入的初始就交给uboot来执行。
[3],xloader执行流程
[首先],看x-loader.lds文件,通过以下命令得到lds文件的路径
$ find -name "*.lds"
打开x-loader.lds如下:
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//指定输出可执行文件是elf格式,32位ARM指令,小端
OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
//指定输出可执行文件的平台为ARM
OUTPUT_ARCH(arm)
//指定输出可执行文件的起始代码段为_start
ENTRY(_start)
SECTIONS
{
//从0x0位置开始,但会在vim board/copener_pad_ref/config.mk 文件里面定义为TEXT_BASE = 0x07041000
. = 0x00000000; //代码以4字节对齐
. = ALIGN();
//指定代码段
.text :
{
cpu/copener/start.o (.text) //代码的第一个代码部分
*(.text) //其它代码部分
} . = ALIGN(); //指定只读数据段
.rodata : { *(.rodata) } . = ALIGN(); //指定可读写数据段
.data : { *(.data) } . = ALIGN(); //指定got段, got段式是uboot自定义的一个段, 非标准段
.got : { *(.got) } . = ALIGN(); //把__bss_start赋值为当前位置,即bss段的开始位置
__bss_start = .; //指定bss段
.bss : { *(.bss) } //把_end赋值为当前位置,即bss段的结束位置
_end = .;
}
由上面可知程序的入口为“cpu/copener/start.S”文件,且入口函数为“_start”,开头已经有了,接下来就是怎么跟踪代码了。
[然后],跟踪start.S文件:
start.S是汇编写的文件,主要是执行了下面的操作
// set the cpu to SVC32 mode
// disable MMU
// Enable all domains to client mode
// Invalidate instruction cache
// Invalidate data cache
// Invalidate entire Unified main TLB ldr pc, _start_armboot /* jump to C code */
_start_armboot: .word start_armboot
start.S的最后通过“ldr pc, _start_armboot”跳入C代码中运行了。通过下面的命令把start_armboot函数所在文件lib/board.c找出。
$ grep -rnws start_armboot
打开board.c,该函数简化后的操作如下
void start_armboot (void)
{
board_init(); //pad_ref.c 空函数
uart_init(CFG_UART_BAUD_RATE); //uart.c 初始化串口
cpu_init(); //cpu.c 设置一下cpu的频率
ddr_init(); //ddr_init.c 初始化ddr
mem_test(); //board.c 内存测试 for(idx = ;idx < ;) {
sram_dat = readl(SRAM_BASE_ADDR + idx);
writel(sram_dat,DDR_DATA_BASE + idx);
idx = idx + ;
} if (check_romloader_fastboot()) { //检测引导设置,若符合usb引导则fastboot USB
TRACE(KERN_INFO, "enter fastboot USB boot\n");
usb_boot(0x02); //进入USB引导操作
} switch(readl(CFG_BOOT_MODE) & 0x3){ //检测其它引导模式进入相应操作
case 0x00:
TRACE(KERN_INFO,"SDIO\n");
sdmmc_continue_boot(0x0); case 0x02:
TRACE(KERN_INFO,"USB\n");
usb_boot(0x02);
break; case 0x03:
TRACE(KERN_UART,"UART\n");
uart_boot(0x03);
break;
default:
TRACE(KERN_ERROR,"No Boot Source\n");
break;
}
TRACE(KERN_ERROR,"No U-boot found\n");
while();
}
而后面的usb_boot(0x02)又会调用handle_fastboot(rxdata, rxsize, boot_mode),进而调用enter_entry(head.entry, boot_mode),最后调用(*(void(*)())buf)(boot_mode),然后就跳进uboot里面去运行了。
(*(void(*)())buf)(boot_mode)是一个函数指针,类型为void(*)(),指向的是buf地址,即head.entry,涉及到的结构体如下:
typedef struct {
char tag[]; //NUFX
unsigned int offset; //
unsigned int entry; //where to place
unsigned int size; //size of binary
unsigned int loader_cksum; //chsum of binary
unsigned int header_cksum; //cksum of first 16 bytes of header
}xl_header;
extern xl_header head;
由config.mk文件指定TEXT_BASE = 0x07041000,然后通过偏移算出Head.entry = 0x7041008。即xloader最后要跳转的地址为0x7041008。
[4]xloader的编译
//清除上一次编译
$ make distclean //配置当前编译
$ make copener_pad_ref_config //编译
$ make //////////////////////////////////////生成文件时打印的信息//////////////////////////////////////
Generate x-load.img from x-load
ELF Entry = 0x7041008
ELF Program number =
ELF Program Header Offset =
Program[]: offset=0x200, Size=0x74f8
Program[]: offset=0x61656100, Size=0x412d3705
Head.tag = NUFX
Head.offset = 0x200
Head.entry = 0x7041008
Head.size = 0x74fc
Head.loader_cksum = 0x55157080
Head.header_cksum = 0xb4604cd2
生成的文件为:
oee@copener:/opt/ns115_jb/bsp/xloader$ ls -lh
-rwxrwxr-x oee oee 30K 5月 : x-load.bin
-rw-rw-r-- oee oee 30K 5月 : x-load.img
xloader的更多相关文章
- Pandaboard ES编译bootloader、xloader、内核、以及安卓系统
Building bootloader and kernel Bootloader Startwith building the bootloader. To learn more about the ...
- 嵌入式Linux引导过程之1.5——从BootRom到Xloader
在开始看Xloader_Entry的代码之前,我想先总结一下从芯片上电到开始运行Xloader的代码的过程,这是我目前理解的一个过程,可能有所出入,待以后继续完善. 当 系统上电之后,首先会将PC寄存 ...
- 嵌入式Linux引导过程之1.6——Xloader的Xloader_Entry
我们已经看完了XLOADER_ENTRY里调用的前两个标号的代码,分别是sys_init和ddr_init.对于一个嵌入式系统来说,这两 个部分的代码是在一开始就执行的,至少是在从bootrom里面的 ...
- 嵌入式Linux引导过程之1.4——Xloader的ddr_init
这里我们来看XLOADER_ENTRY中调用的第二个标号ddr_init处的代码,这部分代码的作用是对外部内存SDRAM进行初始化,在我 spearplus开发板中,使用的是DDR SDRAM.在调用 ...
- 嵌入式Linux引导过程之1.3——Xloader的sys_init
上一篇文章对XLOADER_ENTRY进行了分析,看到其中调用的第一个标号就是sys_init,本文就对这个标号对应的代码段进行粗略的分析,这里我也还有好多没有搞明白的,就先留着,日后慢慢明白,先把自 ...
- 嵌入式Linux引导过程之1.1——Xloader的xloader.lds
本文中的所有代码版本都是基于ST的SpearPlus开发板的. xloader是在系统上电之后,执行完ROM中的frimware后最先开始执行的用户程序,它的体积很小,执行的功能也很简单,主要是对系统 ...
- 嵌入式Linux引导过程之1.2——Xloader的XLOADER_ENTRY
根据上文中获得的线索,本文分析init.S中的XLOADER_ENTRY. 在init.S中,定义了好多与平台相关的寄存器地址宏以及好多其他函数,我们在用到的时候再回过头来分析,这里,我们只看其中的一 ...
- Android刷机教程
我的机器是Nexus 5 一. 安装驱动 如何进入fastboot模式 1. 拔掉数据线,将手机关机 2. 关机后同时按住[音量减小键]和[开关机键]即可进入Fastboot模式 开启usb调试 - ...
- ns115 step by step
一,安装环境: sudo apt-get install git-core gnupg flex bison gperf build-essential zip curl zlib1g-dev lib ...
随机推荐
- Python 中的数据结构总结(一)
Python 中的数据结构 “数据结构”这个词大家肯定都不陌生,高级程序语言有两个核心,一个是算法,另一个就是数据结构.不管是c语言系列中的数组.链表.树和图,还是java中的各种map,随便抽出一个 ...
- [DNS][转]EDNS
随着业务的复杂化和多样化,RFC1035中定义的DNS消息格式和它支持的消息内容已经不足以满足一些DNS服务器的需求,于是,RFC2671 中提出了一种扩展DNS机制EDNS(Extension Me ...
- MVC中使用js拼接的元素验证问题
MVC使用的验证方式本质上是jquery.validation js 拼接的格式如下: <input class='required' title='请输入项目名称!' id='Project ...
- putty连接报NetWork error:connection refused
首先通过物理终端进入到linux上,手工检查ssh发现没运行 /etc/init.d/sshd status 使用rpm -V 命令可检查到ssh的软件包正常 /rpm -V openssh-serv ...
- 安全协议系列(三)----CCMP与WPA-PSK
本节讨论 CCM 在 WiFi 中的实际应用 -- CCMP 协议 根据 RFC 3610,完成 CCMP 报文的加解密,需要提供:分组密钥(K).随机数(Nonce).附加认证数据(AAD),这三个 ...
- 可拖拽的ListView
今天研究了一下可拖拽的ListView ,当ListView的数据不能充满整个模拟器的时候,系统自带的listview上拖下拽都不好使,显得很单调, 为了实现上拖下拽时也能实现滚动的效果,这时候就需要 ...
- 使用NFC读卡器ACR122u读取银行卡信息
使用NFC读卡器 ACR122u 使用ACR122u Tool 软件发送spdu指令即可 一些运行结果 如上所示,银行卡的卡号,有效期,卡主姓名,身份证号,进期消费记录均可读出
- C++线程中packaged_tack
packaged_tack<>函数是一个你可以先定义好任务的类型,但是不想马上启动.函数可以在你想要启动任务是启动,你只需要调用你声明的函数就可以. #include <future ...
- NUBWO/狼博旺 NO-3000台式电脑耳机头戴式游戏电竞语音耳麦带话筒
产品名称:NUBWO/狼博旺 NO-3000 套餐类型: 官方标配 颜色分类: 蓝色耳机(发光) 黑蓝7.1声道USB 黑色 黑蓝色(发光) 黑蓝色 黑蓝单插孔 白色(发光) 白色 佩戴方式: 头戴护 ...
- 电量/信号的显示 imageLevel
通过imageLevel可以对1个imageView在不同的level值显示不同的图片,通常用于显示电量,wifi信号,蓝牙信号等信息: 1.编写xml ,在drawable目录 <level- ...