基于Nandflash的Bootloader的设计与实现

news2025/1/19 23:08:01

摘要:Bootloader是系统上电或复位后首先运行的一段代码,是连接操作系统和硬件的桥梁,负责初始化硬件和引导操作系统等。目前已有很多通用的Bootloader,但是如何根据特定的嵌入式平台,移植自己的引导程序是一个重点和难点。文章详细说明了从Nandflash引导操作系统要完成的主要任务和实现方法,并给出了在S3C2410上实现Nandflash启动的试验结果。
关键词:Bootloader; 移植;NandflashS3C2410

0
引言
  Bootloader通常称为系统的引导加载程序,是系统加电或复位后执行的第一段代码[ 1 ]。一般它只在系统启动时运行非常短的时间,但对于嵌入式系统来说,这是一个非常重要的系统组成部分。通过这段小程序,可以初始化硬件设备、建立内存空间的映射图,从而将系统的软硬件环境带到一个合适的状态,以便为调用操作系统内核准备好正确的环境,并同时提供基本输入、输出系统监控功能和程序调试功能。

  Bootloader是严重地依赖于硬件而实现的。每种不同体系结构的处理器都有不同的Bootloader。除了依赖于处理器的体系结构以外,Bootloader实际上也依赖于具体的嵌入式板级设备的配置,也就是说,对于两块不同的嵌入式板而言,即使它们是基于同一种处理器而构建的,要想让运行在一块板子上的Bootloader程序也能运行在另一块板子上,通常也都需要修改与目标硬件相关的代码。因此有必要分析Bootloader,并理解和找出其中的原理和规律,就其特定的嵌入式系统,移植或开发属于自己的Bootloader

1 系统硬件平台简介
  本系统采用的是SamSung公司的S3C2410处理器[ 2 ],它是专门为移动手持设备提供的高性价比和高性能的嵌入式微处理器解决方案。其内核是ARM920T,最高能工作在202.8MHz,为了减少系统总成本和减少外围器件,它集成了如下部件:分别为16KB指令和数据Cahce1LCD控制器、SDRAM控制器、NANDFLASH控制器、3通道UART4通道DMA4个具有PWM功能的计时器和1个内部时钟、8通道10ADC、触摸屏接口、I²S总线接口,2USB主机接口、1USB设备接口,2SPI接口、SDMMC卡接口、看门狗定时器、117位通用IO口、24位外部中断源、8通道10AD控制器等。本文涉及的S3C2410开发板的硬件结构如图1所示,本文主要阐述从Nandflash引导操作系统要完成的主要任务和实现方法,至于从Norflash引导操作系统,不打算具体实现。

2 存储空间分布和映射图
  硬件平台的Nandflash(型号是:K9F1208U0M[ 3])空间为64MBSDRAM(型号是:HY57V561620[ 4 ],32Mx2)空间为64M0x30000000-0x33ffffff),采用如图2所示的存储空间分布图,因为Nandflash只能存储程序,无法运行程序。为了能够从Nandflash启动,上电复位时,S3C2410通过硬件逻辑把Nandflash的前4KB的内容复制到片内SRAM中,而片内SRAM被映射到地址0x0,这样就可以从地址0x0处取到有效指令,开始执行bootloader,完成把Nandflash中的内核代码复制到sdram中等工作。

2 引导代码和操作系统内核在Nandflash和存储空间中的分布情况

3 Bootloader的设计流程
  Bootloader引导程序是硬件上电复位后首先运行的代码,由它来加载嵌入式操作系统。然后由操作系统接管整个系统,进行进程管理、内存管理、磁盘管理和各个外设管理等工作。 BootLoader是操作系统内核运行之前的一段自举程序,用来初始化硬件设备、改变处理器运行模式和重组中断向量,建立内存空间的映射图,将系统的软硬件环境带到一个由用户定制的特定状态,然后加载操作系统内核。从操作系统的角度来看,Bootloader的总目标就是正确地调用内核来执行。Bootloader一般分为stage1stage2两大部分[ 5 ],对于依赖于CPU体系结构的代码,比如设备初始化代码等,通常都放在stage1中,而且通常都用汇编语言来实现,以达到短小精悍的目的,也就是前面说的启动代码。而stage2则通常用C语言来实现,这样可以实现复杂的功能,而且代码会具有更好的可读性和可移植性。

