0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

使用DFU方案實現(xiàn)STM32單片機的高級開發(fā)

技術讓夢想更偉大 ? 來源:csdn ? 作者:ArthurZheng150 ? 2022-11-28 09:33 ? 次閱讀

什么是 DFU

DFU全稱為Device Firmware update,是ST官方推出的一個通過USB接口進行IAP升級的方案,同串口ISP一樣,他們都集成在了芯片內部的Bootloader區(qū)段,可以通過配置boot引腳來啟動。(具體可參照ST文檔:AN2606)。不過內置DFU的芯片大部分型號都比較新,如果你用的型號沒有內置DFU程序,沒關系我們也可以通過CubeMX來快速生成和移植一個DFU功能程序到你的Flash中來使用。

DFU方案完整的組件包括單片機DFU Demo代碼、PC端升級程序、PC端Demo代碼以及相關資料手冊等。通過使用DFU方案,我們可以快速的集成升級功能到開發(fā)的產品中,同時還能夠快速的開發(fā)與之配套的升級程序。

使用CubeMX生成初始工程

由于官方提供的DFU例程并不多,我們很難找到現(xiàn)成的可已使用DFU程序,但是通過CubeMX我們可以很快速的配置和生成DFU的Bootloader,下面我們正式開始。

新建CubeMX工程

首先選定好IC的型號,進入配置界面,由于只是Bootloader代碼所以這里我們只需要配置USB功能和一個做Bootloader觸發(fā)的引腳就可,其余的時鐘等部分一切按照正常方式配置。

設置USB引腳功能

設定USB模式為Device(HS還是FS并不影響DFU的功能,按照應用選擇就可)。

cae4ea6a-6eb1-11ed-8abf-dac502259ad0.png

開啟DFU組件

在MiddleWares中加入USB DFU組件

caece1ca-6eb1-11ed-8abf-dac502259ad0.png

設置DFU參數

開啟DFU組件后,CubeMX的程序設置窗口的MiddleWares中就會出現(xiàn)DFU程序設置按鈕。

cb0498c4-6eb1-11ed-8abf-dac502259ad0.png

點開它將APP加載的地址改為0x0800_c000,這個加載地址根據你實際的應用設置,目前我們選擇讓flash的前三個sector為Bootloader的區(qū)域。

cb0e3366-6eb1-11ed-8abf-dac502259ad0.png

第二個全是字段的參數是用來在DFU連接升級軟件式傳輸給軟件用來獲取Flash結構字符串數據,很好理解這個小協(xié)議的內容,點擊設置后,下方的CubeMX的參數說明也寫的很清晰,這里就不多說了。當然這些參數也在工程生成后在 usbd_conf.h 和 usbd_dfu_if.c 文件中修改。

cb21e7da-6eb1-11ed-8abf-dac502259ad0.png

最后的設置

最后我們添加一個外部的按鍵作為觸發(fā)單片機啟動時進入DFU的方式,按鍵按下后就啟動DFU模式,否則直接加載后方APP程序,這里選用PA0引腳,給它設置個User Label 就叫 USER_BTN_GPIO_Port。

cb2daafc-6eb1-11ed-8abf-dac502259ad0.png

修改補全工程

實現(xiàn) DFU 功能代碼

打開 src 目錄下的 usbd_dfu_if.c 文件補全其中的功能代碼

