网络公司手机网站,微商城分销系统制作,查钓鱼网站,wordpress设置图片大小--基于STM32F103ZET6的UART通讯实现 一、什么是IAP#xff0c;为什么要IAP IAP即为In Application Programming(在应用中编程)#xff0c;一般情况下#xff0c;以STM32F10x系列芯片为主控制器的设备在出厂时就已经使用J-Link仿真器将应用代码烧录了#xff0c;如果在设备使…--基于STM32F103ZET6的UART通讯实现 一、什么是IAP为什么要IAP IAP即为In Application Programming(在应用中编程)一般情况下以STM32F10x系列芯片为主控制器的设备在出厂时就已经使用J-Link仿真器将应用代码烧录了如果在设备使用过程中需要进行应用代码的更换、升级等操作的话则可能需要将设备返回原厂并拆解出来再使用J-Link重新烧录代码这就增加了很多不必要的麻烦。站在用户的角度来说就是能让用户自己来更换设备里边的代码程序而厂家这边只需要提供给用户一个代码文件即可。 而IAP却能很好的解决掉这个难题一片STM32芯片的Code(代码)区内一般只有一个用户程序。而IAP方案则是将代码区划分为两部分两部分区域各存放一个程序一个叫bootloader(引导加载程序)另一个较user application(用户应用程序)。bootloader在出厂时就固定下来了在需要变更user application时只需要通过触发bootloader对userapplication的擦除和重新写入即可完成用户应用的更换。如图1-1所示 图 1-1 在程序执行初始进入bootloader在bootloader里面检测条件是否被触发(可通过按键是否被按下、串口是否接收到特定的数据、U盘是否插入等等)如果有则进行对user application进行擦除和重新写入操作如果没有则直接跳转到user application执行应用如果有则进行擦除用户代码并重新写入新的用户代码。 二、STM32F103ZET6硬件条件 STM32F103ZET6的启动方式有三种内置FLASH启动、内置SRAM启动、系统存储器ROM启动通过BOOT0和BOOT1引脚的设置可以选择从哪中方式启动这里选择内置的FLASH启动。其FLASH的地址为0x08000000—0x0807ffff共512KB这些都能从芯片数据手册中直接得到。而这里首要的一个问题是中断的问题。正常情况下发生中断的过程为发生中断(中断请求)à到中断向量表查找中断函数入口地址à跳转到中断函数à执行中断函数à中断返回。也就是说在STM32的内置的Flash中有一个中断向量表来存放各个中断服务函数的入口地址内置Flash的分配情况大致如下图2-1。 图2-1 在只有一个程序的情况下程序执行的走向应该如图2-2所示(借用网友的原图)。 图2-2 STM32F10x有一个中断向量表这个中断向量表存放在代码开始部分的后4个字节处(即0x08000004)代码开始的4个字节存放的是堆栈栈顶的地址当发生中断后程序通过查找该表得到相应的中断服务程序入口地址然后再跳到相应的中断服务程序中执行。上电后从0x08000004处取出复位中断向量的地址然后跳转到复位中断程序的入口(标号①所示)执行结束后跳转到main函数中(标号②所示)。在执行main函数的过程中发生中断则STM32强制将PC指针指回中断向量表处(标号③所示)从中断向量表中找到相应的中断函数入口地址跳转到相应的中断服务函数(标号④所示)执行完中断函数后再返回到main函数中来(标号⑤所示)。 若在STM32F103x中使用IAP方案则内置的Flash分配情况大致如下图2-3。 图2-3
在内置的Flash里面添加一个BootLoader程序BootLoader程序和user application各有一个中断向量表假设BootLoader程序占用的空间为NM字节则程序的走向应该如图2-2所示(借用网友的原图并做改动其中虚线部分为原图步骤④⑤的走向本人改为指向灰色部分)。 图2-2 上电初始程序依然从0x08000004处取出复位中断向量地址执行复位中断函数后跳转到IAP的main(标号①所示)在IAP的main函数执行完成后强制跳转到0x08000004NM处(标号②所示)最后跳转到新的main函数中来(标号③所示)当发生中断请求后程序跳转到新的中断向量表中取出新的中断函数入口地址再跳转到新的中断服务函数中执行(标号④⑤所示)执行完中断函数后再返回到main函数中来(标号⑥所示)。 对于步骤④⑤网友认为是“在main执行的过程中如果CPU得到一个中断请求PC指针仍强制跳转到地址0x08000004中断向量表处而不是新的中断向量表如图标号④所示程序再根据我们设置的中断向量表偏移量跳转到对应中断源新的中断服务程序中如图标号⑤所示”。我对此的理解是“当发生中断后程序从0x08000004(旧)处的中断向量表中得到相应的中断服务函数入口地址继而跳转到相应的中断服务程序”。但是旧的中断向量列表里边存放的是IAP程序中断函数的入口地址它是如何得到user程序中断函数的入口地址呢所以我觉得此种说法是错误的。“当发生中断时PC指针强制会跳转到0x08000004处”这种说法并没有错只是忽略了后续的一些知识要点而导致这个说法出现矛盾。 对于步骤④⑤我认为的是在main函数的执行过程中如果CPU得到一个中断请求PC指针本来应该跳转到0x08000004处的中断向量表由于我们设置了中断向量表偏移量为NM因此PC指针被强制跳转到0x08000004NM处的中断向量表中得到相应的中断函数地址(待求证)再跳转到相应新的中断服务函数执行结束后返回到main函数中来。 三、实现过程 STM32F103ZET6的Flash地址为0x08000000—0x0807ffff共512KB把这512KB的空间分为两块第一块大小为32KB存放BootLoader程序剩余的空间存放用户程序(根据实际情况分配这两块空间的大小BootLoader程序占用的空间越小越好则BootLoader地址为0x08000000—0x08007fff用户程序地址为0x08008000—0x0807ffff。BootLoader流程图大致应该如下
1、初始化时钟。
2、初始化中断向量表地址。
3、初始化按键。 (使用按键触发方式上电时如果按键被按下则进行用户程序更新操作)
4、初始化串口。
5、检测按键是否被按下是则执行步骤6否则执行步骤10。
6、擦除用户程序(擦除0x08008000—0x0807ffff地址空间Flash)。
7、从串口读取新的用户代码数据把代码写入用户程序空间。
8、检测串口数据接收完毕是则执行步骤9否则跳回步骤7。
9、用户程序更新完毕等待重新上电或硬件复位。
10、跳转到用户程序(强制将PC指针跳转到0x080080004处)。 到这里首先要解决的问题就有
1、如何进行对STM32的Flash进行擦除和写入操作。
2、中断向量表偏移如何设置。
3、如何改变代码存放的地址空间(因为BootLoader要存放在0x08000000处用户程序要存放在0x08008000处,而默认的代码存放的地址空间为0x08000000)。
4、怎么进行PC指针的强制跳转跳转时需要做些什么。
5、串口接收的用户代码数据是什么样的代码数据是一种什么样的文件。 问题的解决
1、使用STM32的固件库函数只需调用几个库函数即可轻松解决使用的固件库为stm32f10x_flash.c文件对Flash的操作过程简要为Flash解锁àFlash擦除àFlash写入àFlash上锁。(对Flash编程的更详细操作参考STM32F10xxx闪存编程手册)
①解锁
FLASH_Unlock(); //解锁Flash
FLASH_SetLatency(FLASH_Latency_2); //因为系统时钟为72M所以要设置两个时钟周期的延时
②擦除
for(i0;i240;i)
{ if(FLASH_ErasePage(FLASH_ADDRi*2048) ! FLASH_COMPLETE) //一定要判断是否擦除成功 return ERROR;
}
说明FLASH_ErasePage(uint32_t Page_Address)即为Flash擦除操作按页擦除每页2KBPage_Address为页的起始地址如0x08000000是第一页起始地址0x08000800为第二页起始地址这里的操作擦除了0x08008000—0x0807ffff地址空间的Flash。
③写入
unsigned char buf[1024]; //假设待写入的代码数据
unsigned short temp; //临时数据
for(i0;i512;i)
{ temp (buf[2*i1]8) | buf[2*i]; //2个字节整合为1个半字 if(FLASH_ProgramHalfWord(ADDR,temp) ! FLASH_COMPLETE) //判断是否写入成功 { Return ERROR; }
ADDR 2; //地址要加2因为每次写入的是2个字节(1个半字)
}
说明因为STM32的Flash写入为双字节(1个半字)写入FLASH_ProgramHalfWord(uint32_t Address, uint16_t Data)函数即为对地址为Address写入1个半字的Data每次写入完成后地址要加2。
④上锁
FLASH_Lock(); //Flash 上锁一个固件库函数即可实现。 2、关于中断向量表的偏移设置对于BootLoader程序只需设置中断向量表的指向在0x08000000处对于用户程序需要设置中断向量表的指向在0x08008000处即可。
①在BootLoader程序的中断向量表指向设置中应有这么一句
NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0); //设置中断向量表指向
其中NVIC_VectTab_FLASH是个宏定义的值为0x08000000。
②在用户程序的中断向量表指向设置用应有这么一句
NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x8000); //设置中断向量表指向 3、确认代码存放的地址空间在IAR和在Keil中的设置是不同的网上有在Keil中设置的方法设立介绍在IAR软件环境下的设置方法。
①在固件库目录\STM32F10x_StdPeriph_Lib_V3.5.0\Project\STM32F10x_StdPeriph_Template\EWARM下找到一个stm32f10x_flash.icf文件将其复制到工程目录中来在打开IAR工程将配置文件添加到工程中如下图3-2所示 图3-1
②在工程中打开stm32f10x_flash.icf该文件修改两个参数即可改变代码存放的地址空间图下图3-2所示。 图3-2 4、关于PC指针的强制跳转想在BootLoader程序中将PC指针跳转到用户代码处可选择下面的操作
typedef void (*pFunction)(void);
pFunction Jump_To_Application;
uint32_t JumpAddress;
#define ApplicationAddress 0x08008000 if (((*(__IO uint32_t*)ApplicationAddress) 0x2FFE0000 ) 0x20000000) //--------①
{ JumpAddress *(__IO uint32_t*) (ApplicationAddress 4); //--------② Jump_To_Application (pFunction) JumpAddress; //--------③ __set_MSP(*(__IO uint32_t*) ApplicationAddress); //--------④ Jump_To_Application(); //--------⑤
}
①因为用户程序开始位置(0x08008000处)的前4个字节存放的是堆栈的地址堆栈地址必定是指向RAM空间的而STM32的RAM空间起始地址为0x20000000所以要进行判断。
②程序跳转地址的确认前面已经说过0x08008004处的4个字节存放的是复位函数的入口地址该句的意思为获得(ApplicationAddress 4)地址处的数据即为获得新的复位函数入口地址。
③令Jump_To_Application这个函数指针指向复位函数入口地址。
④堆栈的初始化重新设定栈顶代地址把栈顶地址设置为用户代码指向的栈顶地址。
⑤跳转到新的复位函数。 5、通过串口来接收代码数据就是PC机通过串口将代码数据发送到STM32中去。这里就涉及到两个问题
①数据怎么得来。
②数据传输的过程需要遵循的协议什么时候开始什么时候结束。
解决①一般我们就将*.hex文件使用JFlash-ARM打开再通过Jlink仿真器烧录到STM32芯片中但是*.hex文件里边包含的数据不纯粹是代码数据还有一些别的东西而*.bin文件数据就全部是代码数据。
在IAR软件环境中打开一个用户工程先设置好中断向量表偏移和代码存放的地址空间后(前面已介绍过这两种方法)。设置工程如下图3-3所示确认后重新编译工程在工程的\Debug\Exe目录下会相应生成一个xxx.bin文件这就是所需要的代码文件。 图3-3
②数据通过串口来传输文件常用的协议有XModem、YModem、ZModem这三种协议在PC端使用这些协议传输文件只需要PC的超级终端或者终端工具SecureCRT即可但是在STM32这边的编程会增加一些困难(因为要先去读懂、解析这些协议在通过编程来实现)。也可选择自己定义一套简单的传输协议但同样会有一些困难(因为要在PC端进行文件和串口编程)。总之不管通过什么办法都行只要能将xxx.bin文件数据通过串口全部发送到STM32并且STM32能够全部接收到这些数据并写入Flash即可(我选择后者自定义传输协议并用VC进行文件和串口编程)。 四、结束语 总的来说STM32的IAP方案实现需要在进行用户程序之前加一段Bootloader程序BootLoader程序的作用就是
①什么都不做直接跳转到用户程序。
②删除原有的用户程序读取*.bin文件数据并将数据重新写入新的用户程序。
对于用户程序相比普通的编程只需要做三步改动即可
①改变中断向量表。
②改变代码存放的地址空间
③修改生成*.bin文件 使用通过UART的IAP方案并不是很好的选择这只是IAP方案的一个机制因为能使用PC机通过串口升级程序同样能通过Jlink烧写程序并且自定义的串口通讯协议在没有校CRC校验的情况下不能及时发现数据传输过程发生的错误。这里推荐使用SD卡(或U盘)进行用户程序更新将*.bin文件复制到SD卡(或U盘)中STM32再通过读取SD卡(或U盘)的*.bin文件进行用户程序更新这也避免了STM32与PC笨重的通讯只需插一个SD卡(或U盘)更显得人性化一些但需要去弄懂STM32如何与SD卡(或U盘)的通讯。