凡走过必留下痕迹,学点什么都会有用的。

本系列博文总结了自己在学习嵌入式Linux编程过程中的收获,若有错误,恳请指正,谢谢!

——参考教材韦东山系列教材


bootloader 是一个用于启动linux内核的C程序,为了达到最终启动内核的目的需要完成以下几个步骤:

step1:硬件相关初始化,为启动内核准备硬件平台;

step2:将内核从NAND FLASH读取到SDRAM;

step3:设置需要传递给内核的启动参数;

step4:跳转到SDRAM,运行内核;

下面将详细讲述各个步骤细节:


第一步:硬件相关初始化,为启动内核准备硬件平台

基本流程见下图1

(1)禁看门狗

     ldr r0, =0x53000000  //WTCON : WATCHDOG TIMER CONTROL REGISTER  Address = 0x53000000
mov r1, # //bit 0 = 0 : Disable the reset function of the watchdog timer.
str r1, [r0] //将r1中的值存到r0所指定的地址中

(2)设置系统时钟

配置时钟之前先参考下芯片手册  S3C2440A_UserManual_Rev13,需要操作两个寄存器:

  1. MPLLCON 设置MPLL锁相环倍频因子;

MPLL计算公式 : Mpll = (2 * m * Fin) / (p * 2S)   ;   m = (MDIV + 8), p = (PDIV + 2), s = SDIV ;Fin取决于外部晶振频率,JZ2440开发板的 [OM2:OM3] 设置为 [0:0] ,因此

main clock和USB clock 依赖外部12MHZ 晶振。

2. CLKDIVN 设置 HCLK , FCLK,PCLK比例关系;

FCLK, HCLK, and PCLK
         FCLK is used by ARM920T.
         HCLK is used for AHB bus, which is used by the ARM920T, the memory controller, the interrupt controller, the LCD controller, the DMA and USB host block.
         PCLK is used for APB bus, which is used by the peripherals such as WDT, IIS, I2C, PWM timer, MMC interface,ADC, UART, GPIO, RTC and SPI.

    #define S3C2440_MPLL_200MHZ     ((0x5c<<12)|(0x01<<4)|(0x02))

    /* 2. 设置时钟 */     //先设置各个时钟的比例对CLKDIVN赋值,之后设置系统时钟对MPLLCON赋值
ldr r0, =0x4c000014 //CLKDIVN : CLOCK DIVIDER CONTROL REGISTER  Address = 0x4c000014
mov r1, #0x03 //FCLK:HCLK:PCLK=1:2:4, HDIVN=1,PDIVN=1 FCLK=200MHZ HCLK=100MHZ PCLK=50MHZ
str r1, [r0] //将r1中的值存到r0所指定的地址中 /* 如果HDIVN非0,CPU的总线模式应该从“fast bus mode”变为“asynchronous bus mode” */
mrc p15, , r1, c1, c0, /* 读出控制寄存器 */
orr r1, r1, #0xc0000000 /* 设置为“asynchronous bus mode” */
mcr p15, , r1, c1, c0, /* 写入控制寄存器 */ /* MPLLCON = S3C2440_MPLL_200MHZ */
ldr r0, =0x4c000004 //MPLLCON Address = 0x4c000004
ldr r1, =S3C2440_MPLL_200MHZ // S3C2440_MPLL_200MHZ值比较复杂,使用ldr伪汇编
str r1, [r0]

(3)初始化SDRAM

在使用SDRAM之前,有必要了解下什么是SDRAM。请参阅《高手进阶_终极内存技术指南——完整》,这篇文章解释了SDRAM的基本原理。系统上电需要对SDRAM进行初始化,完成对模式寄存器(MR,Mode Register)的配置。主要设置操作模式、CAS潜伏期时间、突发传输方式和突发长度,完成配置后可以根据地址读写SDRAM。本文操作的SDRAM型号是K4S561632N,S3C2440提供标准的总线接口(BANK6),用于操作SDRAM,因此建立了硬件连接之后,操作S3C2440相应的控制寄存器即可完成对SDRAM的配置和读写,具体操作参阅《[嵌入式Linux应用开发完全手册]》第6章 存储控制器。

 /* 3. 初始化SDRAM */
