STM32高阶应用--使用DFU方案实现固件升级

一起学嵌入式 2023-11-23 08:26

扫描关注一起学嵌入式,一起学习,一起成长

什么是 DFU

DFU全称为Device Firmware update,是ST官方推出的一个通过USB接口进行IAP升级的方案,同串口ISP一样,他们都集成在了芯片内部的Bootloader区段,可以通过配置boot引脚来启动。(具体可参照ST文档:AN2606)。

不过内置DFU的芯片大部分型号都比较新,如果你用的型号没有内置DFU程序,没关系我们也可以通过CubeMX来快速生成和移植一个DFU功能程序到你的Flash中来使用。

DFU方案完整的组件包括单片机DFU Demo代码、PC端升级程序、PC端Demo代码以及相关资料手册等。

通过使用DFU方案,我们可以快速的集成升级功能到开发的产品中,同时还能够快速的开发与之配套的升级程序。

使用CubeMX生成初始工程

由于官方提供的DFU例程并不多,我们很难找到现成的可已使用DFU程序,但是通过CubeMX我们可以很快速的配置和生成DFU的Bootloader,下面我们正式开始。

  1. 新建CubeMX工程

    首先选定好IC的型号,进入配置界面,由于只是Bootloader代码所以这里我们只需要配置USB功能和一个做Bootloader触发的引脚就可,其余的时钟等部分一切按照正常方式配置。

  2. 设置USB引脚功能

    设定USB模式为Device(HS还是FS并不影响DFU的功能,按照应用选择就可)。

  3. 开启DFU组件

    在MiddleWares中加入USB DFU组件

    设置DFU参数

    开启DFU组件后,CubeMX的程序设置窗口的MiddleWares中就会出现DFU程序设置按钮。

    点开它将APP加载的地址改为0x0800_c000,这个加载地址根据你实际的应用设置,目前我们选择让flash的前三个sector为Bootloader的区域。

    第二个全是字段的参数是用来在DFU连接升级软件式传输给软件用来获取Flash结构字符串数据,很好理解这个小协议的内容,点击设置后,下方的CubeMX的参数说明也写的很清晰,这里就不多说了。
    当然这些参数也在工程生成后在 usbd_conf.h 和 usbd_dfu_if.c 文件中修改。
  4. 最后的设置

    最后我们添加一个外部的按键作为触发单片机启动时进入DFU的方式,按键按下后就启动DFU模式,否则直接加载后方APP程序,这里选用PA0引脚,给它设置个User Label 就叫 USER_BTN_GPIO_Port。

