基于C语言和GEL语言的Flash编程新方法

发布者:EnchantedWish最新更新时间:2011-02-16 手机看文章 扫描二维码
随时随地手机看文章

  在DSP应用系统开发的后期,一般需要将用户程序写进Flash等非易失性存储器,以便采用并行引导的方法实现用户程序的自举加载。这一步骤称 为“烧写”;针对Flash的烧写又称为Flash编程。以往的编程方法大多采用汇编语言编写程序,可读性较差,并将引导表的制作也放在程序中实现;用户 程序一变,烧写程序就得重新编写,不具有通用性。参考文献[1]采用C语言完成Flash读写,较清晰地体现了Flash编程的思想,但是它采用指针访问 Flash空间,不能对高端Flash(64 K字存储空间以外)进行访问,且将引导表作成数组的方法仍显机械。

  这里提出的Flash编程方法完全采用C语言编写烧写程序,运用函数地址访问高端Flash,借助数据文件将引导表加载到数据空间。GEL (General Extension Language,通用扩展语言)作为一种程序扩展语言,被广泛用于调试及程序运行环境的定制。这里将GEL语言运用于Flash编程,可以控制C程序在 数据加载完成后执行烧写过程,从而实现大引导表的烧写。

1  DSP开发板及Flash存储器

  笔者使用的DSP开发板上有1片TMS320VC5402通用DSP芯片、1片SST39VF400A存储芯片(Flash)、键盘和液晶显示 器等。其中Flash容量为256 K字(1字=16位),组织为128个扇区或8个块。为充分发挥Flash容量大的特点,本系统在硬件上将Flash空间的映射设计为:在上电自举过程 中,Flash空间的0x04000~0x0FFFF映射到数据空间的0x4000~0xFFFF;上电自举完成后,整个Flash空间 0x00000~0x3FFFF映射到程序空间的0x80000~0xBFFFF,即映射到了TMS320VC5402的扩展程序空间,处于高地址,因此 称为“高端Flash”。由此可知,对系统进行应用开发时,Flash总是表现为高端Flash。

按此在新窗口浏览图片
图1  Flash编程流程

2  Flash编程流程

  用户程序一般以可执行COFF(公共目标文件格式)文件格式存在(后缀名为.out),Flash编程所要完成的就是将此可执行文件转换成特定 的ASCII码引导表的格式,并按此格式顺序写进Flash。Flash编程流程如图1所示。下面仅以一个动画显示程序qq.out为例,介绍如何将其烧 写进Flash。

2.1  生成引导表

  通过Hex转换工具,将用户程序qq.out文件转换成十六进制形式的ASCII码流文件(ASCIIHex格式文件[2])qq.asc。首先编写一个convert.cmd命令文件。部分内容如下:

  qq.out/*用户程序*/
  -a/*转换成ASCIIHex格式文件*/
  -map qq.mxp/*包含引导表的长度等信息*/
  -o qq.asc/*转换成qq.asc*/

  执行命令行“hex500 convert.cmd”将产生qq.mxp和qq.asc文件。其中qq.mxp文件有这样的信息:“CONTENTS: 00000000…0000433b”。表示qq.asc中的引导表长度为0x433C字,内容大致为:“10 AA 7F FF 00 02 00 00…”。

2.2  转换成数据文件

  编程将ASCIIHex格式文件qq.asc转换成CCS(Code Composer Studio,代码集成开发环境)支持的数据文件(后缀名为.dat)。例中的引导表已属较大的表,这里将其转换为两个数据文件qq_dat1.dat和 qq_dat2.dat,以在同一缓冲区分两次装载,避免因缓冲区太小而容纳不了引导表的情况发生。

  CCS支持的数据文件的第一行为文件头信息,格式为:

  幻数  数据格式  起始地址  页类型  数据块大小

其后是文件内容,每行表示一个数据。其中幻数固定为“1651”,数据格式可以选择“1”(十六进制整型)、“2”(十进制整型)、“3”(十进制长整型)、“4”(十进制浮点型)。

  利用VC6.0编写该转换程序是简单的,程序运行后产生的qq_dat1.dat文件将是:“1651 1 4000 1 2000 0x10AA…”。从文件头信息可知,加载该文件可将引导表装载到数据空间0x4000起始的长度为0x2000的缓冲区中。