Flash 解鎖

  1. uint16_tMEM_If_Init_HS(void)
    {
    HAL_FLASH_Unlock();
    return(USBD_OK);
    }
    

    Flash 上鎖

    uint16_tMEM_If_DeInit_HS(void)
    {
    HAL_FLASH_Lock();
    return(USBD_OK);
    }
    

    Flash 擦除

    staticuint32_tGetSector(uint32_tAddress)
    {
    uint32_tsector=0;
    
    if((Address=ADDR_FLASH_SECTOR_0))
    {
    sector=FLASH_SECTOR_0;
    }
    
    ......
    
    }
    elseif((Address=ADDR_FLASH_SECTOR_22))
    {
    sector=FLASH_SECTOR_22;
    }
    else
    {
    sector=FLASH_SECTOR_23;
    }
    returnsector;
    }
    
    uint16_tMEM_If_Erase_HS(uint32_tAdd)
    {
    uint32_tstartsector=0;
    uint32_tsectornb=0;
    /*VariablecontainsFlashoperationstatus*/
    HAL_StatusTypeDefstatus;
    FLASH_EraseInitTypeDeferaseinitstruct;
    
    /*Getthenumberofsector*/
    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_tMEM_If_Write_HS(uint8_t*src,uint8_t*dest,uint32_tLen)
    {
    uint32_ti=0;
    
    for(i=0;i4)
    {
    /*Devicevoltagerangesupposedtobe[2.7Vto3.6V],theoperationwill
    bedonebybyte*/
    if(HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD,(uint32_t)(dest+i),*(uint32_t*)(src+i))==HAL_OK)
    {
    /*Checkthewrittenvalue*/
    if(*(uint32_t*)(src+i)!=*(uint32_t*)(dest+i))
    {
    /*Flashcontentdoesn'tmatchSRAMcontent*/
    return(USBD_FAIL);
    }
    }
    else
    {
    /*ErroroccurredwhilewritingdatainFlashmemory*/
    return(USBD_FAIL);
    }
    }
    return(USBD_OK);
    }
    

    Flash 讀取

    uint8_t*MEM_If_Read_HS(uint8_t*src,uint8_t*dest,uint32_tLen)
    {
    /*ReturnavalidaddresstoavoidHardFault*/
    uint32_ti=0;
    uint8_t*psrc=src;
    
    for(i=0;i/*ReturnavalidaddresstoavoidHardFault*/
    return(uint8_t*)(dest);
    }
    

    獲取 Flash 擦寫時間參數

    uint16_tMEM_If_GetStatus_HS(uint32_tAdd,uint8_tCmd,uint8_t*buffer)
    {
    /*USERCODEBEGIN11*/
    uint16_ttime;
    
    time=TimingTable[GetSector(Add)];
    
    switch(Cmd)
    {
    caseDFU_MEDIA_PROGRAM:
    buffer[1]=(uint8_t)time;
    buffer[2]=(uint8_t)(time<8);
    buffer[3]=0;
    break;
    
    caseDFU_MEDIA_ERASE:
    default:
    buffer[1]=(uint8_t)time;
    buffer[2]=(uint8_t)(time<8);
    buffer[3]=0;
    break;
    }
    return(USBD_OK);
    /*USERCODEEND11*/
    }
    

    usbd_dfu_if.h 文件添加的宏定義

    /*Defineflashaddress*///BLANK1#defineADDR_FLASH_SECTOR_00x08000000#defineADDR_FLASH_SECTOR_10x08004000#defineADDR_FLASH_SECTOR_20x08008000#defineADDR_FLASH_SECTOR_30x0800C000#defineADDR_FLASH_SECTOR_40x08010000#defineADDR_FLASH_SECTOR_50x08020000#defineADDR_FLASH_SECTOR_60x08040000#defineADDR_FLASH_SECTOR_70x08060000#defineADDR_FLASH_SECTOR_80x08080000#defineADDR_FLASH_SECTOR_90x080A0000#defineADDR_FLASH_SECTOR_100x080C0000#defineADDR_FLASH_SECTOR_110x080E0000//BLANK2#defineADDR_FLASH_SECTOR_120x08100000#defineADDR_FLASH_SECTOR_130x08104000#defineADDR_FLASH_SECTOR_140x08108000#defineADDR_FLASH_SECTOR_150x0810C000#defineADDR_FLASH_SECTOR_160x08110000#defineADDR_FLASH_SECTOR_170x08120000#defineADDR_FLASH_SECTOR_180x08140000#defineADDR_FLASH_SECTOR_190x08160000#defineADDR_FLASH_SECTOR_200x08180000#defineADDR_FLASH_SECTOR_210x081A0000#defineADDR_FLASH_SECTOR_220x081C0000#defineADDR_FLASH_SECTOR_230x081E0000/*Flashopratetimefromdatasheetpage128*/#defineFLASH_SECTOR_16KB_WRITE_ERASE_TIME500//500usbframe,means500ms#defineFLASH_SECTOR_64KB_WRITE_ERASE_TIME1100#defineFLASH_SECTOR_128KB_WRITE_ERASE_TIME2000
  2. 修改Main文件

    首先在main文件前添加幾個用于加載APP程序的變量和函數定義

    typedefvoid(*pFunction)(void);
    
    pFunctionJumpToApplication;
    uint32_tJumpAddress;1234

    然后再 main 函數中加入 外部按鍵的判斷、APP程序加載以及USB DFU初始化功能

    intmain(void)
    {
    /*Resetofallperipherals,InitializestheFlashinterfaceandtheSystick.*/
    HAL_Init();
    
    /*Configurethesystemclock*/
    SystemClock_Config();
    
    /*Initializeallconfiguredperipherals*/
    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);//Fordebug
    /*Testifusercodeisprogrammedstartingfromaddress0x0800C000*/
    if(((*(__IOuint32_t*)USBD_DFU_APP_DEFAULT_ADD)&0x2FF80000)==0x20000000)
    {
    HAL_GPIO_WritePin(GPIOG,LD4_Pin,GPIO_PIN_SET);//Fordebug
    /*Jumptouserapplication*/
    JumpAddress=*(__IOuint32_t*)(USBD_DFU_APP_DEFAULT_ADD+4);
    JumpToApplication=(pFunction)JumpAddress;
    
    /*Resetofallperipherals*/
    HAL_DeInit();
    
    /*Setinterruptvectortoappcode*/
    SCB->VTOR=USBD_DFU_APP_DEFAULT_ADD;
    
    /*Initializeuserapplication'sStackPointer*/
    __set_MSP(*(__IOuint32_t*)USBD_DFU_APP_DEFAULT_ADD);
    JumpToApplication();
    }
    }
    
    MX_USB_DEVICE_Init();
    
    while(1)
    {
    }
    }
    