修改补全工程

  1. 实现 DFU 功能代码

    打开 src 目录下的 usbd_dfu_if.c 文件补全其中的功能代码

    Flash 解锁

    uint16_t MEM_If_Init_HS(void)
    {
        HAL_FLASH_Unlock();
        return (USBD_OK);
    }

    Flash 上锁

    uint16_t MEM_If_DeInit_HS(void)
    {
        HAL_FLASH_Lock();
        return (USBD_OK);
    }

    Flash 擦除

    static uint32_t GetSector(uint32_t Address)
    {
         uint32_t sector = 0;

         if ((Address < ADDR_FLASH_SECTOR_1) && (Address >= ADDR_FLASH_SECTOR_0))
         {
            sector = FLASH_SECTOR_0;
         }

         ......

          }
          else if ((Address < ADDR_FLASH_SECTOR_23) && (Address >= ADDR_FLASH_SECTOR_22))
          {
            sector = FLASH_SECTOR_22;
          }
          else
          {
            sector = FLASH_SECTOR_23;
          }
          return sector;
        }

        uint16_t MEM_If_Erase_HS(uint32_t Add)
        
    {
          uint32_t startsector = 0;
          uint32_t sectornb = 0;
          /* Variable contains Flash operation status */
          HAL_StatusTypeDef status;
          FLASH_EraseInitTypeDef eraseinitstruct;

          /* Get the number of sector */
          startsector = GetSector(Add);

          eraseinitstruct.TypeErase = FLASH_TYPEERASE_SECTORS;
          eraseinitstruct.VoltageRange = FLASH_VOLTAGE_RANGE_3;
          eraseinitstruct.Sector = startsector;
          eraseinitstruct.NbSectors = 1;
          status = HAL_FLASHEx_Erase(&eraseinitstruct, §ornb);

          if (status != HAL_OK)
          {
            return (USBD_FAIL);
          }
          return (USBD_OK);
    }

    Flash 写入

    uint16_t MEM_If_Write_HS(uint8_t *src, uint8_t *dest, uint32_t Len)
    {
        uint32_t i = 0;

        for (i = 0; i < Len; i += 4)
        {
            /* Device voltage range supposed to be [2.7V to 3.6V], the operation will
           be done by byte */

            if (HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, (uint32_t)(dest + i), *(uint32_t *)(src + i)) == HAL_OK)
            {
              /* Check the written value */
              if (*(uint32_t *)(src + i) != *(uint32_t *)(dest + i))
              {
                /* Flash content doesn't match SRAM content */
                return (USBD_FAIL);
              }
            }
            else
            {
              /* Error occurred while writing data in Flash memory */
              return (USBD_FAIL);
            }
        }
        return (USBD_OK);
    }

    Flash 读取

    uint8_t *MEM_If_Read_HS(uint8_t *src, uint8_t *dest, uint32_t Len)
    {
          /* Return a valid address to avoid HardFault */
          uint32_t i = 0;
          uint8_t *psrc = src;

          for (i = 0; i < Len; i++)
          {
            dest[i] = *psrc++;
          }
          /* Return a valid address to avoid HardFault */
          return (uint8_t *)(dest);
    }

    获取 Flash 擦写时间参数

    uint16_t MEM_If_GetStatus_HS(uint32_t Add, uint8_t Cmd, uint8_t *buffer)
    {
          /* USER CODE BEGIN 11 */
          uint16_t time;

          time = TimingTable[GetSector(Add)];

          switch (Cmd)
          {
          case DFU_MEDIA_PROGRAM:
            buffer[1] = (uint8_t)time;
            buffer[2] = (uint8_t)(time << 8);
            buffer[3] = 0;
            break;

              case DFU_MEDIA_ERASE:
              default:
            buffer[1] = (uint8_t)time;
            buffer[2] = (uint8_t)(time << 8);
            buffer[3] = 0;
            break;
          }
          return (USBD_OK);
          /* USER CODE END 11 */
    }

    usbd_dfu_if.h 文件添加的宏定义

    /* Define flash address */
    // BLANK 1

    #define ADDR_FLASH_SECTOR_0 0x08000000


    #define ADDR_FLASH_SECTOR_1 0x08004000


    #define ADDR_FLASH_SECTOR_2 0x08008000


    #define ADDR_FLASH_SECTOR_3 0x0800C000


    #define ADDR_FLASH_SECTOR_4 0x08010000


    #define ADDR_FLASH_SECTOR_5 0x08020000


    #define ADDR_FLASH_SECTOR_6 0x08040000


    #define ADDR_FLASH_SECTOR_7 0x08060000


    #define ADDR_FLASH_SECTOR_8 0x08080000


    #define ADDR_FLASH_SECTOR_9 0x080A0000


    #define ADDR_FLASH_SECTOR_10 0x080C0000


    #define ADDR_FLASH_SECTOR_11 0x080E0000

    // BLANK 2

    #define ADDR_FLASH_SECTOR_12 0x08100000


    #define ADDR_FLASH_SECTOR_13 0x08104000


    #define ADDR_FLASH_SECTOR_14 0x08108000


    #define ADDR_FLASH_SECTOR_15 0x0810C000


    #define ADDR_FLASH_SECTOR_16 0x08110000


    #define ADDR_FLASH_SECTOR_17 0x08120000


    #define ADDR_FLASH_SECTOR_18 0x08140000


    #define ADDR_FLASH_SECTOR_19 0x08160000


    #define ADDR_FLASH_SECTOR_20 0x08180000


    #define ADDR_FLASH_SECTOR_21 0x081A0000


    #define ADDR_FLASH_SECTOR_22 0x081C0000


    #define ADDR_FLASH_SECTOR_23 0x081E0000


    /* Flash oprate time from datasheet page 128 */

    #define FLASH_SECTOR_16KB_WRITE_ERASE_TIME 500       //500 usb frame,means 500ms


    #define FLASH_SECTOR_64KB_WRITE_ERASE_TIME 1100


    #define FLASH_SECTOR_128KB_WRITE_ERASE_TIME 2000
  2. 修改Main文件

    首先在main文件前添加几个用于加载APP程序的变量和函数定义

    typedef void (*pFunction)(void);

    pFunction JumpToApplication;
    uint32_t JumpAddress;1234

    然后再 main 函数中加入 外部按键的判断、APP程序加载以及USB DFU初始化功能

    int main(void)
    {
          /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
          HAL_Init();

          /* Configure the system clock */
          SystemClock_Config();

          /* Initialize all configured peripherals */
          MX_GPIO_Init();

          if (HAL_GPIO_ReadPin(USER_BTN_GPIO_Port, USER_BTN_Pin) == GPIO_PIN_SET)
          {
            HAL_GPIO_WritePin(GPIOG, LD3_Pin, GPIO_PIN_SET); // For debug
            /* Test if user code is programmed starting from address 0x0800C000 */
            if (((*(__IO uint32_t *)USBD_DFU_APP_DEFAULT_ADD) & 0x2FF80000) == 0x20000000)
            {
              HAL_GPIO_WritePin(GPIOG, LD4_Pin, GPIO_PIN_SET); // For debug
              /* Jump to user application */
              JumpAddress = *(__IO uint32_t *)(USBD_DFU_APP_DEFAULT_ADD + 4);
              JumpToApplication = (pFunction)JumpAddress;

              /* Reset of all peripherals */
              HAL_DeInit();

              /* Set interrupt vector to app code */
              SCB->VTOR = USBD_DFU_APP_DEFAULT_ADD;

              /* Initialize user application's Stack Pointer */
              __set_MSP(*(__IO uint32_t *)USBD_DFU_APP_DEFAULT_ADD);
              JumpToApplication();
            }
          }

          MX_USB_DEVICE_Init();

          while (1)
          {
          }
    }
  3. 编译程序下载进入单片机