ldr r0, =MEM_CTL_BASE //SDRAM控制器寄存器首地址,总计需要对13个寄存器配置完成SDRAM的初始化工作
adr r1, sdram_config //读取sdram_config的当前地址到r1,此地址取决于当前PC值,是相对跳转
add r3, r0, #(*) //r3=r0+52 r3值代表了全部控制寄存器所占地址空间大小,总计52字节
:
ldr r2, [r1], # //从r1所指的地方加载指存入r2,之后r1=r1+4,地址加4字节,32位
str r2, [r0], # //将r2的值存入r0所指的地方,之后r0=r0+4
cmp r0, r3
bne 1b //b表示back
 sdram_config:
.long 0x22011110 //BWSCON
.long 0x00000700 //BANKCON0
.long 0x00000700 //BANKCON1
.long 0x00000700 //BANKCON2
.long 0x00000700 //BANKCON3
.long 0x00000700 //BANKCON4
.long 0x00000700 //BANKCON5
.long 0x00018005 //BANKCON6
.long 0x00018005 //BANKCON7
.long 0x008C04F4 //REFRESH
.long 0x000000B1 //BANKSIZE
.long 0x00000030 //MRSRB6
.long 0x00000030 //MRSRB7

  下图是SDRAM的原理图,由两片32M的SDRAM级联成64M,访问一次传递4字节数据。

SDRAM 原理图

SDRAM由BANK6控制,bank6地址空间为0x30000000~0x38000000,共计128MB。上图所示SDRAM存储空间为64M,因此SDRAM地址空间为0X30000000~0X33FFFFFF。

(4)重定位代码

将启动代码从片内SRAM复制到SDRAM中,使用C代码编写。调用C代码需要设置栈,至于为什么要设置栈请参阅http://www.cnblogs.com/xmphoenix/archive/2012/04/28/2475399.html。这里涉及到三个参数:从哪去读取启动代码,将代码复制到哪去,启动代码多大。

  • 从哪去读取启动代码(源地址)

启动代码烧写在NAND,待系统上电后,片内SRAM会加载NAND的起始4K内容。因此启动代码起始地址为:0x00000000

  • 读到哪去(目的地址)

在链接文件中指定了启动代码在SDRAM的中存放地址:0x33f80000,启动代码存到SDRAM顶端512k的地址空间中。

  • 读多少(大小)

启动代码大小 = 代码段+只读数据段+数据段。BSS段不包含在启动代码中,bss段存放没有初始化的全局变量,或者初始化为0的全局变量,程序运行前将bss段清零。

 /* 4. 重定位 : 把bootloader本身的代码从flash复制到它的链接地址去 */
ldr sp, =0x34000000 //设置栈,copy函数使用C语言写,调用C代码之前需要设置栈
bl nand_init mov r0, # //传递参数给 copy_code_to_sdram, 对于NOR 启动
ldr r1, =_start //链接地址即为代码存储区的首地址 _start = 0x33f80000;
ldr r2, =__bss_start //代码段不包括bss段
sub r2, r2, r1 //除去BSS段外的所有代码的空间大小 bl copy_code_to_sdram //参数依次为r0,r1,r2
bl clear_bss

完成以上4个步骤后,CPU会转到SDRAM中取指令执行。