編譯程序下載進入單片機

使用DfuSe

從ST官網DfuSe的程序安裝包,并安裝。然后我們按下之前寫的觸發(fā)按鍵并復位單片機,讓單片機初始 USB DFU 功能,這時如果你插著單片機的USB線,系統(tǒng)應該已經識別了。如果沒有右鍵更新驅動程序,手動指定驅動搜索路徑在DfuSe安裝目錄下的 BinDriver 內。如果直接無法識別USB設備,建議在CubeMx配置完工程后就編譯下載測試一下,看看是不是你在移植過程中哪里寫錯了。

然后我們需要生成一個地址設定在0x0800_c000后的測試程序,就先編寫一個 Blink LED 的程序吧,生成bin、hex或S19文件。然后我們打開DfuSe軟件的Dfu file manager來生成DFU軟件用的.dfu格式的文件。選擇第一項,第二個是用來將.dfu反向變換回來的。大概的操作已經標在圖片上了,操作比較簡單就不做詳細介紹了,記得把Address的地址改到偏移后的地址上否則下載會出錯,其他參數可以不用修改。

然后我們打開DfuSe程序,在Upgrade中選擇生成好的blink.dfu文件,勾選校驗功能,下載程序。成功后復位單片機,LED開始閃爍,移植成功。

更多

仔細區(qū)看看DfuSe的安裝目錄,里面有DFU的資料文檔,還有DFU的工程源代碼,可以用來改寫自己需要的DFU升級程序。

審核編輯:郭婷


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 芯片
    +關注

    關注

    452

    文章

    50213

    瀏覽量

    420951
  • usb
    usb
    +關注

    關注

    60

    文章

    7877

    瀏覽量

    263717
  • STM32
    +關注

    關注

    2264

    文章

    10854

    瀏覽量

    354301

原文標題:STM32高級開發(fā)——使用DFU方案

