历史上的今天
返回首页

历史上的今天

今天是:2025年01月22日(星期三)

正在发生

2020年01月22日 | MSP430程序升级方式探讨

2020-01-22 来源:eefocus

对MSP430系列单片机进行编程的方式有以下三种:利用JTAG接口,利用BSL固件和利用用户自定义的升级固件。由于利用自定义升级固件进行程序升级的方式比较灵活,并且用途广泛,本文将对它作重点介绍。


1. 利用JTAG接口

MSP430系列的单片机都集成了JTAG接口,该接口实现了遵循IEEESTD1149.1规定的测试访问端口状态机(TAP Controller)。它使用一个四线串行接口(TEST用于引脚较少的芯片)。数据或指令从TDI(测试数据输入)移入;串行数据从 TDO(测试数据输出)移出; TCK(测试时钟)作为时钟信号输入;TMS(测试模式选择)信号控制TAP 控制器的状态。利用该接口可以移入指令和数据,从而控制目标芯片的地址线和数据线,达到读写目标芯片FLASH和仿真调试的目的[1]。另外TI现在推出了新型的调试接口—SPY-BI-WIRE,它采用两线制,其中一根为数据线(双向),一根为时钟线。


利用该接口的优点是不需要设计额外的电路和程序,采用仿真器即可下载程序。缺点是一旦用户为了保证代码的安全,烧断了JTAG的熔丝,那么就永久性的破坏了该接口,也就不能再使用该接口了。


2.利用BSL接口

BSL是 Bootstrap Loader的缩写,中文名称是程序装载器。它实质是固化在芯片中的一段通信程序(占用0C00h-1000h的地址空间),利用它可实现对FLASH的擦除和读写。由于其是固化在芯片中的,因此不必担心其被更改或丢失。


该接口使用5根线:GND,TX(P1.1/P1.0),RX(P2.2/P1.1),RST,TCK(TEST),在RST和TCK(TEST)加特定的电平时序信号,即可启动BSL程序,从而实现与目标芯片的通信。通信的字符格式是8个数据位,一个停止位和一个偶校验位。起始波特率为9600bps(BSL 1.6版本可更改波特率到38400bps)。BSL协议要求首先接受一个80h字符用于同步时钟,然后发送应答字符90h。接着接受8个字符,并根据命令跳转到相应的处理例程。BSL程序的C语言描述如下

Void main()

