
利用 USB DFU实现 IAP功能 1 n. L2 B6 j9 N前言 伴随着固件升级需求的增加,STM32提供了灵活的升级方式。本文一步一步介绍如何利用USB DFU Class以及ST提供的DfuSe demo软件工具实现IAP(In Application Program)功能,但并不涉及到DFU class移植。 + q! u ^% Y* G& V: e 一 实现环境 开发板:STM32F746G-DISCO 开发库:STM32CubeF7 v1.3.0 集成开发环境:IAR v7.70.1.11486 :MDK-ARM Plus v5.20.0.0 (只需两者中一种)' z3 ` n8 Y2 [; h B. \6 s 优化级别 : High (IAR) Level 3 (-O3) (MDK-ARM)5 ~, j! ]( ^) y. x U$ T DFU演示软件: Dfu file manager v3.0.5 DfuSeDemo v3.0.5 实现过程在STM32F7系列上展开,但USB DFU实现的IAP功能并不只局限于STM32F7系列,可以通过用户手册判断所采用微控制器型号是否支持USB以及IAP功能。STM32Cube开发库为各系列STM32提供了齐全的USB DFU例程可供参考。) n2 F) @5 b* m" q 二 IAP介绍! o& S; n. h6 C. h0 i+ c3 n; ^ IAP(In application program)主要为使用者提供了一种更加灵活的固件升级方式,可以根据应用需要定义何时、何种情况发生时进行固件升级。在介绍步骤前,需要对IAP原理有一定认识。结合下图(仅供参考,IAP实际应用方式更加灵活),IAP的应用中,用户程序与IAP驱动程序位于不同的存储区域。在应用过程中,利用IAP驱动程序将用户固件加载到固定位置,完成升级。更多关于IAP应用介绍请参考AN3965。 其中,IAP驱动程序首先烧录固化,并不会随着用户程序的升级而改变。用户固件升级完毕后,在IAP驱动程序引导下,跳转到用户固件对应Flash位置,执行用户程序。 * g- Z6 k+ H( A# L% I# c ![]() 6 _; J/ R5 x0 V 三 实现步骤& O% g9 X/ _& A, a8 m m - j2 L( }7 y' B6 x, ?, p. f5 | 3.1 DFU工具安装; s9 _8 W+ E- C7 [ 安装DFU demo工具DfuSe v3.0.5,主要提供了驱动、Dfu file manager 和 DfuSeDemo。关于DfuSe的详细内容(安装步骤、使用介绍)请参考UM0412。 3.2 IAP驱动固件生成与装载; G: w1 V3 a2 A( W- N+ A& N 利用STM32CubeF7 v1.3.0开发库提供的USB DFU例程生成IAP驱动固件(文件夹路径: …\ STM32Cube_FW_F7_V1.3.0\Projects\STM32746G-Discovery\Applications\USB_Device\ DFU_Standalone)。 利用烧录工具烧录到开发板中。本例程实现重启后,如果板上User按键按下或者没有有效的用户应用程序,则进入USB DFU模式,等待升级;否则,进入用户应用。4 E$ G& P) k e' d$ O# s% R 下面是摘取的部分例程。" D% }; G# F. Q/ i4 P! C) W5 E+ g$ @ + Q: v: s! A1 N- _1 T( @( l6 h ![]() - p' _, @' Z1 H+ w9 `1 | 用户固件对应的首地址被定义在0x08008000。用户也可以自定义地址,需要注意如下几点:% }9 M" [$ F8 T f; v- I 1. 禁止定义在0x08008000地址前(STM32F7系列,0x0~0x0800FFFF属于Sector0,已经开辟为IAP驱动程序区域) 2. 自定义地址所属Sector在升级时会全部擦除,即使定义地址并不位于对应Sector的首地址 3. 自定义地址需要保持512-byte对齐 4. 自定义地址需要与用户固件对应的装载地址以及用户中断向量表地址保持一致; v n" T& Y/ W% s, E6 ?6 E( V4 k: d' o 在实现跳转到用户程序的代码中,将用户中断向量表中第一个4字节指向的栈首地址分配给MSP。第二个4字节为复位中断向量,指向执行首地址。 在开发IAP驱动程序时,需要避免PC指针跳到用户程序区域。同时,充分考虑Stack & Heap大小,避免出现USB DFU正常识别,但不能正常工作情况,如下图所示。' x7 `7 R3 N. T4 R! U) Y 4 Q. b* W" s2 ~8 D2 s! ` ![]() 3.3 用户固件生成6 }" t& I ^/ W% B" X6 O# F 本文采用STM32CubeF7 v1.3.0开发库中TIM_TimeBase例程作为用户程序(文件夹路径:STM32Cube_FW_F7_V1.3.0\Projects\STM32746G-Discovery\Examples\TIM\TIM_TimeBase)。 IAP升级所需用户固件,主要有两处需要变动:% r% F* S0 `( O1 I& { 1. 在链接工具中,程序装载地址7 H i4 C% E) q, R! s 2. 中断向量表首地址" y, f2 J: A3 |) |4 g# j* L& I 上述两种变动需要保持一致,并且与IAP中定义的用户区域首地址保持一致。 本IAP驱动程序中,用户固件装载地址相对于0x08000000偏移地址为0x8000。在MDK中,改动前后比较如下图所示。- c5 j/ H" Y# @# m. G # F3 w% `# }% K ?, w ![]() 在IAR中,更改步骤如下所示。- y. N6 Y! R# n6 O( K& c( a5 y. m ! j( u3 E* l! A2 P7 Q ![]() H: M( r+ l0 z+ ]9 c% J 修改完毕后,生成hex文件(STM32746G_DISCOVERY.hex)。# r* m3 r0 C/ _+ y4 P0 y( ~. l , ^ c _% C- m- x& z/ P 3.4 用户固件转换与升级 打开Dfu file manager工具,点击OK,进入转换工具主界面。5 s4 |- O9 B' H7 p : {9 z5 i' X. l% ?& I5 d ![]() 按照下图步骤,首先载入生成的HEX文件(STM32746G_DISCOVERY.hex);然后在Target ID、 Target Name、 Device properties中自定义内容,这部分内容在DfuSe Demo工具中选择.dfu文件时会予以显示,并不影响烧录文件内容。最后,点击Generate完成HEX文件DFU文件转换。5 D; j: N( @. j& Y$ Z0 c8 m ! _7 D# G( _. M3 U% z. `2 K4 I ![]() 为STM32F746G-DISCO板供电,通过USB线将板上USB_FS接口与电脑相连。由于ST Discovery板已经加载了IAP应用程序,并且没有有效的用户应用程序,因此在IAP应用程序中,直接进入USB DFU模式,以供电脑识别USB DFU器件。正常识别情况如左下图。 右下图情况是由于没有正确装载Driver导致,可以通过右击“DFU in FS Mode”选择更新驱动程序软件\浏览计算机以查找驱动程序软件(驱动文件位于DfuSe v3.0.5安装目录\ DfuSe v3.0.5\Bin\ Driver)。 ![]() P2 d1 U7 y4 x& q3 u" r# Z& t 打开DfuSeDemo,结合下图步骤,点击’Choose’加载之前转换的.dfu文件;选择配置;点击’Update’完成擦除与下载;另外,可以通过点击’Verify’验证是否下载成功。更多关于DfuSe内容,请参考UM0412。0 \ h9 B3 D- ?0 @( z" r0 B ![]() u8 K5 c1 G3 ]+ ?9 i8 h. E 用户固件升级完成。重启后,正常进入用户应用。如果重启过程中,板上USER键按下,进入USB DFU模式,等待升级。 7 Z4 }# g! G! `+ c% K, m 四 小结& g& j, U/ `, L; ~& {3 ]. |: w 介绍了利用USB DFU实现IAP过程,以及实现过程中注意事项。在DfuSe v3.0.5安装目录中包含了必要的文档,介绍如何使用DfuSe工具,以及如何开发基于STMicroelectronics DFU方案的上位机应用。 ( y$ G+ \$ T+ w2 G 相关文档 AN4657 STM32 in-application programming (IAP) using the USART AN3965 使用 USART 实现STM32F40x/STM32F41x 的 IAP& m0 t h0 ?5 l9 t# k p UM0412 Getting started with DfuSe USB device firmware upgrade STMicroelectronics extension" }+ m0 ]9 y' x9 q4 S 9 H- x4 q! ~ U1 ]" L l 相关工具&链接/ R$ k8 ^. z0 S4 L9 R DfuSe_Demo http://www.st.com/content/st_com/en/products/development-tools/software-development-tools/stm32-software-development-tools/stm32-programmers/stsw-stm32080.html?dl=537587854391172693#! i9 d: {: n O+ _3 j STM32CubeF7 4 l' ^7 J1 v& | http://www.st.com/content/st_com/en/products/embedded-software/mcus-embedded-software/stm32-embedded-software/stm32cube-embedded-software/stm32cubef7.html! z1 X, u' o& ?$ C& X2 l4 U) @ 7 M: w! s( v! ~) @+ m 文档下载地址: https://www.stmcu.org.cn/document/detail/index/id-217183$ y# s0 o& j% [" U7 O + g9 x# Q# R! r' ~! Z9 X" F 实战经验汇总:: X5 n4 W; s3 o- F$ |8 l+ @ https://www.stmcu.org.cn/module/forum/thread-576401-1-1.html 5 e5 ?( o" v0 }. r5 b' |& ] |
STM32 USB HID键盘例程
最全USB HID开发资料,悉心整理一个月,亲自测试
刘氓兔的杂谈【001】-片上USB 高速PHY
【经验分享】在进行 USB CDC 类开发时,无法发送 64整数倍的数据
【源码】STLINK-V3MINI 高速USB仿真器,成功改刷【高速CMSIS-DAP】
在线直播|无需编写任何代码即可在STM32上实现USB-C Power Delivery
STM32 USB CDC 虚拟多串口
圈圈发布USB图书第二版有感,以及分享一些我学习USB过程...
USB Audio设计与实现
【MCU实战经验】+STM32F107的USB使用
在下在用STM32F103ZE实现USB DFU的时候出现以下疑惑:
1、在USB设备标识符中指出有1个接口,而所有的接口描述符都是通过一个宏实现8 J$ p( V7 s* V
#define USBD_DFU_IF_DESC(n) 0x09, /* bLength: Interface Descriptor size */ \; V+ z3 b6 h0 E
USB_DESC_TYPE_INTERFACE, /* bDescriptorType */ \
0x00, /* bInterfaceNumber: Number of Interface */ \0 r* A0 z6 ]! s
(n), /* bAlternateSetting: Alternate setting */ \
0x00, /* bNumEndpoints*/ \2 G( C" |$ f6 C% ?2 o, C
0xFE, /* bInterfaceClass: Application Specific Class Code */ \ _8 \" T* {- x3 ^3 s O, [
0x01, /* bInterfaceSubClass : Device Firmware Upgrade Code */ \7 d( F5 K: e2 c$ H
0x02, /* nInterfaceProtocol: DFU mode protocol */ \
USBD_IDX_INTERFACE_STR + (n) + 1 /* iInterface: Index of string descriptor */ \% Z" p: n3 e. C
但是在这个接口描述符中指出端点数为0,但是发现USB状态机里面用的都是端点4的状态,这里有点费解
2、在DFU上位机中有个“Leaves DFU mode”按键,用于退出DFU模式,但是总是在49%时出错,查阅资料发现网上有许多人出现此问题。测试发现,如果在点击“Leaves DFU mode”按键后,USB设备只要响应完返回状态这个请求后,把USB设备的USB接口从PC拔出,则上位机显示成功。
点评
你好,我的毕设课题就是跟这个相关的,可以麻烦你讲讲固件升级在客户中的应用情况吗?