一直提醒自己要更新CSDN博客,但是确实这段时间到了一个项目的关键节点,杂七杂八的事情突然就一涌而至。STM32、FPGA下位机代码和对应Labview的IAP升级助手、波形设置助手上位机代码笔者已经调试通过,因为不想去水博客、凑数量,复制粘贴炒冷饭,所以导致整体上更新得比较慢。
一方面需要组织好语言描述、搭配好图片说明、安排好篇幅章节,另一方面对于比较重要的东西写完以后还会修改精炼,可能这就是CSDN排名上不去的原因,各方面原因导致有时候不能每周都保证更新一篇当然会尽力做到周更,往往一周没更新则需要几周连更才能回到之前的排名吧。
回归主题不管是撰写什么样的技术博客,笔者都更倾向于层层分章叙述,这可能也是在学习工作后养成的逻辑性吧,一般性地首先说明技术背景即这项技术通常应用在哪里;其次交代这项技术的原理,嵌入式研发一方面需要不断的项目迭代积累,另一方面也需要懂得些原理性的知识,否则在出故障的时候你如何去复现定位排查问题;再次会展开实施细节,在了解完背景应用和原理知识后,就到了具体的落地实施,怎么更好地去落地兼顾产品的稳定性、易用性、准确性和物料成本、开发周期、人力投入等本身就是一种权衡取其最优解;最后会把开发当中遇到的一些常见问题和注意事项列举出来,并给出自己的解决方法吧。
在这里笔者举三个现实落地的项目其技术处理上包含一些相似点主要在ADC采样这块,大家可以从中直观比较感受应对不同的现实需求,硬件选型和软件设计上也是相辅相成的:1. 程控交流电源项目需要把输出端的高压信号通过分压电阻和运放调理到ADC芯片的采样范围内,且频率范围是1-500hz的可设定正弦波,电压范围是1-300V高压信号,整体上要求比较高的采样精度,为了加快这款产品的开发速度,选取了STM32F405RGT6作为ADC的采集端芯片硬件上搭配了一片14位双通道SPI的ADC,这款芯片本身Flash较大可以存储Bootloader、校准表数据、App程序且主频168Mhz速度较快通过定时器重加载可以保证对各个频率段的正弦波信号进行等间隔采集,随着频率的提高采样点从512到128,再进行后期运算;2. 气体浓度检测项目需要把光源发出毫伏级的模拟电压信号通过运放调节到ADC芯片采样区间内,也是采样一个5hz固定频率的类正弦波信号,再去计算折合成当前某种气体的浓度,整个项目对精度要求非常高且对产品价格较为敏感,所以选择了2ksps较低采样速率的24位高精度ADC,搭配STM32F103RCT6对ADC进行每个周期1024个点等间隔采样计算还原等,整体硬件成本很低但保证浓度精度却很高;3. 天线单元测量项目需要把一个上Mhz不同频率叠加的正弦波进行滤波计算、FFT还原波形等,而且是有16路模拟信号需要处理,主控芯片选型上有几种应用方案:一是用ZYNQ但PL和PS等细节相关技术整体学习开发周期很长不易把控项目进度;二是用高端带DSP48资源丰富的FPGA芯片比如K7和V7等,用纯FPGA逻辑进行采样、计算、通信等,但开发周期会非常长,顶级的FPGA芯片如V7价格动则上万并且引脚也很多对PCB层数要求很高;三是用中低端的FPGA芯片如A7,搭配DSP或者STM32进行一些外扩的浮点数计算,整体硬件设计和软件开发上更容易衔接,关于ADC芯片的选型如果使用16个ADC去实时采样显然硬件成本会很高且产品市场竞争力很小,所以选择一款高性价比的16位4通道的LVDS接口的ADC,这样16路模拟信号用4个4通道ADC即可以完成采样,前端FPGA实时地切换不同的通道进行高速采样把16路的ADC得到的二进制数进行FIR滤波、FFT傅里叶变化等,后端选择主频216Mhz的STM32F746进行后期浮点数计算把ADC采样计算处理得到的二进制数转换成对应的角度数据、波距数据等等,前端选择XC7A100T-2FBG484I即中档A7芯片外接两颗DDR3颗粒进行ADC采集的数据缓存,在节约成本的同时也取得了非常精确的测量结果。
可能上面的内容其本身和STM32 IAP升级相关性并不大,但是写在这里主要帮助大家解除一些学习上的迷茫或者说写给曾经的自己,笔者在读研时候刚接触到STM32等一些MCU开发平台,跑过原子和野火的例程开发板demo后会感到不少迷茫,一方面会觉得通过学习这些市面上的主流开发板入门了STM32中常用的一些中断和配置等,但是另一方面具体在项目中怎么用,这样做好不好或者说对不对,稳定性和可靠性有没有保证等这些没有教程去手把手教学,而且看来看去就感觉网络上的STM32教程好像都是原子教程的改写版,比如IAP升级原子给出了STM32标准库的串口循环队列demo,然后几乎所有市面付费的教程千篇一律地这样写仿佛像考研计算机408统考一样全国考生都是一个老师教出来的,而在这背后可能真的大部分人只是在不断地复制粘贴,很少有人去静下心仔细思考或者亲手实践吧。
笔者想在第二篇博客里把IAP升级的原理说明清楚,第一篇博客则写的是IAP升级的项目背景,IAP(In Application Programming)即在应用编程, IAP 是指用户自己的程序在运行过程中对User Flash 的部分区域进行烧写,目的是为了在产品发布后可以方便地通过预留的通信口对产品汇总的固件程序进行更新升级。
然后我们再来看看如何去实现IAP,通常去实现IAP功能是需要用户程序可以在运行中实现独立更新自身Flash代码的目标。所以在设计固件程序时要编写两个项目代码,第一个项目程序不执行正常的功能操作,而只是通过某种通信方式比如USBTMC、LAN、USART接收数据执行对第二部分代码的更新,第二个项目代码才是所需要的真正功能代码。 这两部分项目代码都同时烧录在 User Flash 中,当STM32芯片上电后首先是第一个项目代码开始运行并做如下操作:
1)检查是否需要对第二部分代码进行更新
2)如果不需要更新则转到4)
3)执行更新操作
4)跳转到第二部分代码执行
第一部分代码必须通过其他手段,比如JTAG或ISP下载器烧入,而第二部分代码则可以既使用第一部分代码IAP功能烧入,也可以连同和第一部分代码一起烧入,以后需要程序更新时再通过第一部分IAP代码更新。
一般性地工程上我们将第一个项目代码称之为Bootloader程序,而第二个项目代码称之为Application程序,当然它们存放在Flash的不同地址范围,一般是从最低地址区开始存放 Bootloader,紧跟其后的就是Application程序,但是需要说明的是如果Flash容量足够大是可以设计很多Application程序的,但是在下文中我们只讨论一个Application程序的情况,下面笔者就STM32单片机分别举例说明普通程序(无Bootloader)的执行流程和IAP程序(有Bootloader)的执行流程。
普通程序执行流程,当STM32程序中只包含了Application程序则运行流程如下图1所示:
图1 STM32普通程序执行流程
STM32 的内部闪存Flash地址起始于0x08000000,一般情况下程序文件就从此地址开始写入。例程中笔者选择了 STM32F103RCT6是一种基于 Cortex-M3 内核的微控制器,其内部通过一张“中断向量表”来响应中断,程序启动后将首先从“中断向量表”取出复位中断向量执行复位中断程序完成启动,而这张 “中断向量表”的起始地址是0x08000004,当中断来临STM32 的内部硬件机制亦会自动将PC指针定位到“中断向量表”处,并根据中断源取出对应的中断向量再去执行中断服务程序。
在上1图中,STM32 在复位后,先从0x08000004地址取出复位中断向量的地址,并跳转到复位中断服务程序,如图标号①所示;在复位中断服务程序执行完之后,会跳转到程序中的main函数,如图标号②所示;而在裸机程序里的main函数一般都是一个死循环,在main函数执行过程中,如果收到中断请求即发生重中断,此时STM32强制将PC指针指回中断向量表处,如图标号③所示;然后根据中断源进入响应的中断服务程序,如图标号④所示;在执行完中断服务程序以后,程序再次返回main函数执行如图标号⑤所示。
而当加入 IAP 程序之后,程序运行流程如下图2所示:
图2 STM32 IAP程序执行流程
在上图2所示流程中,STM32复位后,还是从0x08000004地址取出复位中断向量的地址并跳转到复位中断服务程序,在运行复位中断服务程序之后跳转到IAP的main函数,如图标号①所示;在执行完IAP以后,即将新的Application代码写入STM32的Flash(图中灰色部分)。新的Application程序复位中断向量起始地址为0x08000004+N+M,跳转至新写入程序的复位向量表,取出新程序的复位中断向量的地址,并跳转执行新程序的复位中断服务程序,随后跳转至新程序的main函数,如图标号②和③所示,同样main函数为一个死循环,并且注意到此时STM32的Flash,在不同位置上共有两个中断向量表。
在main函数执行过程中,如果CPU得到一个中断请求,PC指针仍强制跳转到地址0x08000004中断向量表处,而不是新程序的中断向量表,如图标号④所示;程序再根据我们设置的中断向量表偏移量,跳转到对应中断源新的中断服务程序中,如图标号⑤所示;在执行完中断服务程序后,程序返回main函数继续运行,如图标号⑥所示。
通过以上两个过程的分析,可以知道IAP程序必须满足两个要求:
1)新的Application程序必须在IAP程序之后的某个偏移量为x 的地址开始;
2)必须将新的Application程序中断向量表相应的移动,移动的偏移量为x;
在介绍完IAP升级原理和STM32启动机制后,接着我们来看看IAP升级过程吧,STM32上电或者复位以后,首先执行Bootloader程序,Bootloader程序启动后,检查串口是否有升级数据,等待10秒钟无升级数据,则自动跳转到功能固件Application程序开始运行,如果收到升级指令则进入升级模式,直到接收完成全部升级数据,并将新的固件Application程序升级到对应的Flash区域,启动运行新固件程序升级过程结束,并在上位机上做出提示详细细节会在下篇博客具体说明。
最后笔者带着大家看一看STM32F103常用系列的Flash结构,STM32F103可以说是一款如同教科书般经典的MCU芯片,价格低廉且功能实用可以说能够胜任任何一款嵌入式产品的应用开发,IAP升级的本质就是STM32芯片通过Bootloader程序接到外部接口发来的BIN文件报文并将其准确写入Application对应的Flash地址中,所以搞清楚STM32F103内部的Flash结构则是IAP开发的关键性一步,就拿STM32F103R8T6来举例,其本身是STM32F103系列中较低端的型号内部具有64KB的Flash空间,其中STM32F103R8T6 要按照IAP升级Flash的结构可以分配如下:
STM32F103R8T6 Flash地址范围 :0x08000000 -- 0x08010000 (总大小64KB)
Bootloader代码区地址范围 :0x08000000 -- 0x08002000 (占用大小8KB)
功能固件代码区地址范围 :0x08002000 -- 0x08010000 (占用大小56KB)
如下表1所示是STM32F103常用系列的FLASH大小和扇区空间,笔者将其总结如下供大家进入STM32F103系列单片机IAP开发时参考,当然STM32F103系列单片机型号太多,下面表格只列举了常用型号的几种,用于描述Flash中Application固件和Bootloader固件存放的地址,便于用于理解程序,又因为STM32F103系列单片机,中等容量和大容量产品的扇区大小不一致,所以bootloader程序中也需要注意!
STM32F103C8T6 | 64KB | 64个扇区 | 每个扇区占1KB | 中等容量 |
STM32F103RBT6 | 128KB | 128个扇区 | 每个扇区占1KB | |
STM32F103RCT6 | 256KB | 128个扇区 | 每个扇区占2KB | 大容量 |
STM32F103ZET6 | 512KB | 256个扇区 | 每个扇区占2KB |
表1 STM32F103系列单片机常用型号的Flash扇区分配