mini-uboot 启动过程简单分析

单片机有最小系统,所谓最小系统,就是单片机能正常工作所需要的最少外设。对于Uboot来说,同样有个最小系统,因为Uboot最主要的功能就是引导内核。下面我们通过一个简单的Mini-Uboot来分析Uboot的启动加载过程。(只是分析过程,此Uboot具有引导内核功能)

注:这个uboot 只是具有基本的内核引导功能,只是作为前期简单的学习使用,入门而已,并不是正常的uboot 启动流程

       具体uboot (u-boot-2013.01)启动过程移步Exynos4412 Uboot 移植(二)—— Uboot 启动流程分析


下面是mini-uboot 的根目录树状图:

     我们拿到一个工程,想了解它的功能,最方便的就是读它的makefile。

一、Makefile

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. sinclude include/config.mk  
  2.   
  3. #ARCH=arm  
  4. #CPU=arm920t  
  5. #VENDOR=samsung  
  6. #SOC=s3c2410  
  7. #BOARD=smdk2410  
  8.   
  9. SRC_TREE:=$(shell pwd)  
  10. MKCONFIG=$(SRC_TREE)/mkconfig    
  11.   
  12. INCLUDE_PATH=include  
  13. DRIVER_PATH=driver  
  14. LIB_DIR=lib  
  15.   
  16.   
  17. CFLAG=-mabi=apcs-gnu -fno-builtin  -fno-builtin-function -g -O0 -c  -I$(INCLUDE_PATH) -I$(DRIVER_PATH) -o   
  18. LDFLAG=-Tcpu/arm/arm_cortexa8/map.lds  -o  
  19. OBJS= cpu/$(ARCH)/$(CPU)/start.o  
  20. OBJS+=lib_arm/board.o  
  21. OBJS+=board/$(VENDOR)/$(BOARD)/lowlevel_init.o  
  22. OBJS+=board/$(VENDOR)/$(BOARD)/mem_setup.o  
  23. OBJS+=board/$(VENDOR)/$(BOARD)/nand.o  
  24. OBJS+=driver/uart.o  
  25. OBJS+=lib/string.o  
  26. OBJS+=common/do_go.o  
  27. OBJS+=common/main.o  
  28.   
  29. ifeq ($(ARCH), arm)   
  30. CROSS_COMPILE=arm-cortex_a8-linux-gnueabi-  
  31. endif  
  32.   
  33. PROJ_NAME=mini_uboot  
  34.       
  35. all: $(OBJS)   
  36.     $(CROSS_COMPILE)ld $(OBJS) $(LDFLAG) $(PROJ_NAME).elf                
  37.     $(CROSS_COMPILE)objcopy -O binary $(PROJ_NAME).elf $(PROJ_NAME).bin   
  38.     $(CROSS_COMPILE)objdump -D $(PROJ_NAME).elf  > $(PROJ_NAME).dis  
  39.     cp *.bin /tftpboot  
  40.   
  41. %.o: %.S  
  42.     $(CROSS_COMPILE)gcc $(CFLAG) $@ $<       
  43. %.o: %.s  
  44.     $(CROSS_COMPILE)gcc $(CFLAG) $@ $<       
  45.   
  46. %.o: %.c  
  47.     $(CROSS_COMPILE)gcc $(CFLAG) $@ $<       
  48.   
  49. fsc100_config:          #    ARCH  CPU       VENDOR   BOARD  SOC  
  50.     $(MKCONFIG) $(@:_config=) arm arm_cortexa8 samsung fsc100 s5pc100  
  51.     #mkconfig  fsc100 arm arm_cortexa8 samsung fsc100 s5pc100  
  52.   
  53. smdk2410_config:            #    ARCH  CPU       VENDOR   BOARD  SOC  
  54.     $(MKCONFIG) $(@:_config=) arm arm920t samsung smdk2410 s3c2410  
  55.   
  56. clean:  
  57.     @rm -rf $(OBJS) *.bin *.elf config.mk  

这里以2440为例,咱们来分析:

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. #ARCH=arm  
  2. #CPU=arm920t  
  3. #VENDOR=samsung  
  4. #SOC=s3c2410  
  5. #BOARD=smdk2410  