3.1 Bootloaderstage1
  这部分代码必须首先完成一些基本的硬件初始化。为stage2 的执行以及随后的内核的执行准备好一些基本的硬件环境。Bootloader stage1 一般通用的内容包括:
  (1)设置中断和异常向量;(2)禁止看门狗;(3) 屏蔽所有的中断, Boot Loader 的执行全过程中可以不必响应任何中断, 中断屏蔽可以通过写CPU 的中断屏蔽寄存器或状态寄存器CPSR 寄存器来完成;(4) 设置CPU 的速度和时钟频率;(5) RAM进行初始化, 包括正确设置系统的内存控制器的功能寄存器等;(6)初始化LEDUART,就是通过GPIO来驱动LED,也可以通过初始化UART向串口打印Bootloader的调试信息来表明系统的状态是OK还是ERROR,以便跟踪系统运行情况;(7)关闭CPU 内部指令/数据高速缓存(cache);(8)为加载Bootloaderstage2准备RAM空间;(9)设置好堆栈;(10)跳转到stage2C入口点;其流程图如图3所示。

3 Bootloaderstage1的实现

3.2 Bootloaderstage2
  为了让程序跳入C 语言的“main”函数, 我们采用直接跳转到“main”函数的方法, 实现代码如下:
  b Main
  进入main 函数后即可以开始本阶段stage2 的初始化任务, 这包括:
  (1) 如果在stage1没有初始化UART,这时候至少初始化一个串口, 以便和终端用户进行交互,当然也可以继续点亮或熄 灭LED来判断程序执行情况;
  (2) 修改时钟频率;
  (3) 使能指令Cache
  (5) 从串口中打印一些必要的交互信息,了解系统状态;
  (6) 初始化中断, 包括屏蔽中断, 清除中断悬挂标志, 初始化中断向量表, 注册需要的中断处理函数等;
  (8)打印版本、时间等信息,并从Nandflash复制内核到SDRAM中;
  (9)修改指针,直接跳到内核在SDRAM中的首地址处,至此,完成了Bootloader的全部运行加载工作;

  下面是main()函数和从Nandflash复制内核到SDRAM中的ReadImageFromNandflash()函数的具体实现,但省略了一些具体细节,包括从串口打印的启动、交互、调试信息和一些具体函数的实现。一些具体函数的实现可以参考三星评估版源代码。
  void Main(void)
  {
  JumpAddr=0x30200000; //拷贝内核到sdram中的起始地址,也是内核开始执行的地址
  ChangeClockDivider(1,1); //1:2:4
  ChangeMPllValue(0x5c,0x1,0x1); // FCLK=202.8MHz
  MMU_EnableICache(); //使能指令Cache
  Uart_Init(); // 初始化串口
  Port_Init(); //初始化I/O
  NF_Init(); //初始化Nandflash控制器
  NF_ReadID(); //读取Nandflash存储器ID
  ReadImageFromNandflash();//把存储在Nandflash中的内核拷贝到SDRAM
  rINTMSK=BIT_ALLMSK; //屏蔽所有中断
  Launch(JumpAddr); //跳转到sdram中内核开始处,并运行内核
  }
  从NandflashFlashK9F1208U0M)拷贝内核到SDRAM的函数具体实现如下:
  void ReadImageFromNandflash(void)
  {
  U8 Image_Buf[512];
  U32 Sram_Space=0;
  U32 j,k, numberblock;
  static U32 i, SECTOR_SIZE=512;
  static U8 isbad;
  volatile U32 IMAGE_BASE=0x30200000; //内核在sdram中运行的开始地址
  rINTMSK = BIT_ALLMSK; //屏蔽所有中断
  i=2; //从第2block开始拷贝内核,第0个用于存储本文的bootloader,第1个没用到
  numberblock=2047; //拷贝多少个blocksdram中,视内核大小设置此值
  while(1)
  {
  nextblock:
  isbad=0;
  isbad=NF_IsBadBlock(i); //判断正在拷贝的block是否是坏block
  if (isbad) //是坏block,就进行相应的处理;否则就忽略此处,进行下面的拷贝
  {
  i=i+1; //调整,指向下一个block
  isbad=0;
  if(i>= numberblock) //判断是否拷贝完了所需的block
  {
  Launch(JumpAddr); //拷贝完了所需的block,就跳到sdram中内核开始处
  }
  goto nextblock;
  }
  for(k=0;k<32;k++) //1 block=32 pages
  { // FMD_ReadSector()函数实现从Nandflash存储器中读取数据到数据缓冲区中
  FMD_ReadSector(i, (U8 *)&Image_Buf, k);
  for (j=0;j<SECTOR_SIZE;j++) //1 page=512 bytes
  { //从数据缓冲区中拷贝到sdram
  *((U8 *)(IMAGE_BASE+Sram_Space+j))=Image_Buf[j];
  }
  Sram_Space=Sram_Space+SECTOR_SIZE; //调整sdram中的偏移地址
  }
  i=i+1; //调整,指向下一个block
  if(i>= numberblock) //判断是否拷贝完了所需的block
  {
  Launch(JumpAddr); //拷贝完了所需的block,就跳到sdram中内核开始处
  }
  }
  }