使用DfuSe

从ST官网DfuSe的程序安装包,并安装。然后我们按下之前写的触发按键并复位单片机,让单片机初始 USB DFU 功能,这时如果你插着单片机的USB线,系统应该已经识别了。

如果没有右键更新驱动程序,手动指定驱动搜索路径在DfuSe安装目录下的 \Bin\Driver 内。如果直接无法识别USB设备,建议在CubeMx配置完工程后就编译下载测试一下,看看是不是你在移植过程中哪里写错了。

然后我们需要生成一个地址设定在0x0800_c000后的测试程序,就先编写一个 Blink LED 的程序吧,生成bin、hex或S19文件。

然后我们打开DfuSe软件的Dfu file manager来生成DFU软件用的.dfu格式的文件。选择第一项,第二个是用来将.dfu反向变换回来的。

大概的操作已经标在图片上了,操作比较简单就不做详细介绍了,记得把Address的地址改到偏移后的地址上否则下载会出错,其他参数可以不用修改。

然后我们打开DfuSe程序,在Upgrade中选择生成好的blink.dfu文件,勾选校验功能,下载程序。成功后复位单片机,LED开始闪烁,移植成功。

更多

仔细区看看DfuSe的安装目录,里面有DFU的资料文档,还有DFU的工程源代码,可以用来改写自己需要的DFU升级程序。

参考资料

ST官网DfuSe

http://www.stmicroelectronics.com.cn/content/st_com/zh/products/development-tools/software-development-tools/stm32-software-development-tools/stm32-programmers/stsw-stm32080.html

原文:https://blog.csdn.net/zhengyangliu123/article/details/78788815

文章来源于网络,版权归原作者所有,如有侵权,请联系删除。



关注【一起学嵌入式】,回复加群进技术交流群。



觉得文章不错,点击“分享”、“”、“在看” 呗!