架构为arm,CPU为arm920t,生产商 samsung,片上系统sc2410,板子为smdk2410。

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. OBJS= cpu/$(ARCH)/$(CPU)/start.o  
  2. OBJS+=lib_arm/board.o  
  3. OBJS+=board/$(VENDOR)/$(BOARD)/lowlevel_init.o  
  4. OBJS+=board/$(VENDOR)/$(BOARD)/mem_setup.o  
  5. OBJS+=board/$(VENDOR)/$(BOARD)/nand.o  
  6. OBJS+=driver/uart.o  
  7. OBJS+=lib/string.o  
  8. OBJS+=common/do_go.o  
  9. OBJS+=common/main.o  

OBJS为依赖文件,生成的.o文件。

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. ifeq ($(ARCH), arm)   
  2. CROSS_COMPILE=arm-cortex_a8-linux-gnueabi-  
  3. endif  

根据相应的架构,制作相应的交叉编译工具。

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. all: $(OBJS)   
  2.     $(CROSS_COMPILE)ld $(OBJS) $(LDFLAG) $(PROJ_NAME).elf                
  3.     $(CROSS_COMPILE)objcopy -O binary $(PROJ_NAME).elf $(PROJ_NAME).bin   
  4.     $(CROSS_COMPILE)objdump -D $(PROJ_NAME).elf  > $(PROJ_NAME).dis  

第一步:连接 ;第二步:格式转换;第三步:反汇编 " >" 为重定向的意思;

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. %.o: %.S  
  2.     $(CROSS_COMPILE)gcc $(CFLAG) $@ $<       
  3. %.o: %.s  
  4.     $(CROSS_COMPILE)gcc $(CFLAG) $@ $<       
  5.   
  6. %.o: %.c  
  7.     $(CROSS_COMPILE)gcc $(CFLAG) $@ $<       

将所有的.S 文件、.s文件、.c文件编译成.o文件。

注意:.S文件可以在编译过程接受参数,.s文件不可以。

 

二、链接文件

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. OUTPUT_FORMAT("elf32-littlearm""elf32-littlearm""elf32-littlearm")  
  2. /*OUTPUT_FORMAT("elf32-arm", "elf32-arm", "elf32-arm")*/  
  3. OUTPUT_ARCH(arm)  
  4. ENTRY(_start)  //指定入口地址  
  5. SECTIONS    //段信息  
  6. {  
  7.     /* . */   
  8.     . = 0x22000000; //elf文件的入口地址  
  9.     . = ALIGN(4);   //指定四字节对齐  
  10.     .text      :    //代码段  
  11.     {  
  12.         cpu/arm/arm_cortexa8/start.o(.text) //确保执行的第一段代码是start.o  
  13.         *(.text)        //所有代码段融合在一起  
  14.     }  
  15.     . = ALIGN(4);  
  16.     .rodata :   //只读数据段  
  17.     { *(.rodata) }  //所有数据段  
  18.     . = ALIGN(4);  
  19.     .data :     //数据段  
  20.     { *(.data) }  
  21.     . = ALIGN(4);  
  22.   
  23.     _start_bss = .; //bss段开始地址  
  24.     .bss :  
  25.      { *(.bss) }  
  26.     _end_bss = .;       //bss段结束地址,两者可确定bss段大小  
  27. }  


三、start.s文件(Uboot执行的第一个文件)