4
试验结果
  由于三星公司的S3C2410集成了Nandflash控制器,它通过硬件逻辑把Nandflash的前4KB内容,即把Bootloader复制到片内sram中,并被映射到地址0x0处。通过跳线设置默认从nandflash启动,那么,系统每次上电或复位后,首先开始运行的就是Bootloader。使用ADS1.2集成开发环境建立Bootloader应用工程,添加必需的文件并设置好编译环境,如BootloaderRO_Base设置为0x0RW_Base设置为0x33ff0000等,调试并最后生成可执行二进制文件,通过JTAG接口把Bootloader烧写到Nandflash的第0block地址开始处,通过usb下载工具把操作系统烧写到第2block地址开始处,复位启动系统运行后的结果如图4所示,该程序用于基于uCOS操作系统的图像采集系统的引导。用同样的方法烧写不同的操作系统内核应用程序,试验结果每一次表明:Bootloader运行良好,启动加载内核快,且简单、实用、可靠。

5 结论
  Bootloader的设计与实现是一个非常复杂的过程,因此要根据具体的硬件和软件需求分析来进行移植或设计。本文设计的Bootloader完成的主要功能包括:试验板硬件的初始化、串口初始化、时钟频率修改以及从Nandflash复制操作系统到SDRAM中运行等,并通过PC机上的超级终端显示了正确的启动运行信息,且可执行代码只有3K左右。因此,本文所详细描述的Bootloader启动运行的全过程,对理解、设计和移植Bootloader具有一定的参考意义。

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

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

相关文章

Java电梯模拟升级版

Java电梯模拟升级版 文章目录 Java电梯模拟升级版前言一、UML类图二、代码三、测试 前言 在上一版的基础上进行升级&#xff0c;楼层采用享元模式进行升级&#xff0c;并对楼层对象进一步抽象 一、UML类图 二、代码 电梯调度器抽象类 package cn.xx.evevator;import java.ut…

如何选择适合本产线的数据采集平台?

工业数据采集是指从工业现场的传感器、仪器仪表、设备等数据源中采集数据&#xff0c;并将其传输到计算机系统或云端进行处理、分析和存储的过程。数据采集平台可以将生产过程中的各种数据进行分析和处理&#xff0c;从而实现智能化生产&#xff0c;提高生产效率和产品质量。 …

Jmeter之Ramp-up Period(in seconds)

1、Ramp-up Period概念 &#xff08;in seconds&#xff09;–并发用户启动周期&#xff0c;告知JMeter 要在多长时间内启动全部Vuser用户。 2、为什么需要有“ramp-up period”&#xff0c;立即启动所有的并发用户数不是更好&#xff1f; 对于绝大多数的网址或应用&#xf…

18.网络游戏逆向分析与漏洞攻防-网络通信数据包分析工具-数据分析工具数据与消息配置的实现

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 上一个内容&#xff1a;17.数据分析工具配置功能的实现 码云地址&#xff08;master 分支&#xff09;&#xff1a;https://gitee.com/dye_your_fingers/titan…

数据库搭建11.2

数据库之搭建 1、rpm -qa|grep 服务名称 案例&#xff1a;rpm -qa|grep mysql 2、将所有msyql的包删除干净 删除方法&#xff1a; &#xff08;1&#xff09;yum remove mysql * 删除linux中的数据库 &#xff08;2&#xff09;yum erase 包名 &#xff0…

微前端之什么是微前端

什么是微前端 微前端分类 基于路由的微前端&#xff1a;组件化微前端&#xff1a;iframe嵌入式微前端&#xff1a; 优点缺点 动态加载/懒加载微前端&#xff1a;微应用容器化方案&#xff1a; 微前端解决方案 single-spa阿里巴巴 Cloud Alfaiframe 方案Web ComponentsModule Fe…

蓝桥杯-Set

目录 HashSet类常用方法 1 add(Object obj)方法 2 size() 方法 3 remove(Object obj)方法 4 contains()方法 5 clear() 方法 例题实战 set 一个不允许出现重复的元素&#xff0c;并且无序的集合&#xff0c;主要有HashSet实现类。 在判断重复元素的时候&#xff0c;Set集…

springcloud:3.5测试慢调用熔断降级

服务提供者【test-provider8001】 Openfeign远程调用服务提供者搭建 文章地址http://t.csdnimg.cn/06iz8 相关接口 测试远程调用&#xff1a;http://localhost:8001/payment/index 服务消费者【test-consumer-resilience4j8004】 Openfeign远程调用消费者搭建 文章地址http://t…