基于JZ2440开发板编写bootloader总结(一)的更多相关文章

  1. Linux学习 :移植U-boot_2016.09到JZ2440开发板

    一.下载源码:ftp://ftp.denx.de/pub/u-boot/ 二.初始化编译: ①新建一个单板: cd board/samsung/ cp smdk2410 smdk2440 -rf   ...

  2. Linux学习 :移植U-boot_2012.04.01到JZ2440开发板

    一.下载U-boot源码:ftp://ftp.denx.de/pub/u-boot/ 二.uboot的启动过程: 部分硬件初始化——>加载完整uboot到RAM——>跳转到第二阶段入口开始 ...

  3. Jz2440开发板熟悉

    title: Jz2440开发板熟悉 tags: ARM date: 2018-10-14 15:05:56 --- 概述 外部晶振为12M Nand Flash 256M,Nor Flash 2M, ...

  4. 基于uFUN开发板和扩展板的联网校准时钟

    项目概述 上周在uFUN试用群里看到管理员说试用活动快结束了,要抓紧完成评测总结,看大家的评测总结也都写了,我也不能落后啊!正好最近做的扩展板到手了,于是赶紧进行调试,做了一个不用校准的时钟,时钟这种 ...

  5. 基于uFUN开发板的RGB调色板

    前言 使用uFUN开发板配合Qt上位机,实现任意颜色的混合,Qt上位机下发RGB数值,范围0-255,uFUN开发板进行解析,然后输出不同占空比的PWM,从而实现通过RGB三原色调制出任意颜色. Qt ...

  6. 基于uFUN开发板的心率计(三)Qt上位机的实现

    前言 上两周利用周末的时间,分别写了基于uFUN开发板的心率计(一)DMA方式获取传感器数据和基于uFUN开发板的心率计(二)动态阈值算法获取心率值,介绍了AD采集传感器数据和数据的滤波处理获取心率值 ...

  7. 基于uFUN开发板的心率计(二)动态阈值算法获取心率值

    前言 上一篇文章:基于uFUN开发板的心率计(一)DMA方式获取传感器数据,介绍了如何获取PulseSensor心率传感器的电压值,并对硬件电路进行了计算分析.心率计,重要的是要获取到心率值,本篇文章 ...

  8. JZ2440开发板:UART(串口)使用(学习笔记)

    查看UART在硬件上的信息,阅读JZ2440原理图可以看到: JZ2440开发板的UART0是可以跟USB相接的,用于打印调试,UART1,UART2两个串口用来外接模块.所以本文仅对UART0进行操 ...

  9. JZ2440开发板:修改ARM芯片时钟(学习笔记)

    想要修改ARM芯片的时钟,需要去查询芯片手册和原理图,获取相关的信息(见下方图片) 首先来看时钟的结构图 根据结构图可以看出,时钟源有两种选择:1. XTIpll和XTOpll所连接的晶振 2. EX ...

随机推荐

  1. Oracle数据库常用关键字以及函数

    常用关键字 insert into---插入数据 delete---删除数据 update---更新一条数据 select---实际工作中尽量不要写* set---设置某些属性 where---给执行 ...

  2. <经验杂谈>C#生成条形码

    虽然二维码满天飞,但也不能忘了条形码,本篇介绍可以在C#中使用的1D/2D编码解码器.条形码的应用已经非常普遍,几乎所有超市里面的商品上面都印有条形码: 条形码的标准: 条形码的标准有ENA条形码.U ...

  3. [SDOI2011]染色 线段树+树链剖分

    考试一共四个半小时,光这道题就打了三个小时..然后又改了俩小时才过.我太蒟蒻了. 其实数据结构这种题就看第一遍打没打顺,顺了就A了,要是再找错再改就慢了,而且样例过了不能说明任何问题(虽然考试的时候我 ...

  4. HTML中表格

    HTML表格 [表格table] 表格用table表示,表格中的每一行tr表示,一行中的每一列用td表示 th表示的是:表头.表头中的文字,默认为加粗居中.th要放在tr中,用于替换掉td. [tab ...

  5. Windows下WebStorm使用SVN(转)

    安装了WebStorm之后,想配置svn,结果在file->settings->Version Contorl->subversion->with conmand line c ...

  6. CSS规范--春风十里不如写好CSS

    先吟几句: 最近看了看春风十里不如你,本来很少看剧的,暑假有点闲就看了,感觉不错,挺喜欢这部剧,就套了个名字,嘿嘿嘿.剧里面印象深刻的是<致橡树>这首诗,念几句: 我如果爱你,绝不像攀援的 ...

  7. 利用webpack构建vue项目

    快速搭建vue项目 一,确认自己有无搭建好node以及npm环境,这些是前提,具体安装方法可参考https://nodejs.org/en/. 二,开始构建项目. 第1步:新建一个文件夹,随意命名. ...

  8. ubuntu14.04 64位安装H3C iNode客户端

    环境: OS:ubuntu14.04LTS 64位 iNode: iNode2.40-R0162 for linux(iNode只有32位的,而且是很久以前的版本) 安装方法: 第一种: 检查本机是6 ...

  9. 打造基于Clang LibTooling的iOS自动打点系统CLAS(二)

    1. 配置LLVM和Clang 在这篇文章里,我们会基于上一篇所述的方案进行展开,详细讲解如何从0开始创建一个基于Clang LibTooling的编译器前端工具.在开始之前,我们假设你已经基本了解何 ...

  10. git 生成公钥、私钥方法与clone使用方法

    我的配置流程 Git配置 Git安装完之后,需做git配置.打开git bash,分别执行以下两句命令 git config --global user.name "用户名" gi ...