[cpp] view plaincopy
在CODE上查看代码片派生到我的代码片
  1. @ 汇编中的宏  
  2. .equ USER_MODE, 0x10    @define USER_MODE 0x10  
  3. .equ IRQ_MODE,  0x12  
  4. .equ SVC_MODE,  0x13  
  5. .equ MODE_MASK, 0x1f  
  6.   
  7. .section .text  
  8. .global _start  
  9.   
  10.     @ 不支持异常处理的,这里只写了复位异常处理  
  11. _start:  
  12. vector:  
  13.     b  reset_handler  
  14.     nop     @undef ......  
  15.     nop  
  16.     nop  
  17.     nop  
  18.     nop  
  19.     nop  
  20.     nop  
  21.   
  22. reset_handler:  
  23.     @step 1: svc close irq fiq      //第一步:将运行模式改成SVC模式    
  24.     mrs r0, cpsr            //修改cpsr模式位  
  25.     bic r0, r0, #0x1f  
  26.     orr r0, r0, #0xc0  @IRQ FIQ     //关闭IRQ FIQ  
  27.     msr cpsr_c, r0  
  28.   
  29.     @step 2: cache 关闭I CACHE D CACHE    //第二步:关闭cache,直接运行,不需缓存  
  30.     mrc p15, 0, r0, c12, c0, 0  
  31.     bic r0, #0x1000  
  32.     bic r0, #0x2  
  33.     mcr p15, 0, r0, c12, c0, 0  
  34.       
  35.     @step 3:                //第三步:调用电路板级初始化程序, system clock , dram, watchdog  
  36.     @bl low_level_init      //初始化时钟、dram、关闭看门狗  
  37.   
  38.     @step 4: sp-> 0x30000000     //第四步:设置栈指针,使其指向一个地址即可  
  39.     ldr sp, =0x2e000000  
  40.   
  41.     @step 5: mini_uboot.bin  > 16KB   bin < 16KB   
  42.     @step 5 代码自搬移  
  43.     @copy_miniuboot_rto_sdram 如果你的代码大于了16KB代码需要实现自我搬移  
  44.   
  45.     @step 6:                //第六步:清除BSS段,BSS段大小由链接文件里确定   
  46.     @STEP 6.1 , 清除 BSS段  
  47.     @  
  48. clear_bss:  
  49.     ldr r0, =_start_bss   @| BSS 起始地址  
  50.     ldr r1, =_end_bss     @| BSS 终止地址  
  51.     mov r2, #0  
  52. bss_loop:  
  53.     cmp r0, r1  
  54.     strne r2, [r0], #4  
  55.     bne bss_loop  
  56.   
  57.     @step  7, 进入C           //跳转到C程序入口  
  58.     b start_armboot  
  59. stop:  
  60.     b stop  
  61.   
  62.     .end  

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/402063.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySql中的varchar类型

2019独角兽企业重金招聘Python工程师标准>>> MySql中的varchar类型&#xff08;转&#xff09; 今天新开始的项目在做数据库设计&#xff0c;发现自己对MySql的varchar类型还不熟悉&#xff0c;故又上网收集资料整理如下。 1.varchar类型的变化 MySQL 数据库的va…

GDB

用GDB调试程序&#xff08;一&#xff09;

Exynos4412 Uboot 的使用与烧写

Uboot版本&#xff1a;u-boot-2013.01 开发板&#xff1a;Exynos4412 本文主要是熟悉U-boot 的使用以及如何将Uboot 烧入Exynos4412。当然在这之前首先必须保证开发板上已经有了U-boot。如果没有&#xff0c;请先烧入一个已经编译好的Uboot.bin&#xff0c;可以利用JTAG或DNW。…

进入保护模式(三)内存的分页

2019独角兽企业重金招聘Python工程师标准>>> 一、分页 先入为主理解的分页和系统中内存内应的分页 1.弄过数据库数据查找展示&#xff0c;有个数据分页展示的概念&#xff0c;这个的目的是为了速度、和展示效果上的提升 2.内存的分页又是怎么回事哪&#xff1f; …

Python2 与 Python3 的常见区别

Python2 与 Python3 的常见区别

关于gcc、glibc和binutils模块之间的关系

一、关于gcc、glibc和binutils模块之间的关系 1、gcc&#xff08;gnu collect compiler&#xff09;是一组编译工具的总称。它主要完成的工作任务是“预处理”和“编译”&#xff0c;以及提供了与编译器紧密相关的运行库的支持&#xff0c;如libgcc_s.so、libstdc.so等。 2、bi…

C#实现简单的 Ping 的功能,用于测试网络是否已经联通

1 /// <summary>2 /// 是否能 Ping 通指定的主机3 /// </summary>4 /// <param name"ip">ip 地址或主机名或域名</param>5 /// <returns>true 通&#xff0c;false 不通</returns>6 …

Binutils工具集 解析

对于嵌入式系统开发&#xff0c;掌握相应的工具至关重要&#xff0c;它能使我们解决问题的效率大大提高。目前&#xff0c;可以说嵌入式系统的开发工具是GNU的天下&#xff0c;因为来自GNU的GCC编译器支持大量的目标处理器。除了GCC&#xff0c;还有一个非常重要的、同样来自于…

Log4net核心组成

一、Logger(日志)1&#xff0e; 记录日志的分类&#xff1a;Log4net能够以多种方式输出日志。支持的日志输出常用的主要媒介有数据库&#xff08;包括MS SQL Server, Access, Oracle9i,Oracle8i,DB2,SQLite&#xff0c;控制台&#xff0c;文件&#xff0c;事件日志(可以用事件查…