{

 Byte B,bArray[8]; 

 LOCKSTATE = LOCK;

 InitPorts();

 While(1)

 {

  B=ReceiveSyncByte(); 

  SyncTimer(B);

 SendAck();

For( I = 0 ; I < 8 ;i++) bArray = ReceiveByte();

Switch(bArray)

{

 0x12: //写FLASH 

 For(I = 0 ; i {

  B=ReceiveByte();

 If(LOCKSTATE ==UNLOCK) WriteByte(B);

 }

 Break;

 0x10://接收口令

 Receive32Byte();

If(PassWordCorrect) LOCKSTATE = UNLOCK;

  Break;

  0x18..0xn:

 If(PassWordCorrect) DoTask(); 

  }

  If(NoError) SendNak();

 else 

  If(TxData) SendData();

 else 

  SendAck(); 

}

其实现细节可能因版本不同而变化。若用户想利用它来实现程序升级,可参考文[2][3]。 利用BSL程序进行升级的优点是节省代码空间,用户不需要实现自己的升级固件,而且现在已经有很多现成的BSL升级工具;缺点是需预留BSL接口,并且需要现场接线。


3.利用自定义升级固件

MSP430系列单片机的FLASH存储器模块是一个可独立操作的物理存储单元。全部模块安排在同一个线性地址空间中,存储器被分为多个512字节的段(信息段大小为128/64字节)。各段可单独擦除,并且在正常工作电压下程序可对FLASH进行擦写操作,因此特别适合在线程序升级(in-system program)。


自定义升级固件就是在程序中内置一段用于升级应用程序的代码,即可利用现有通信接口进行远程代码的升级。其实现原理是在目标芯片中放置2段代码:一段为应用程序,一段为升级程序。两者的地址段不重叠,这样就可以利用升级程序擦除应用程序并写入新的代码。


3.1引导程序

复位后先进入引导程序,由它来决定进入升级程序或应用程序。引导程序的意义在于当应用程序不存在或错误时能直接进入升级程序,从而保证升级不成功可进行再次升级。

引导程序的描述如下

Void main()

{

 While(1)

 {

  If(ResetVectorValid()) Application();

  Updata();

 }

}


其中的ResetVectorValid()函数用于检测应用程序是否存在或是否有效。实现可以是检测EnterApplication的入口地址是否合法,一种简单的实现是

#define ResetVectorValid() (ResetVector !=FFFF)

其中ResetVector为应用程序的入口地址,该地址通常放在一个固定的地址中,升级程序后修改该入口地址。Application() 为应用程序,它若正常执行不会返回 , 只有在接受到升级指令后才返回。可在Application()中使用Return语句进入升级程序。


Updata()为升级程序,其入口处必须加检测指令,以确认是正常进入升级程序。进入升级程序后,通信端应先发送擦除指令,擦除原有代码,然后发送升级代码更新FLASH。若是具有外部扩展存储器或用户程序较小,可先接收整个程序段,若校验正确再写入,这样可靠性会更高。


这里有个策略就是最先擦除包含ResetVector的块,最后写入ResetVector的值,这样可以尽量保证不会进入不完整的应用程序。


3.2应用程序的编写

应用程序的编写没有什么大的变化,需要在通信协议中加入自定义的一个升级命令,用于进入升级程序。另外需更改链接文件(*.XCL),指定应用程序的地址范围,如下以应用程序地址范围为2500-F7DC为例(用//注释掉的为默认的设置)

// Code

//-Z(CODE)CSTART=2500-FFDF

//-Z(CODE)CODE=2500-FFDF

-Z(CODE)CSTART=2500-F7DF

-Z(CODE)CODE =2500-F7DF

// Constant data

//-Z(CONST)DATA16_C,DATA16_ID,DIFUNCT,CHECKSUM=2500-FFDF

-Z(CONST)DATA16_C,DATA16_ID,DIFUNCT,CHECKSUM=2500-F7DF


// Interrupt vectors

//-Z(CONST)INTVEC=FFE0-FFFF

-Z(CONST)INTVEC=F7E0-F7FF

修改完毕后将该文件添加到工程。编译后的代码即可作为升级代码。


3.3升级程序的编写

新建一个工程,按如上的方法将升级代码定位到与应用程序不重叠的区域,如F800-FFFF,此时不修改 

 -Z(CONST)INTVEC=FFE0-FFFF

在升级程序中将除复位中断外的所有中断映射到应用程序中,一种办法是嵌入汇编,采用汇编的定位指令ORG;或者写15个中断影射函数,如下

//重新映射中断向量地址

 #pragma vector=0x0 

  __interrupt void intvec_0(void)

 {

  asm("br & 0F7E0h"); //假设F7E0中存放中断15的地址

 }

另外也可以采用动态确定中断入口地址的方法,即将中断向量地址放入约定好的RAM中,如下

 __no_init void (*intvec1[16])() @ 0x200; //定义指向函数指针的数组,用于映射新的中断向量

//重新映射中断向量地址

#pragma vector=0x0 //

__interrupt void intvec_0(void)

 {

  asm("push R15");

  asm("mov #0x200,R15");

  asm("call @R15");

  asm("pop R15");

 }


然后在应用程序中进行中断向量的映射,如

intvec1[TIMERA0_VECTOR/2]=Timer_A_0;

即在TIMERA0中断时执行Timer_A_0()函数。这样做的优点是可以在运行时动态决定中断函数的入口,即如高级语言中的虚函数(Virtual Function)。

当这两个函数块编写完毕后就可以进行工程测试了。


3.4应用程序与升级程序同时完成

也许您还希望两个函数在一个工程里完成。这时除了需要修改链接文件外,还需要注意以下几点:

(1)将升级程序的所有函数定位到升级程序空间,即在函数前面加定位指令

#pragma location="UPDATECODE" // UPDATECODE为升级程序所在段的名称


(2)修改函数返回调用的例程。当函数返回时会调用弹出寄存器的默认例程,而这些例程可能并不在升级程序的地址空间内。一种解决方法是利用编译环境生成的LST文件(汇编代码),逐个修改函数返回时调用的弹出寄存器例程,这样就可以保证两者代码独立。这样做的缺点是每次更改C语言代码后,就要重新修改汇编代码,比较繁琐。另一种方法是考虑到升级程序所做的就是接受和发送数据,一般不需要使用中断。这样就可以在升级函数前面加入__monitor 编译指令,指明该函数为原子操作。这类函数入口处先压入SR并禁止中断,返回时使用RETI返回,此时编译器并不调用例程弹出保存的寄存器,而是根据进栈情况逐个弹出寄存器。


(3)更改SWITCH语句。使用SWITCH语句时编译器也会产生默认例程调用,很难屏蔽掉,故只有将SWITCH修改为多个判断语句。

推荐阅读

史海拾趣

DREMEL公司的发展小趣事

随着全球化的趋势日益明显,Dremio公司开始积极拓展国际市场。他们制定了详细的国际化战略,包括在海外市场设立分支机构、与当地企业建立合作关系、参加国际展会等。这些举措不仅提高了Dremio公司在国际市场的知名度,也为其带来了更多的商业机会。通过国际化战略的实施,Dremio公司的业务规模不断扩大,品牌影响力也日益增强。

请注意,以上故事均基于虚构,旨在展示一个类似Dremio公司在电子行业中的可能发展路径和故事。如有需要,请进一步核实相关事实和数据。

Display Engineering Services公司的发展小趣事

随着电子行业的不断发展,数据治理和云数据湖领域也面临着越来越多的挑战。Dremio公司始终保持敏锐的市场洞察力,积极应对各种挑战。他们不断投入研发资源,推出了一系列创新产品和技术解决方案。例如,他们开发了一种基于人工智能的数据分析引擎,能够自动识别和预测数据中的异常和趋势。这一技术不仅提高了数据分析的准确性和效率,也为企业提供了更全面的数据支持。

明微公司的发展小趣事

随着全球半导体行业向中国市场的转移,明微公司意识到拓展海外市场的重要性。为了加快海外市场拓展步伐,公司积极加强与国际知名企业的合作与交流,引进先进的技术和管理经验。同时,公司还积极参加国际展会和论坛等活动,提升品牌影响力。这些举措使明微公司在国际市场上逐渐树立起良好的口碑和形象。

Hitachi (Renesas )公司的发展小趣事

明微公司自创立以来,始终坚持以技术创新为核心竞争力。在数字电视芯片领域,明微团队通过深入研发,成功推出了一款具有高性能、低功耗特点的芯片产品,该产品凭借其卓越的性能和合理的价格,迅速在市场上获得了认可。此后,明微公司继续加大研发投入,不断推出具有竞争力的新产品,逐步在电子行业中崭露头角。

振华(CEC)公司的发展小趣事

为了进一步提升企业的竞争力和市场份额,振华积极实施国际化战略。公司加强与国外企业的合作与交流,积极参与国际市场竞争,通过引进外资、设立海外研发机构等方式,不断拓展海外市场。同时,振华还注重提升产品的国际竞争力,加强与国际标准的对接和认证工作,确保产品能够满足不同国家和地区的市场需求。

DIOTECH公司的发展小趣事

DIOTECH公司成立于电子行业的黄金时期,凭借其创始人在半导体领域的深厚造诣,公司迅速在高性能处理器领域取得了技术突破。通过不断优化产品设计和生产工艺,DIOTECH的产品在性能和稳定性上均达到了行业领先水平,逐渐赢得了市场的认可。随着智能手机的普及,DIOTECH的高性能处理器成为了市场上的抢手货,公司的营收和市场份额均实现了快速增长。

问答坊 | AI 解惑

还是全屏的问题!

我把下面函数加进去以后编译出错 void CRFCEDlg::FullScreen(HWND hDlg) {         RECT rc; ::GetWindowRect(hDlg, &rc);          SHFullScreen(hDlg, SHFS_HIDETASKBAR); ::MoveWin ...…

查看全部问答>

进程间如何做到互斥?

gpio寄存器都是以组的形式出现(一个寄存器对应一组gpio.pin),我们的多个驱动会用到同一组gpio的不同pin来做一些使能复位等控制,那么会出现对不同pin但同组即同寄存器操作的状况吗?在同一进程下,我们可以用线程同步的互斥体来避免冲突。如果是 ...…

查看全部问答>

用数据线连接手机并控制STK卡菜单,这个能否实现?请高手指点

就是用手机数据线连接计算机后,直接用AT命令(或其他方式)控制SIM卡(STK)的菜单 执行SIM卡(STK)菜单的功能,是否能实现,如果能请说出详细的步骤。或者其他的实现方法 主要就是能在计算机上操作SIM卡(STK)菜单执行移动的增值服务功能。 如果不用 ...…

查看全部问答>

关于定时器和串口通讯问题

51有两个定时器,如果两个定时器的初始化一样,怎么知道是用哪个定时器作为波特率发生器。 初学者,请指教。…

查看全部问答>

格式化分区的问题

运行到pfnFormatVolume(hPart, NULL,  &pfo, NULL, NULL)后, 就停在这了。 重启后发现分区Part01里的内容已经清空。pfnFormatVolume出不来会是什么原因? 代码帖上来了: void CFormatDlg::OnFormat() {       &nb ...…

查看全部问答>

请大家帮忙推荐一下,开发atmega48用哪种环境比较好?

请大家帮忙推荐一下,开发atmega48用哪种环境比较好?谢谢大家了!…

查看全部问答>

突发奇想

原来提交方案的时候曾经提到过光控功能,可惜后来因为比较忙,就偷懒给省略掉了。今天在看以前画的电路图时突发奇想,画了个光控部分的草图。因为今天比较晚了,所以先传上原理图,改日再对其原理做介绍。 实现的功能是: 1、灯正在使用中,检 ...…

查看全部问答>

基于AT91SAM7X256下ucos的通信项目(外包)

【项目内容描述】把原来裸机写的3000行代码移植到ucos上,使具有tcp/ip的socket通讯功能,和别的主机通过tcp/ip通信 最好在北京。   费用大概1.5万左右,还可以再商量。   联系qq:340283020…

查看全部问答>

怎么在MSP430F149里读取实时时间?

430怎么读取实时的时间呢?求指导啊!…

查看全部问答>

*键+-键切换层都不能用,重新安装也不行

*键+-键切换层都不能用,重新安装也不行,怎么办,各位大侠帮帮忙吧,支支招。 还有,我同事的原件流水号可以按不同规格的排序,我的为啥不行,例如不同阻值的可以连续,…

查看全部问答>