- 浏览: 12861 次
最新评论
自己动手编写嵌入式Bootloader之(3)
2011年04月03日
这一部分将对前文没有提到的几段关键代码进行简单说明,介绍一下源代码组织结构和Makefile系统,展示一下实验运行结果,并提供全部源代码下载。
1. 定时器初始化和延时程序
因为在 CS8900A的驱动程序中需要用到延时,因此有必要对S3C2440的计时器进行使能和初始化,并编写延时程序。
S3C2440A共有5个定时器,编号为Timer0 ~ Timer4。其中Timer0 ~ Timer3都有输出引脚,可以通过定时器来控制引脚电平周期性的变化,这称为脉冲宽度调制(PWM:Pulse Width Modulation)功能。而Timer4没有输出引脚,也就没有PWM功能,所以Timer4常被程序里的延时函数使用。
定时器部件的时钟源为PCLK,但是需要经过两级预分频之后才真正供定时器使用。第一级预分频由TCFG0寄存器控制,其位[7:0]设置预分 频器0的值,供Timer0和Timer1使用,位[15:8]设置预分频器1的值,供Timer2 ~ Timer4使用。第二级预分频由TCFG1寄存器控制,其每四位控制一个定时器,可以从2分频、4分频、8分频、16分频、外接TCLK0/TCLK1 这五种频率中选择。
我们的延时函数使用Timer4,其它定时器全部关闭。初始化程序中设置:TCFG0 = 0x0f00; 表示Timer4的第一级预分频值为 15+1 = 16。寄存器TCFG1使用默认值全0,表示第二级预分频为2分频。前面已经设置PCLK为50MHz,这样Timer4实际的工作频率为:
50MHz/16/2 = 50000000/32 = 1562500Hz
注意计算时钟频率时的MHz是指10^6,而不是2^20;同理KHz是指1000Hz,而不是1024Hz。
我们在TCON中把Timer4设为"自动加载"。当Timer4启动时,TCNTB4的值将被自动装入内部寄存器TCNT4,然后在工作频率 下,TCNT4开始减1计数,当到达0时,TCNTB4的值又被自动装入TCNT4,下一个计数流程开始。我们把TCNTB4设为15625,则一个计数 流程的的长度为10毫秒。
假设要延时的时间为msec毫秒,则共需要的计数值为 tmo = msec*15625/10,设一个变量timestamp保存已经过去的时间戳,每次读取TCNT4的值后更新timestamp,直到它大于 tmo 。程序如下:
Code highlighting produced by Actipro CodeHighlighter (freeware)
http://www.CodeHighlighter.com/
while( timestamp = thisdec ) /* normal mode */
{
timestamp += lastdec - thisdec;
}
else/* we have an overflow ... */
{
timestamp += lastdec + TIMER_LOAD_VAL - thisdec;
}
lastdec = thisdec;
}
TCNT4的值可由寄存器TCNTO4读出。程序中保存了最近两次读出的TCNTO4值, 如果本次值比上次小,说明在同一个计数流程内;如果本次值比上次大,说明已经进入了下一个计数流程。
内容导航
2. 串口标准输入输出
要想在Bootloader中使用scanf()和print()并不容易,因为不能直接使用C库函数。scanf()要从串口获得输入, print()要向串口进行输出。必须自己实现常用的C库函数, 不仅包括输入输出函数,还包括字符串操作函数如strcmp(), strcpy()等。幸好在《嵌入式Linux应用开发完全手册》这本书的源代码中提供了这样简化的C库,所以就直接拿来用了。
代码中定义了两个全局数组作为输入输出缓冲区:
static unsigned char g_pcOutBuf[ 1024 ];
static unsigned char g_pcInBuf[ 1024 ];
其实我们可以把这两个缓冲区定位在CPU的 SteppingStone 里面,这样可以节省2K的空间。
scanf()的实现里面调用 getc() 函数, printf() 的实现里面调用 putc() 函数。我们自己写getc()函数为从串口读取字符, putc()函数实现为向串口发送字符, 这样标准输入输出就跟串口联系在一起了。
Code highlighting produced by Actipro CodeHighlighter (freeware)
http://www.CodeHighlighter.com/
/* 发送一个字符 */
void putc(unsigned char c)
{
/* 等待,直到发送缓冲区中的数据已经全部发送出去 */
while (!(UTRSTAT0 & TXD0READY));
/* 向UTXH0寄存器中写入数据,UART即自动将它发送出去 */
UTXH0 = c;
}
/* 接收字符 */
unsigned char getc(void)
{
unsigned char ret;
/* 等待,直到接收缓冲区中的有数据 */
while(!(UTRSTAT0 & RXD0READY));
/* 直接读取URXH0寄存器,即可获得接收到的数据 */
ret = URXH0;
if (ret == 0x0d || ret == 0x0a)
{
putc(0x0d);
putc(0x0a);
}
else
{
putc(ret);
}
return ret;
} 内容导航
3. 源代码组织结构
源代码跟目录下只有两个文件, 主Makefile和链接脚本sboot.lds。
文件夹start内有start.S和nand.c,前者是上电后最初运行的汇编代码,后者含有Nand Flash的读函数,负责把S-Boot代码从Nand拷贝到RAM中。
文件夹main内有main.c,是一个死循环,提供若干菜单供用户选择,然后调用相应功能的程序。
文件夹lib内是简化和移植过的C标准库,包括输入输出和字符串操作函数。
文件夹include内是一些头文件。
文件夹app内有boot_linux.c和tftp.c,从名字就能看出它们的功能。
文件夹device内含有设备驱动程序,如串口初始化、定时器初始化和延时函数、网卡驱动、网络协议实现等。
每个文件夹内都有自己的Makefile,根目录下的主Makefile会进入各个子目录并调用各自的Makefile。每个子目录下的 Makefile把自己编译的代码链接成一个build-in.o文件, 主Makefile把各个子目录下的build-in.o链接成一个可执行文 件。
编译器使用自己制作的 arm-hwlee-linux-gnueabi-gcc. 可以从这里下载。 给gcc增加 -nostdinc 选项, 表示不使用标准C库函数,不到/usr/include目录下寻找包含文件, 只在-I$(INCLUDEDIR)指定的目录寻找包含文件。
内容导航
4. 提供全部源代码下载:
文件:S-Boot.tar.gz
大小:41KB
下载:下载
5. 运行结果截图
图中,首先选择3从TFTP服务器下载内核到RAM中, 然后选择4从RAM成功启动内核。
选择2还有通过串口Kermit协议下载内核的功能,前文没有对这部分代码作分析,有时间再补上。下面附一张截图:
发表评论
-
如何为嵌入式开发建立交叉编译环境
2012-01-20 12:16 701如何为嵌入式开发建立交叉编译环境 2010年10月12日 ... -
windows编程新手常见问题
2012-01-20 12:16 404windows编程新手常见问题 ... -
C++标准库简介
2012-01-20 12:16 683C++标准库简介 2011年03 ... -
汇编调用C库函数
2012-01-20 12:16 793汇编调用C库函数 2010年07月29日 1.先包含(由 ... -
python难吗?
2012-01-19 16:59 861python难吗? 2010年06月04日 学的人很少的 ... -
python
2012-01-19 16:59 962python 2010年06月08日 面对珍珠(Pe ... -
学习python
2012-01-19 16:59 706学习python 2011年09月29日 ... -
Python for s60[1_pys60简介]
2012-01-19 16:59 732Python for s60[1_pys60简介] ... -
100%解决Python平台安装更新错误的问题
2012-01-19 16:58 647100%解决Python平台安装更新错误的问题 2010年1 ... -
2012-1-12
2012-01-17 06:44 5572012-1-12 2012年01月12日 瀹剁┓浜轰 -
屎上最难!最搞笑的脑筋急转弯!不看答案你能猜对几个?
2012-01-17 06:43 707屎上最难!最搞笑的脑筋急转弯!不看答案你能猜对几个? 201 ... -
UNIX编程随笔(三十四)socket地址
2012-01-16 05:25 978UNIX编程随笔(三十四)socket地址 2010年04月 ... -
C#托管Socket的实现方法
2012-01-16 05:25 1027C#托管Socket的实现方法 2009年10月23日 ... -
UNIX编程随笔(三十五)连接以及读写socket
2012-01-16 05:24 1196UNIX编程随笔(三十五)连接以及读写socket 2010 ... -
socket 通信基础2
2012-01-16 05:24 695socket 通信基础2 2011年01月05日 ◆先看 ... -
ActionScript3.0 Socket编程
2012-01-16 05:24 576ActionScript3.0 Socket编程 2 ...
相关推荐
[原创]自己动手编写嵌入式Bootloader之(1).pdf
自己动手编写嵌入式Bootloader之(3) 自己动手编写嵌入式Bootloader之(3) 自己动手编写嵌入式Bootloader之(3)
自己动手编写嵌入式Bootloader之(2) 自己动手编写嵌入式Bootloader之(2) 自己动手编写嵌入式Bootloader之(2) 自己动手编写嵌入式Bootloader之(2)
[原创]自己动手编写嵌入式Bootloader之(3).pdf
[原创]自己动手编写嵌入式Bootloader之(2).pdf
在这种情况下必须在第一条指令处做一些初始化工作,这段初始化程序与操作系统独立分开,称之为bootloader。 实际上,很少有必要自己写一个Bootloader,因为U-Boot已经强大到能够满足各种需要。但是强大必然复杂,一...
对于wince 开发Bootloader比较重要
嵌入式BootLoader技术内幕.pdf
嵌入式BootLoader技术内幕 嵌入式BootLoader技术内幕
基于ARM9嵌入式BootLoader设计与实现,徐学均,,介绍了基于ARM9嵌入式系统在上电启动后操作系统运行之前BootLoader的启动过程,并设计出了具有良好模块性和可移植性的启动程序,试验�
自己写的嵌入式bootloader stage1,用LED完成测试(基于vivi 简单明了的程序,有利于分析代码
ARM_linux的嵌入式bootloader设计ARM_linux的嵌入式bootloader设计ARM_linux的嵌入式bootloader设计ARM_linux的嵌入式bootloader设计
文档总结了运行在嵌入式平台上的bootloader的编写过程,很详细,希望对大家有所帮助!
基于ARM的嵌入式Bootloader分析与移植.pdf基于ARM的嵌入式Bootloader分析与移植.pdf基于ARM的嵌入式Bootloader分析与移植.pdf
嵌入式Bootloader技术内幕
(经典)嵌入式BootLoader 技术内幕
Linux嵌入式BootLoader开发之BLOB_小巧精悍的引导程序 更多资源请访问http://www.59186618.com