ARM 软中断指令SWI

前面我们学习ARM工作模式中&#xff0c;处理器模式切换可以通过软件控制进行切换&#xff0c;即修改CPSR模式位&#xff0c;但这是在特权模式下&#xff0c;当我们处于用户模式下&#xff0c;是没有权限实现模式转换的。若想实现模式切换&#xff0c;只能由另一种方法来实现&am…

iops 条带深度 队列深度 NCQ

iops 条带深度 队列深度 NCQIOPS 即I/O per second&#xff0c;即每秒进行读写&#xff08;I/O&#xff09;操作的次数&#xff0c;多用于数据库等场合&#xff0c;衡量随机访问的性能。并发IO的概念&#xff1a;并发IO&#xff0c;指多个IO可以同时被处理&#xff0c;比如IO1…

yum install 失败

https://blog.csdn.net/weixin_45621658/article/details/110734514 原因&#xff1a;centos6的默认源在2012年-12月左右被官方搞掉了 下列是错误详情 Bash [rootc8-20 ~]# yum makecache Loaded plugins: fastestmirror Loading mirror speeds from cached hostfile YumRepo …

ARM 汇编复习题

一、选择题 1. ARM属于&#xff08;A&#xff09; [A] RISC架构  [B] CISC架构 2. ARM指令集是&#xff08;C &#xff09;位宽&#xff0c;Thumb指令集是&#xff08;B &#xff09;位宽的。 [A] 8位  [B] 16位 [C] 32位 [D] 64位 3. ARM指令集是&#xff…

ARM处理器工作模式

一、ARM体系的CPU有以下7种工作模式&#xff1a; 1、用户模式&#xff08;usr&#xff09;&#xff1a;正常的程序执行状态 2、快速中断模式&#xff08;fiq&#xff09;&#xff1a;用于支持高速数据传输或通道处理 3、中断模式&#xff08;irq&#xff09;&#xff1a;用于普…

sphinx

2019独角兽企业重金招聘Python工程师标准>>> ./configure --prefix/usr/local/sphinx --with-mysql/usr/local/mysql make && make install cd /usr/local/sphinx/etc cp sphinx.conf.dist sphinx.conf vim sphinx.conf mysql -u test < /usr/local/sphi…

Exynos4412裸机开发综合练习

下面是一个案例需求&#xff1a; 1、编写一段程序,该程序的主要功能是监控电路板上的电压值,若电压值超过当前的电压限制则通过蜂鸣器报警,通过按键解除报警; 2、其具体要求如下; a) 程序下载20s后,进入电压采集状态(使用RTC ALARM功能完成), 要求1s采集1次电路板电压值;(采用…

Exynos4412 裸机开发 —— IIC总线

前言&#xff1a; I2C(Inter-Integrated Circuit)总线(也称 IIC 或 I2C) 是有PHILIPS公司开发的两线式串行总线&#xff0c;用于连接微控制器及外围设备&#xff0c;是微电子通信控制领域广泛采用的一种总线标准。它是同步通信的一种特殊形式&#xff0c;具有接口线少、控制方式…

Exynos4412裸机开发 —— A/D转换器

一、Exynos4412 A/D转换器概述 1、简述 10位或12位CMOS再循环式模拟数字转换器&#xff0c;它具有10通道输入&#xff0c;并可将模拟量转换至10位或12位二进制数。5Mhz A/D 转换时钟时&#xff0c;最大1Msps的转换速度。A/D转换具备片上采样保持功能&#xff0c;同时也支持待机…

Exynos4412裸机开发 —— UART

一、Exynos4412 UART 的特性 Exynos4412 中UART&#xff0c;有4 个独立的通道&#xff0c;每个通道都可以工作于中断模式或DMA 模式&#xff0c;即 UART 可以发出中断或 DMA 请求以便在UART 、CPU 间传输数据。UART 由波特率发生器、发送器、接收器和控制逻辑组成。 使用系统时…

Exynos4412裸机开发 —— 看门狗定时器

一、看门狗定时器概述 看门狗&#xff08;WatchDog Timer) 定时器和PWM的定时功能目的不一样。它的特点是&#xff0c;需要不同的接收信号&#xff08;一些外置看门狗芯片&#xff09;或重新设置计数器&#xff0c;保持计数值不为0。一旦一些时间接收不到信号&#xff0c;或计数…