一起学嵌入式 公众号【一起学嵌入式】,RTOS、Linux编程、C/C++,以及经验分享、行业资讯、物联网等技术知
评论
  • 11-29学习笔记11-29学习笔记习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记
    youyeye 2024-12-02 23:58 73浏览
  • 学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习笔记&记录学习习笔记&记学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&学习学习笔记&记录学习学习笔记&记录学习学习笔记&记录学习学习笔记&
    youyeye 2024-11-30 14:30 78浏览
  • 艾迈斯欧司朗全新“样片申请”小程序,逾160种LED、传感器、多芯片组合等产品样片一触即达。轻松3步完成申请,境内免费包邮到家!本期热荐性能显著提升的OSLON® Optimal,GF CSSRML.24ams OSRAM 基于最新芯片技术推出全新LED产品OSLON® Optimal系列,实现了显著的性能升级。该系列提供五种不同颜色的光源选项,包括Hyper Red(660 nm,PDN)、Red(640 nm)、Deep Blue(450 nm,PDN)、Far Red(730 nm)及Ho
    艾迈斯欧司朗 2024-11-29 16:55 174浏览
  • TOF多区传感器: ND06   ND06是一款微型多区高集成度ToF测距传感器,其支持24个区域(6 x 4)同步测距,测距范围远达5m,具有测距范围广、精度高、测距稳定等特点。适用于投影仪的无感自动对焦和梯形校正、AIoT、手势识别、智能面板和智能灯具等多种场景。                 如果用ND06进行手势识别,只需要经过三个步骤: 第一步&
    esad0 2024-12-04 11:20 58浏览
  •         温度传感器的精度受哪些因素影响,要先看所用的温度传感器输出哪种信号,不同信号输出的温度传感器影响精度的因素也不同。        现在常用的温度传感器输出信号有以下几种:电阻信号、电流信号、电压信号、数字信号等。以输出电阻信号的温度传感器为例,还细分为正温度系数温度传感器和负温度系数温度传感器,常用的铂电阻PT100/1000温度传感器就是正温度系数,就是说随着温度的升高,输出的电阻值会增大。对于输出
    锦正茂科技 2024-12-03 11:50 111浏览
  • 《高速PCB设计经验规则应用实践》+PCB绘制学习与验证读书首先看目录,我感兴趣的是这一节;作者在书中列举了一条经典规则,然后进行详细分析,通过公式推导图表列举说明了传统的这一规则是受到电容加工特点影响的,在使用了MLCC陶瓷电容后这一条规则已经不再实用了。图书还列举了高速PCB设计需要的专业工具和仿真软件,当然由于篇幅所限,只是介绍了一点点设计步骤;我最感兴趣的部分还是元件布局的经验规则,在这里列举如下:在这里,演示一下,我根据书本知识进行电机驱动的布局:这也算知行合一吧。对于布局书中有一句:
    wuyu2009 2024-11-30 20:30 124浏览
  • 戴上XR眼镜去“追龙”是种什么体验?2024年11月30日,由上海自然博物馆(上海科技馆分馆)与三湘印象联合出品、三湘印象旗下观印象艺术发展有限公司(下简称“观印象”)承制的《又见恐龙》XR嘉年华在上海自然博物馆重磅开幕。该体验项目将于12月1日正式对公众开放,持续至2025年3月30日。双向奔赴,恐龙IP撞上元宇宙不久前,上海市经济和信息化委员会等部门联合印发了《上海市超高清视听产业发展行动方案》,特别提到“支持博物馆、主题乐园等场所推动超高清视听技术应用,丰富线下文旅消费体验”。作为上海自然
    电子与消费 2024-11-30 22:03 98浏览
  • 光伏逆变器是一种高效的能量转换设备,它能够将光伏太阳能板(PV)产生的不稳定的直流电压转换成与市电频率同步的交流电。这种转换后的电能不仅可以回馈至商用输电网络,还能供独立电网系统使用。光伏逆变器在商业光伏储能电站和家庭独立储能系统等应用领域中得到了广泛的应用。光耦合器,以其高速信号传输、出色的共模抑制比以及单向信号传输和光电隔离的特性,在光伏逆变器中扮演着至关重要的角色。它确保了系统的安全隔离、干扰的有效隔离以及通信信号的精准传输。光耦合器的使用不仅提高了系统的稳定性和安全性,而且由于其低功耗的
    晶台光耦 2024-12-02 10:40 120浏览
  • 概述 说明(三)探讨的是比较器一般带有滞回(Hysteresis)功能,为了解决输入信号转换速率不够的问题。前文还提到,即便使能滞回(Hysteresis)功能,还是无法解决SiPM读出测试系统需要解决的问题。本文在说明(三)的基础上,继续探讨为SiPM读出测试系统寻求合适的模拟脉冲检出方案。前四代SiPM使用的高速比较器指标缺陷 由于前端模拟信号属于典型的指数脉冲,所以下降沿转换速率(Slew Rate)过慢,导致比较器检出出现不必要的问题。尽管比较器可以使能滞回(Hysteresis)模块功
    coyoo 2024-12-03 12:20 111浏览
  • 当前,智能汽车产业迎来重大变局,随着人工智能、5G、大数据等新一代信息技术的迅猛发展,智能网联汽车正呈现强劲发展势头。11月26日,在2024紫光展锐全球合作伙伴大会汽车电子生态论坛上,紫光展锐与上汽海外出行联合发布搭载紫光展锐A7870的上汽海外MG量产车型,并发布A7710系列UWB数字钥匙解决方案平台,可应用于数字钥匙、活体检测、脚踢雷达、自动泊车等多种智能汽车场景。 联合发布量产车型,推动汽车智能化出海紫光展锐与上汽海外出行达成战略合作,联合发布搭载紫光展锐A7870的量产车型
    紫光展锐 2024-12-03 11:38 101浏览
  • 最近几年,新能源汽车愈发受到消费者的青睐,其销量也是一路走高。据中汽协公布的数据显示,2024年10月,新能源汽车产销分别完成146.3万辆和143万辆,同比分别增长48%和49.6%。而结合各家新能源车企所公布的销量数据来看,比亚迪再度夺得了销冠宝座,其10月新能源汽车销量达到了502657辆,同比增长66.53%。众所周知,比亚迪是新能源汽车领域的重要参与者,其一举一动向来为外界所关注。日前,比亚迪汽车旗下品牌方程豹汽车推出了新车方程豹豹8,该款车型一上市就迅速吸引了消费者的目光,成为SUV
    刘旷 2024-12-02 09:32 119浏览
  • 作为优秀工程师的你,已身经百战、阅板无数!请先醒醒,新的项目来了,这是一个既要、又要、还要的产品需求,ARM核心板中一个处理器怎么能实现这么丰富的外围接口?踌躇之际,你偶阅此文。于是,“潘多拉”的魔盒打开了!没错,USB资源就是你打开新世界得钥匙,它能做哪些扩展呢?1.1  USB扩网口通用ARM处理器大多带两路网口,如果项目中有多路网路接口的需求,一般会选择在主板外部加交换机/路由器。当然,出于成本考虑,也可以将Switch芯片集成到ARM核心板或底板上,如KSZ9897、
    万象奥科 2024-12-03 10:24 68浏览
  • RDDI-DAP错误通常与调试接口相关,特别是在使用CMSIS-DAP协议进行嵌入式系统开发时。以下是一些可能的原因和解决方法: 1. 硬件连接问题:     检查调试器(如ST-Link)与目标板之间的连接是否牢固。     确保所有必要的引脚都已正确连接,没有松动或短路。 2. 电源问题:     确保目标板和调试器都有足够的电源供应。     检查电源电压是否符合目标板的规格要求。 3. 固件问题: &n
    丙丁先生 2024-12-01 17:37 102浏览
  • 国产光耦合器正以其创新性和多样性引领行业发展。凭借强大的研发能力,国内制造商推出了适应汽车、电信等领域独特需求的专业化光耦合器,为各行业的技术进步提供了重要支持。本文将重点探讨国产光耦合器的技术创新与产品多样性,以及它们在推动产业升级中的重要作用。国产光耦合器创新的作用满足现代需求的创新模式新设计正在满足不断变化的市场需求。例如,高速光耦合器满足了电信和数据处理系统中快速信号传输的需求。同时,栅极驱动光耦合器支持电动汽车(EV)和工业电机驱动器等大功率应用中的精确高效控制。先进材料和设计将碳化硅
    克里雅半导体科技 2024-11-29 16:18 180浏览
  • 遇到部分串口工具不支持1500000波特率,这时候就需要进行修改,本文以触觉智能RK3562开发板修改系统波特率为115200为例,介绍瑞芯微方案主板Linux修改系统串口波特率教程。温馨提示:瑞芯微方案主板/开发板串口波特率只支持115200或1500000。修改Loader打印波特率查看对应芯片的MINIALL.ini确定要修改的bin文件#查看对应芯片的MINIALL.ini cat rkbin/RKBOOT/RK3562MINIALL.ini修改uart baudrate参数修改以下目
    Industio_触觉智能 2024-12-03 11:28 87浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