java 中 string常用方法及相关的例子

我将为您详细讲解 Java 中 String 类的常用方法及其相关例子。String 类是 Java 中最常用的类之一&#xff0c;它代表字符串&#xff0c;提供了许多用于操作字符串的方法。 1. 字符串比较 - equals(Object obj): 比较字符串的内容是否相等。 - equalsIgnoreCase(String str): 比…

【官宣】2024广州国际酒店工程家具及商业空间展览会

2024广州国际酒店工程家具及商业空间展览会 Guangzhou International Hotel Engineering Furniture and commercial space exhibition 2024 时间&#xff1a;2024年12月19-21日 地点&#xff1a;中国进出口商品交易会展馆 承办单位&#xff1a;广州佛兴英耀展览服务有…

9.12零钱兑换(LC518-M)(开始完全背包,与01背包的不同仅在于遍历顺序)

算法&#xff1a; 这是一道典型的背包问题&#xff0c;一看到钱币数量不限&#xff0c;就知道这是一个完全背包。 但本题和纯完全背包不一样&#xff0c;纯完全背包是凑成背包最大价值是多少&#xff0c;而本题是要求凑成总金额的物品组合个数&#xff01; 动规五步曲&#…

[Redis]——数据一致性,先操作数据库,还是先更新缓存?

目录 一、操作缓存和数据库时有三个问题需要考虑&#xff1a; 1.删除缓存还是更新缓存&#xff1f; 2.如何保证缓存与数据库的操作同时成功或失效 3.先操作缓存还是先操作数据库&#xff08;多线程并发问题&#xff09; 二、 缓存更新的最佳策略 一、操作缓存和数据库时有…

网络学习:Vlan间路由

目录 一、vlan间路由实现的方法 二、精确匹配转发&#xff08;交换机&#xff09;流程 三、最长匹配转发&#xff08;路由器&#xff09; 四、交换机最长匹配转发 五、总结 一、vlan间路由实现的方法 方法1&#xff1a;使用路由器的物理接口 特点&#xff1a;在路由器上…

LeetCode-第67题-二进制求和

1.题目描述 给你两个二进制字符串 a 和 b &#xff0c;以二进制字符串的形式返回它们的和。 2.样例描述 3.思路描述 将两个二进制字符串转换成整型&#xff0c;然后相加后的整型转为二进制字符串 4.代码展示 class Solution(object):def addBinary(self, a, b):# 将字符串…

Linux 之二:CentOS7 的 IP 常用命令和配置及 xshell 基本使用方法

1. 进入虚拟机 点击右键---进入终端--输入 ip adrr 或 ifconfig 查看ip地址 下面输入命令 ifconfig&#xff08;注意&#xff1a;不是 ipconfig &#xff09; 或 ip addr 来查看当前系统 IP 查看到IP 后&#xff0c;比如&#xff1a;上面是 192.168.184.137 1.1 IP 常用命令…

(文末送书)《低代码平台开发实践:基于React》

最近&#xff0c;我发现了一个超级强大的人工智能学习网站。它以通俗易懂的方式呈现复杂的概念&#xff0c;而且内容风趣幽默。我觉得它对大家可能会有所帮助&#xff0c;所以我在此分享。点击这里跳转到网站。 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&am…

LeetCode234.回文链表

题目 给你一个单链表的头节点 head &#xff0c;请你判断该链表是否为回文链表。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 示例 输入&#xff1a;head [1,2,2,1] 输出&#xff1a;true 输入&#xff1a;head [1,2] 输出&#xff1a;false 思…

mysql 时间精度问题

timestamp到2038年&#xff0c;还有14年时间&#xff0c;一个系统如果能活到那一刻也是相当不错了。 这里先看一下个datetime的问题,下面的插入数据的时间戳是2024-03-06 21:20:50.839 INSERT INTO psi_io_balance ( id, as_id, bill_date, order_id, busi_type, direction, c…

5. gin集成Gorm

文章目录 一&#xff1a;连接DB二&#xff1a;增删改查三&#xff1a;使用PostMan测试 代码地址&#xff1a;https://gitee.com/lymgoforIT/golang-trick/tree/master/24-gin-learning 本节主要介绍如何使用gin集成gorm&#xff0c;并完成用户的创建、修改、删除、查询等功能 …

使用Http请求下载文件带来的问题

java.io.IOException: Broken pipe 当使用http请求的方式将文件作为响应内容给浏览器&#xff0c;这个时候如果浏览器未开启自动下载(浏览器可能会终止这个tcp连接)&#xff0c;文件不会下载成功&#xff0c;但是这个时候请求已经到达服务器&#xff0c;如果这个时候&#xff0…