2.3  Flash烧写

  利用GEL程序将引导表形成的数据文件qq_dat1.dat和qq_dat2.dat逐次装载到数据空间,调用C程序执行烧写过程。

  由于Flash空间映射到TMS320VC5402程序空间的0x80000~0xBFFFF,故实际编写程序时使用的Flash空间的地址均需偏移0x80000。例如,Flash空间的0x5555地址单元实际上为0x85555。

  正如前面所介绍的,虽然高端Flash囊括了整个Flash空间,但是对于C54x系列芯片,其C语言指针的宽度为16位,只能访问64 K字范围(0x0000~0xFFFF)之内的存储空间,而不能访问高端Flash(0x80000~0xBFFFF)。

  参考文献[3]讨论了用C语言指针不能访问C54x系列DSP扩展程序空间的问题,提出了用函数名代替指针来访问扩展程序空间的方法,并给出了 可供C程序调用的pfunc_ext.lib库。这一方法本质上是将函数名代表的程序空间地址(20位)传送到40位的累加器,进行累加器寻址,因此使用 该库恰好可以解决指针不能访问高端Flash的问题。库中以下两个函数是有用的:

int PFUNC_wordRead(PFUNC addrProg);
//读取(扩展)程序空间地址addrProg处的一个字
void PFUNC_wordWrite(PFUNC addrProg,int wData);
//将字wData写到(扩展)程序空间地址addrProg处

  为应用pfunc_ext.lib库,需定义一些函数,并在命令文件中为这些函数所在的自定义代码段分配段地址,以使这些函数的函数名指向Flash特定的地址单元。例如,可以编写一个C程序源文件,定义一个空函数FLASH_5555以指向0x85555:

#pragma CODE_SECTION(FLASH_5555,"bigpointer")
void FLASH_5555(void){}

  Flash的其他地址可依此方法得到, pfunc_ext.lib库的具体说明见参考文献[3]。

  下面应用pfunc_ext.lib库编写了Flash擦除和编程的3个基本函数flash_erase()、 flash_word_write()、flash_serial_write(),分别完成Flash擦除、字编程和连续编程。其中连续编程只是循环调 用了字编程函数。擦除和字编程的流程分别如图2和图3所示。擦除函数的代码如下:

//实现片擦除、块擦除或扇区擦除,type定义擦除方式,addr给出扇区起始地址或块起始地址
unsigned int flash_erase(PFUNC addr,unsigned type){
  //执行SST39VF400A的擦除命令序列
  PFUNC_wordWrite(FLASH_5555,FLASH_CMD1);
  //0xAA﹥*(0x85555)
  …
  PFUNC_wordWrite(addr,type);//擦除类型命令
  …
  }

按此在新窗口浏览图片
图2  擦除流程

按此在新窗口浏览图片
图3  字编程流程

  有了这些基本函数,就可以在主函数中完成Flash的烧写。下面的主函数实现将引导表烧写进Flash。

