摘 要:为解决特殊场合DSP程序升级困难的问题,以TMS320F28035为例,介绍了一种基于串口通信的适合于TMS320C2000系列DSP实现程序更新的在线升级方法。描述了该在线升级方法的基本思想和实现步骤,给出了关键部分的程序代码。实验证明,该方法简单可靠,可用于嵌入式设备软件程序的升级更新中。关键词: 在线升级; DSP;串口通信; Flash TMS320C2000系列DSP是美国德州仪器公司(简称TI)推出的集微控制器和高性能DSP特点于一身的DSP系列。该系列的DSP具有强大的控制信号处理能力[1],能够实现复杂的控制算法。随着电子技术的不断发展以及用户需求的不断提升,可能需要经常对已经投入使用的嵌入式设备程序进行更新,而目前一般的程序升级方法是实地取下设备,露出JTAG端口后通过仿真器来更新程序[2-4]。这种方法虽然简单有效,但对于某些特殊场合,会给程序升级带来了极大的不便[2]。本文以TMS320F28035为例,描述了一种可以脱离JTAG仿真器,不改变DSP上电启动方式,实现TMS320C2000系列DSP应用程序在线更新的方法。
1 在线升级的基本思想 一般的基于DSP的软件程序更新是在CCS环境下通过JTAG接口操作来实现的。基于JTAG接口的方法虽然易于操作,而且调试方便,但经常受空间以及传输距离的限制。例如一台DSP系统安装在复杂、封闭的环境下,当程序需要更新或升级时,利用JTAG接口难以实现程序的在线升级[3]。而基于串口通信的在线升级技术是通过用底层程序烧写应用程序的方法来达到程序升级的目的,该方法则不受复杂系统和复杂环境的限制。另外,在线升级方法不需要改变DSP的启动方式,直接采用DSP默认的内部Flash方式启动[5],从而省去了要对DSP的一些引脚进行硬件设置的麻烦。底层程序指已经固化在DSP指定Flash空间中的程序,不允许用户修改和擦除,主要用于实现在线升级的时机判断、数据接收及代码烧写等功能,该程序中使用了Flash2803x_API库存函数(详见2.2节);应用程序即为用户的升级程序[3]。
F28035 DSP每次上电复位,先运行底层程序,与PC机建立联系,然后根据PC机的指令来判断是否需要升级应用程序。若需要,则将通过串口发送来的应用程序代码烧写至F28035片内Flash指定扇区;否则将继续执行原有的应用程序。当应用程序很大或DSP的RAM空间比较小时,可采用将应用程序代码分批发给DSP,DSP接收并烧写完一批代码后,再进行下一批代码的接收和烧写工作,直到所有的应用程序代码都烧写完毕。
2 在线升级的具体实现
2.1 应用程序 用户的应用程序经过CCS编译连接生成具有模块化格式的目标文件(.out),该文件中的代码和数据分别存放在不同的段中,因而不能直接用来烧写Flash,需将其转换为Flash能识别的数据格式——二进制文件 (.bin)。本文采用hex2000.exe和FileOshell.exe工具来实现文件转换。首先,应用程序经过编译连接生成.out文件,然后通过hex2000.exe把.out文件转换成.hex文件,再通过FileOshell.exe将文件转换成.bin文件。先做一个批处理文件,内容如下:
Example_2803xAdcSoc.out
-map Example_2803xAdcSoc.map
-o Example_2803xAdcSoc.hex
-m
-memwidth 16
-image
ROMS
{
Flash28035: origin = 0x3e8000, len= 0x1000, romwidth=
16, fill=0xFFFF
}
其中,Example_2803xAdcSoc.out 是应用程序经过CCS生成的文件;-map是生成map文件;-o是生成hex文件;-m是Motorola-S 格式;-memwidth 16指存储器位数为16 bit;-image指选择映像文件;ROMS 是所需要转换的起始地址、长度、位数及填充。本文选择从0x3e8000开始,长度是4 KB,即FlashH,FlashH中未用的部分用0xFFFF填充,本文把这个批处理文件命名为:Example_2803xAdcSoc.cmd。接下来要生成.bin文件,先做一个MS-DOS型批处理文件,其内容如下: IFileIOShell.exe -i Example_2803xAdcSoc.hex -o Example_2803xAdcSoc.bin
注意要把Example_2803xAdcSoc.out、hex2000.exe、FileIO
Shell.exe、Example_2803xAdcSoc.cmd和MS-DOS型批处理文件放在同一目录下,然后双击MS-DOS型批处理文件,即生成所需要的Example_2803xAdcSoc.bin文件。
2.2 底层程序 底层程序用于实现将串口发送的数据烧写至Flash的指定部分,涉及到应用程序的正确定位和复位后的启动过程,是实现整个在线升级的重点。底层程序流程图如图1所示。底层程序主要实现以下功能[3]:
(1)上电复位查询功能。上电复位后通过接收上位机发送的命令判断是否升级。若上位机发送的是升级命令,则跳转到底层程序中升级部分执行;否则,跳转到原有的应用程序处执行。
(2)搬移烧写程序的功能。由于F28035片上Flash不支持在其中一个扇区运行程序去擦除或烧写其他扇区,故完成接收数据和烧写Flash工作的这部分程序(即底层程序中的升级部分程序)需搬移至片内RAM或片外RAM上运行。实现程序搬移的函数为:
void MemCopy (Uint16 *SourceAddr, Uint16 *Source End
Addr, Uint16 *DestAddr)
{
while (SourceAddr < SourceEndAddr)
{
*DestAddr++ = *SourceAddr++;
}
return ;
}
其中,SourceAddr为Flash中升级程序的起始地址,SourceEndAddr为Flash中升级程序的结束地址;DestAddr为搬移至内存的首地址。
(3)接收上位机发送的应用程序代码并保存到DSP
指定的内存中(一般为RAM区)。这是通过串口RS232来实现的。并确定用于数据保存的这部分内存未被占用。例如,若需要将应用程序代码暂存到F28035的L0 SARAM区域(地址空间0x3F8000-0x3F8800)。定义数组Uint16 BlockBuffer[2048]用于存储应用程序代码,在底层程序中采用存储器定位语句,将上面的缓冲数组定位到相应的存储空间:
#pragma DATA_SECTION(BlockBuffer,“BlockTransferbuffer”);
在底层程序CMD文件中,采用定位语句,将BlockTransferbuffer定位到DSP的L0 SRAM空间:
BlockTransferBuffer:> L0 SARAM PAGE=2
//地址空间:0x3F8000~0x3F8800
通过以上底层程序的设置,可将应用程序缓存到指定的RAM区域中。
(4)代码接收结束后,将内存中的代码烧写至指定Flash扇区,该步骤通过调用Flash2803x_API库函数完成。底层程序中所用到的Flash2803x_API库函数如下[6]:
①擦除扇区的函数为Uint16 Flash28035_Erase(Uint16 SectorMask,&Fstatus),其中,SectorMask为即将被擦除的扇区;&Fstatus为执行擦除操作后返回的状态值,用来判断擦除操作是否成功。②将程序烧写到Flash扇区的函数为Uintl6 Flash28035_Program(&FlashAddr, &BuffAddr,Length,&Fstatus),其中,&FlashAddr为即将被烧写的Flash扇区的起始地址;&BuffAddr为即将准备烧写的程序当前存放在内存空间的首地址;Length为程序长度;&Fstatus为执行烧写操作后返回的状态值,用来判断烧写操作是否成功。③校验烧写到Flash中的程序为Uint16 Flash28035_Verify(&FlashAddr,&BuffAddr,Length,&Fstatus),其中,&FlashAddr指定从Flash内开始比较的首地址;&BuffAddr为被比较文件的存储首地址;Length是需要比较的16 bit字的个数,程序长度;&Fstatus是执行校验操作后返回的状态值,用来判断校验操作是否成功。
2.3 底层程序和应用程序的定位 DSP F28035上电复位后,CPU将从内部Boot Rom获得复位向量。复位向量指向Boot Rom并执行其内部的Bootloader程序,执行完毕后确定从内部Flash启动。程序指针跳转到Flash的0x3F7FF6处。由于这个地址是固定的,因此底层程序必须烧写在以这个地址为起始地址的空间内。DSP进入底层软件程序中运行,首先通过接收上位机的命令来判断是否进行在线升级,如果进行在线升级,则跳转到相应升级程序中执行;否则,跳转到原有的应用程序处执行。由底层程序跳转到原有的应用程序处执行时,采用绝对地址跳转。部分程序如下所示:
#define Jumpgxcx (void (*)(void))0x3E8FFE
//定义应用程序的跳转地址
SCI_SendStatus(“upgrade program? (y/n):”)
//向上位机询问是否升级
temp = SCIA_GetByteData_app();
//接收上位机发送来的是否升级命令
if (temp==’y’)
{
main2(); //如果升级,则跳转到升级程序中执行
}
Else
{
(*Jumpgxcx)();
//如果不升级,则采用绝对地址跳转到应用程序中执行
}
}
底层程序的cmd配置与应用程序的cmd配置要保持一致,不能产生地址冲突。同时,要注意底层程序和应用程序的跳转地址配置。
底层程序cmd文件的部分配置如下:
BEGIN : origin = 0x3F7FF6, length = 0x000002
RESET : origin = 0x3FFFC0, length = 0x000002 /*
codestart : > BEGIN PAGE = 0
应用程序cmd文件的部分配置如下:
BEGIN : origin = 0x3E8FFE, length = 0x000002
codestart : > BEGIN PAGE = 0
3 烧写步骤 首先把底层程序通过JTAG接口烧写到F28035中,然后再进行应用程序的烧写。应用程序的烧写步骤为:先把串口调试工具的参数配置为波特率9 600 bit/s、8 bit数据位、1 bit停止位、没有奇偶校验位;选择发送文本文件方式,发送应用程序的.bin文件到DSP。由于F28035的RAM区比较小,可以采取把应用程序代码分为多次发送的方式。烧写过程如图2所示。
本文介绍了一种基于串口通信的DSP应用程序在线升级技术,可以在不打开机箱的条件下实现模块软件的更新升级。经过实验发现,采用在线升级技术来更新程序所耗费的时间比采用JTAG口烧写程序所耗费的时间要长一些,但解决了复杂情况下程序升级困难的问题。总之,该方法简单可靠,可应用于嵌入式设备的软件程序更新升级中。