文章出處:【微信號:技術讓夢想更偉大,微信公眾號:技術讓夢想更偉大】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    單片機STM32可以用Python寫嗎?可以的開發(fā)板有哪些?

    近年來,隨著嵌入式技術的發(fā)展,Python語言逐漸被引入到單片機開發(fā)中,尤其是一些高性能的單片機上。這一趨勢給開發(fā)者帶來了極大的便利,尤其是在快速原型設計和實驗中。本文將詳細探討
    的頭像 發(fā)表于 09-05 08:00 ?2004次閱讀
    <b class='flag-5'>單片機</b><b class='flag-5'>STM32</b>可以用Python寫嗎?可以的<b class='flag-5'>開發(fā)</b>板有哪些?

    stm32單片機用什么軟件編程

    STM32單片機是一種廣泛應用于嵌入式系統(tǒng)領域的微控制器,具有高性能、低功耗、豐富的外設接口等特點。要對STM32單片機進行編程,需要選擇合適的軟件工具。 概述
    的頭像 發(fā)表于 09-02 10:16 ?817次閱讀

    stm32單片機引腳介紹及功能

    STM32系列單片機是STMicroelectronics(意法半導體)推出的一款基于ARM Cortex-M內核的32位微控制器。STM32單片機具有高性能、低功耗、豐富的外設和靈活
    的頭像 發(fā)表于 09-02 09:38 ?3056次閱讀

    stm32單片機的優(yōu)勢有哪些?

    STM32作為一款強大而多功能的單片機,憑借其性能、功能、開發(fā)者生態(tài)系統(tǒng)以及廣泛的應用領域,成為了眾多單片機中的佼佼者。無論是專業(yè)開發(fā)
    發(fā)表于 07-29 09:29

    STM32單片機有哪幾種常見的開發(fā)環(huán)境?

    STM32單片機是一款廣泛應用于嵌入式系統(tǒng)開發(fā)單片機,針對其開發(fā),有以下幾種常見的方式:STM32
    的頭像 發(fā)表于 05-18 08:04 ?2790次閱讀
    <b class='flag-5'>STM32</b><b class='flag-5'>單片機</b>有哪幾種常見的<b class='flag-5'>開發(fā)</b>環(huán)境?

    stm32單片機學習路線

    、函數、指針等。 2.了解電子電路基礎 對于單片機開發(fā)來說,了解電子電路的基礎知識是非常重要的,包括基本的電路原理、電阻、電容、電感等元件的作用和特性。 第二步STM32入門 1.了解
    發(fā)表于 05-10 15:34

    STM32單片機最小化系統(tǒng)設計原理

    STM32最小系統(tǒng),就是能讓STM32單片機能夠正常工作所必須擁有的組成部分的集合,也是STM32單片機正常運行的必要環(huán)境。
    發(fā)表于 04-23 14:54 ?2509次閱讀
    <b class='flag-5'>STM32</b><b class='flag-5'>單片機</b>最小化系統(tǒng)設計原理

    使用STM32L4R9單片機開發(fā)板,DfuSeDemo無法檢測到設備的原因?

    使用STM32L4R9單片機開發(fā)板,官方例程STM32Cube_FW_L4_V1.16.0Projects32L4R9IDISCOVERYApplicationsUSB_DeviceD
    發(fā)表于 04-09 07:59

    51單片機開發(fā)板的主要功能 51單片機開發(fā)板能做什么

    51單片機開發(fā)板是一種基于8051系列單片機芯片的開發(fā)板,具有豐富的功能和廣泛的應用。下面將詳細介紹51單片機
    的頭像 發(fā)表于 01-23 15:52 ?3319次閱讀

    STM32單片機的特點和功能是什么

    STM32單片機是一款基于ARM Cortex-M內核的32位閃存微控制器,由STMicroelectronics公司(意法半導體)生產。STM32單片機具有高性能、低功耗、豐富的外設
    的頭像 發(fā)表于 01-03 15:33 ?1.2w次閱讀

    stm32單片機5v電壓怎么提供

    STM32單片機一般需要供電電壓為3.3V,但也有一些型號的單片機可以工作在5V電壓下。在供電電壓方面,有幾種方法可以為STM32單片機提供
    的頭像 發(fā)表于 12-22 14:30 ?1.7w次閱讀

    STM32單片機實現(xiàn)固件在線升級(IAP)

    1,固件升級方案綜述單片機的固件升級方式有很多種。1、ICP:簡單說就是在單片機開發(fā)時使用燒錄器升級程序,比如使用J-Link燒錄單片機程序
    的頭像 發(fā)表于 12-16 08:00 ?4765次閱讀
    <b class='flag-5'>STM32</b><b class='flag-5'>單片機</b><b class='flag-5'>實現(xiàn)</b>固件在線升級(IAP)

    STM32單片機常用的引腳及功能

    STM32單片機是由意法半導體公司(STMicroelectronics)推出的一種高性能、低功耗的微控制器。它具有豐富的外設功能和強大的處理能力,廣泛應用于各種嵌入式系統(tǒng)的開發(fā)中。 在STM
    的頭像 發(fā)表于 12-07 16:22 ?2.1w次閱讀

    你用過哪些編程語言開發(fā)單片機?

    C語言是最常用的一種用于單片機開發(fā)語言,也是一種高級編程語言,具有較好的可移植性和可讀性。對于單片機,通常使用嵌入式C來進行開發(fā)。 舉
    發(fā)表于 12-04 10:18 ?1221次閱讀

    stm32和51單片機的區(qū)別

    OTPROM。閃存具有更大的存儲容量和更高的讀寫速度。 接口 STM32單片機具有更多的外設接口,包括USB、CAN和以太網等。而51單片機的接口較少,只能支持一些基本的外設。 開發(fā)環(huán)
    發(fā)表于 11-20 13:18