STM32高级开发——使用DFU方案

李肖遥 2022-11-28 08:01
    关注、星标公众号,直达精彩内容

素材来源:https://blog.csdn.net/zhengyangliu123/article/details/78788815

整理:技术让梦想更伟大 | 李肖遥


什么是 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组件

  4. 设置DFU参数

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

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

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

  5. 最后的设置

    最后我们添加一个外部的按键作为触发单片机启动时进入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升级程序。

参考资料

如何使用CubeMx生成一个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

版权声明:本文来源网络,免费传达知识,版权归原作者所有。如涉及作品版权问题,请联系我进行删除。

‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧  END  ‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧

关注我的微信公众号,回复“加群”按规则加入技术交流群。


点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。

李肖遥 公众号“技术让梦想更伟大”,作者:李肖遥,专注嵌入式,只推荐适合你的博文,干货,技术心得,与君共勉。
评论
  • 在当今数字化时代,嵌入式系统无处不在,从我们手中的智能手机、智能手表,到汽车中的电子控制系统、工业自动化设备,嵌入式开发技术的发展历程宛如一部波澜壮阔的科技史诗,深刻地改变了我们的生活和社会的运作方式。萌芽阶段(20 世纪 60 - 70 年代)嵌入式开发的起源可以追溯到 20 世纪 60 年代。当时,计算机技术刚刚兴起,体积庞大且价格昂贵。为了满足特定设备的控制需求,工程师们开始尝试将计算机技术应用于一些专用系统中。这一时期,嵌入式系统的概念逐渐萌芽。1965 年,美国数字设备公司(DEC)推
    Jeffreyzhang123 2024-12-31 10:08 128浏览
  • 本文介绍Linux系统查看硬件配置及常用调试命令,方便开发者快速了解开发板硬件信息及进行相关调试。触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。查看DDR获取内存信息下面数字以MB为单位。root@ido:/# free -mtotalused free shared buff/cache availableMem:
    Industio_触觉智能 2024-12-31 10:01 70浏览
  • 全球领先的光学解决方案供应商艾迈斯欧司朗(瑞士证券交易所股票代码:AMS)近日宣布,于2024年6月26-28日携汽车智能照明和智能座舱解决方案及相关产品和技术,亮相位于昆山市花桥国际博览中心的第十九届汽车灯具产业发展技术论坛暨上海国际汽车灯具展览会(以下简称:ALE),展示艾迈斯欧司朗优质的光发射器、光学元件、微型模组、传感器等产品在汽车领域的创新应用,展位号B馆T202。乘着车灯产业盛会的东风,全国汽车标准化技术委员会灯具及灯光分技术委员会秘书长卜伟理, 蔚来汽车内外车灯团队专家及电气工程副
    艾迈斯欧司朗 2024-12-30 16:55 69浏览
  • 时源芯微EMC(电磁兼容)领域,充斥着众多专业术语,令人眼花缭乱。1 电磁兼容(EMC)定义:指电气装置或系统在共同的电磁环境条件下,既能保持正常功能,又不会对周围环境产生不良影响。2  电磁环境Electromagnetic Environment定义:指存在于某一特定场所的所有电磁现象的总和。3半电波暗室Semi-anechoic Chamber定义:一种屏蔽室,除地面安装反射接地平板外,其余内表面均覆盖有吸波材料。4 远场Far Field定义:在天线辐射的场域中,功率密度与距离
    时源芯微 2024-12-31 10:52 111浏览
  • 近日,紫光展锐正式推出高性能4G 智能穿戴平台W377E。该产品面向不同的应用场景,拥有丰富特性和超低功耗,进一步壮大紫光展锐的智能穿戴产品组合。紫光展锐面向中高端和海量的智能穿戴市场,持续提供技术先进、高效安全、高质可靠的解决方案。 紫光展锐W377E采用四核A53架构,支持13M摄像头,video 1080P 30fps,集成了Wi-Fi、蓝牙和GPS功能。W377E搭载Android 8.1系统,提供更加丰富多样的APP体验,支持在线好友聊天、支付宝等功能,且支持64位系统的AP
    紫光展锐 2024-12-31 16:26 122浏览
  • 先临三维普及化手持3D扫描仪Einstar搭载艾迈斯欧司朗SFH 4726AS红外LED,打造真彩扫描、人眼安全、超高性价比的照明解决方案;得益于双堆叠发射器技术和透明硅树脂封装,OSLON® Black系列的SFH 4726AS实现小尺寸、高功率、高效率和优化的热管理,有效减小散热设计的压力;采用3.75×3.75mm的透明硅树脂封装,内置1mm2堆叠红外芯片,单颗光功率可达2W以上,为手持扫描仪这样小巧空间的应用提供更多光源设计空间。全球领先的光学解决方案供应商艾迈斯欧司朗(瑞士证券交易所股
    艾迈斯欧司朗 2024-12-30 16:50 61浏览
  • 根据Global Info Research项目团队最新调研,预计2030年全球CAN-Bus总线电缆产值达到2287百万美元,2024-2030年期间年复合增长率CAGR为3.6%。一、行业概述CAN-Bus总线电缆是一种专门用于CAN总线(Controller Area Network,控制器局域网络)通信系统中的电缆。CAN总线是一种广泛应用于汽车、工业自动化、机器人等领域的串行通信协议,主要用于微控制器和设备之间的数据交换。CAN-Bus电缆负责在多个设备间传输数据,保证信息的稳定传输和
    GIRtina 2024-12-31 11:00 102浏览
  • 感光百科:4000万片出货量背后的“技术经”目前,仅单点dToF,艾迈斯欧司朗的累积出货量就已超过了4000万片。大量采用的背后必然代表无数用户对这一技术的认可和信赖。究竟是什么魔力,让dToF传感器拥有如此强大的吸引力?又是怎样的know-how积累让艾迈斯欧司朗的dToF产品如此受到青睐?让我们再次回到底层原理来探究dToF 4000万片出货量背后的“技术经”。01、底层原理决定应用上限dToF传感器,顾名思义,直接飞行时间测量,基于光速不变,测量光子从发射端到接收端的光程差,从而直接计算二
    艾迈斯欧司朗 2024-12-30 18:21 94浏览
  • “新时代,共享未来”2024年11月5日-10日,第七届中国国际进口博览会在上海圆满落幕。其中,3万平方米的汽车展区展览面积,也让观众笑谈进博会再次进入“车展”时间。全球15大整车品牌,40多家企业参展,其中不乏耳熟能详,七届进博会“全勤生”的国际知名OEM,也会有首次参加的“新面孔”,它们共同构成了今年汽车展区的“全家福”。近年来,中国汽车进口均超百万辆。因此,对跨国车企来说,中国市场的重要性不言而喻。同时,在电动化和智能化赛道中弯道超车的中国汽车市场,也是全球汽车行业竞争最为激烈的市场,在这
    艾迈斯欧司朗 2024-12-30 17:05 59浏览
  • 2024年12月30日 调研咨询机构环洋市场咨询出版的《全球CPU渲染行业总体规模、主要厂商及IPO上市调研报告,2024-2030》主要调研全球CPU渲染总体规模、主要地区及国家的市场规模、主要企业规模和份额、主要细分市场规模、下游主要应用规模以及未来发展前景预测。统计维度包括收入、市场份额以及增速。同时也重点分析全球市场主要厂商(品牌)包括企业简介、总部、产地、CPU渲染产品介绍、规格/型号、收入、毛利率及市场份额、最新发展动态、优势与不足。历史数据为2019至2023年,预测数据为2024
    GIRtina 2024-12-30 16:36 127浏览
  • 高精度,抗干扰,设计灵活……一句话总结:打铁还需自身硬!2040年,人形机器人的数量或将超过人类,达到100亿个。 10月29日,沙特投资倡议论坛,在谈及AI和人形机器人赛道时,马斯克再一次大胆预测,描绘出上述令人震撼的未来景象,且预估每台机器人的价格将稳定在2万-2.5万美元之间。当下,技术飞速发展,众多机器人公司如雨后春笋般涌现。余建华,艾迈斯欧司朗资深应用工程师,深感对这个市场的期待,他表示每一个机器人关节的背后,都离不开位置传感器的精准掌控。这片市场的潜力,让人憧憬不已。在工业及消费中
    艾迈斯欧司朗 2024-12-30 17:25 55浏览
  •         在之前的文章中,我们介绍了IEEE 802.3cz[1]协议,MultiGABSE-AU物理层中XMII、PCS子层以及两个可选功能的相关内容,本篇将介绍MultiGABSE-AU物理层PMA子层及PMD子层的相关机制。PMA子层        PMA子层位于PCS子层和PMD子层之间,规范中定义了PMA子层的三个功能:PMA Transmit、PMA Receive及PHY Control。&nbs
    经纬恒润 2024-12-30 18:16 104浏览
  • 产品概述MG600Q2YMS3 是一款基于硅基碳化物(SiC)技术的高功率N沟道MOSFET模块,适用于高功率开关和电机控制应用,如轨道牵引系统。其设计旨在满足高效能和快速切换需求,为工业和能源领域提供可靠解决方案。主要特性1. 高电压和电流能力   耐压 (VDSS):1200 V   漏极电流 (ID):600 A2. 高效率与低损耗   碳化硅材料降低导通损耗和开关损耗,实现更高的转换效率。3. 快速切换性能 支持高频操作,适用于
    东芝铠侠代理 2024-12-31 10:33 74浏览
  •  随着电子技术的进步,电路中的隔离需求日益增加。晶体管光耦作为一种非接触式信号传输器件,因其独特的隔离特性和可靠性,成为了现代电子设备和工业控制中不可或缺的元件。本文将带您深入了解晶体管光耦的结构、工作原理和核心特点。晶体管光耦的基本结构晶体管光耦,通常由两个主要部分组成:发光二极管(LED)和光敏晶体管。其物理结构简单却极具创新性:发光二极管(LED):电流通过时,LED会发出红外光。光敏晶体管:接收来自LED的红外光信号,从而引导电流在输出端导通。这种结构的最大特点在于输入与输出端
    晶台光耦 2024-12-30 16:15 105浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