void main(){
  …
  asm("erase:");//擦除0x80000~0x97FFF,块擦除
  for(i=0;i<3;i++)
  flag=flash_erase((PFUNC)i,FLASH_BLOCK_ERASE);
  asm("program1:");//连续编程
  flag=flash_serial_write(FLASH_BASE,MEM_BASE,usercode_length1);//FLASH_BASE指向0x84000
  asm("program2:");//连续编程
  …
  asm("program_bootaddr:");//字编程
  flag=flash_write_word(FLASH_FFFF,0x4000);
}

  例中采用了块擦除的方式。MEM_BASE是多次加载引导表的缓冲区起始地址,为与数据文件qq_dat1.dat中文件头对应,应保证 MEM_BASE指向0x4000。其方法类似于上述函数名的地址分配(使用#pragma DATA_SECTION伪指令)。最后完成字编程,使Bootloader上电时得以在数据空间的0xFFFF处读取引导表在数据空间的起始地址,例中 为0x4000。

  为使主函数正确执行,需借GEL语言的运行调试功能,由此设计的GEL程序真正体现了Flash烧写的流程。GEL程序流程如图4所示,部分代码如下:

menuitem "PROGRAMMING";
hotmenu FlashFiring(){…
  GEL_Load("ProgramFlash.out");//加载C烧写程序
  …
  if(flag){
  GEL_Load("qq_dat1.dat");//加载数据文件
  GEL_Go(program1); //执行连续编程
  …
  GEL_Load("qq_dat2.dat");//加载数据文件
  GEL_Go(program2); //执行连续编程…
}
  }

按此在新窗口浏览图片
图4  GEL程序流程

  GEL程序在C程序每次执行前设定正确的环境变量并初始化缓冲区。例如,数据文件的长度usercode_length1就是需要根据实际的数据文件长度进行设定的环境变量;而在进行连续编程之前,需要GEL程序重新加载MEM_BASE缓冲区。

3  运行结果

  在CCS环境下选择File/Load GEL,装载以上GEL程序,选择GEL/ PROGRAMMING/FlashFiring,即可实现Flash烧写。拔掉仿真器,给系统重新上电,可以看到液晶显示器上QQ企鹅的动画。

4  结论

  通过函数地址可以进行Flash的全空间访问;采用C语言编写Flash擦除和编程函数,增强了程序的可读性;将引导表作成多个数据文件,一方 面适于大引导表的加载,另一方面使Flash编程算法与编程数据完全分离,提高了算法的通用性;仅在GEL程序中修改参数即可实现另一用户程序的烧写,体 现了方法的灵活性。

参考文献

[1]  张勇.C/C++语言硬件程序设计——基于TMS320C5000系列DSP[M].西安:西安电子科技大学出版社,2003(5):206-230.
[2]  Texas Instruments. TMS320C54x Assembly Language Tools User's Guide. SPRUF102,200-210.
[3]  David M Alter. Using C to Access Data Stored in Program Memory on the TMS320C54x DSP[R]. SPRA177A, Texas Instruments Application Report,2005-08.
[4]  北京合众达电子技术有限公司. SEED——MMI5402用户指南. 2004-06.

王海涛(硕士),主要研究方向为机器视觉、焊接自动控制;
张文明(教授),主要研究方向为焊接自动控制;
王滨(教授),主要研究方向为机器视觉。

引用地址:基于C语言和GEL语言的Flash编程新方法

上一篇:Altera为Nois II处理器提供C语言硬件加速工具
下一篇:数字逻辑电路教学中的C语言描述和应用

热门资源推荐
热门放大器推荐
    系统发生错误

    系统发生错误

    您可以选择 [ 重试 ] [ 返回 ] 或者 [ 回到首页 ]

    [ 错误信息 ]

    页面错误!请稍后再试~

小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
  • 学习ARM开发(16)
    ARM有很多东西要学习,那么中断,就肯定是需要学习的东西。自从CPU引入中断以来,才真正地进入多任务系统工作,并且大大提高了工作效率。采 ...
  • 学习ARM开发(17)
    因为嵌入式系统里全部要使用中断的,那么我的S3C44B0怎么样中断流程呢?那我就需要了解整个流程了。要深入了解,最好的方法,就是去写程序 ...
  • 学习ARM开发(18)
    上一次已经了解ARM的中断处理过程,并且可以设置中断函数,那么它这样就可以工作了吗?答案是否定的。因为S3C44B0还有好几个寄存器是控制中 ...
  • 嵌入式系统调试仿真工具
    嵌入式硬件系统设计出来后就要进行调试,不管是硬件调试还是软件调试或者程序固化,都需要用到调试仿真工具。 随着处理器新品种、新 ...
  • 最近困扰在心中的一个小疑问终于解惑了~~
    最近在驱动方面一直在概念上不能很好的理解 有时候结合别人写的一点usb的例子能有点感觉,但是因为arm体系里面没有像单片机那样直接讲解引脚 ...
  • 学习ARM开发(1)
  • 学习ARM开发(2)
  • 学习ARM开发(4)
  • 学习ARM开发(6)
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved