• 正文
    • 1 環(huán)境搭建
    • 2 BootLoader工作原理以及常見分區(qū)介紹
    • 3 BootLoader的制作
    • 4 燒錄下載配置
    • 5 運行測試
  • 推薦器件
  • 相關(guān)推薦
申請入駐 產(chǎn)業(yè)圖譜

STM32 IAP應(yīng)用開發(fā)——自制BootLoader

2024/07/15
1.2萬
加入交流群
掃碼加入
獲取工程師必備禮包
參與熱點資訊討論

實際上,BootLoader不僅僅在操作系統(tǒng)上使用,在一些內(nèi)存小,功能應(yīng)用較為簡單的單片機設(shè)備上面也可以通過BootLoader來完成固件升級。

我之前也有發(fā)過一些關(guān)于STM32遠程升級的文章,但用的是第三方BootLoader,而且是基于操作系統(tǒng)實現(xiàn)的。BootLoader占用的內(nèi)存也比較大,而且不開源。
所以這一講我就來介紹一下如何自己制作一個簡單的BootLoader程序。

1 環(huán)境搭建

關(guān)于STM32以及Keil的環(huán)境這里就不具體介紹了,網(wǎng)上教程也很多,不懂的同學(xué)自行查閱資料。

2 BootLoader工作原理以及常見分區(qū)介紹

不管用的是什么MCU,要實現(xiàn)固件升級都離不開BootLoader,BootLoader是一個統(tǒng)稱,它其實只是一段引導(dǎo)程序,在MCU啟動的時候會先運行這段代碼,判斷是否需要升級,如果不需要升級就跳轉(zhuǎn)到APP分區(qū)運行用戶代碼,如果需要升級則先通過一些硬件接口接收和搬運要升級的新固件,然后再跳轉(zhuǎn)到APP分區(qū)運行新固件,從而實現(xiàn)固件升級。

請?zhí)砑訄D片描述

常見分區(qū)方式介紹:

1.Application

沒有加入Bootloader之前,我們單片機內(nèi)部的flash就是一整塊的,所有的應(yīng)用代碼都放在這。

2.Bootloader + Application

在原有的flash區(qū)域里面劃分出兩個區(qū)域,Bootloader和Application,這種分區(qū)方式的好處在于既可以實現(xiàn)升級功能,App區(qū)又可以分到較大的空間,缺點是沒有存放新固件的區(qū)域,需要從外部導(dǎo)入進來,而且一旦傳輸?shù)倪^程被異常打斷,那么原有的App代碼也無法正常運行了,也就是傳說中的“變磚”。

請?zhí)砑訄D片描述

3.Bootloader + Application + Download

這種分區(qū)方式是比較萬能的一種,優(yōu)點是新固件是先存放到Download區(qū)的,哪怕搬運的過程中出現(xiàn)異常中斷的情況,也不會“變磚”,缺點是需要單獨劃分一塊內(nèi)存跟APP區(qū)差不多的區(qū)域用來存放新固件,變相的減少了APP區(qū)的空間,對于內(nèi)存較小的單片機來說壓力就比較大了。

請?zhí)砑訄D片描述

4.Bootloader + Application1 + Application2

這種方式可以同時存在兩套App,優(yōu)點在于升級了新固件以后,還保留了原來的舊版固件,必要的時候還可以進行版本的回退。

請?zhí)砑訄D片描述

5.Bootloader + Setting + Application + Download

這種方式跟第3種基本一樣,只是增加了一個區(qū)域用來存放升級相關(guān)的一些參數(shù)以及用戶的一些配置。

在這里插入圖片描述

3 BootLoader的制作

BootLoader的制作需要根據(jù)實際的需求來做,不同的運行方式或者升級方式在做法上都是有區(qū)別的,包括BootLoader所需要的內(nèi)存空間也不盡相同。

不過不管是用什么方式,Bootloader都應(yīng)該盡可能做的更小更簡潔,這樣的話內(nèi)存的開銷就更小,對于內(nèi)存較小的MCU來說壓力就沒那么大了。

我下面要做的這個bootloader是上面講的常見分區(qū)方式里面的第5種。

分區(qū)介紹:

我用的是STM32F103,內(nèi)存是128K的(想用內(nèi)存更小的MCU也是可以的,改下各個分區(qū)的內(nèi)存分配就行了)。

分區(qū)表如下:

name offset size
boot 0x08000000 0x00003000
setting 0x08003000 0x00001000
app 0x08004000 0x0000E000
download 0x08012000 0x0000E000

功能描述:

運行bootloader的時候先從setting里面讀一些參數(shù),確定是否需要升級,如果需要,則把download分區(qū)的固件搬運到app分區(qū),如果不需要升級則直接跳轉(zhuǎn)到app分區(qū).
至于新固件的下載傳輸過程,我放到App里面去處理了,這跟我的項目實際需求有關(guān)系,App部分這里就先不往下拓展了,后面我會專門寫一篇博客來介紹。

各個功能模塊的具體講解:

1、分區(qū)定義

先把各個分區(qū)的內(nèi)存地址以及大小定義好,方便后面使用。

#define FLASH_SECTOR_SIZE       1024
#define FLASH_SECTOR_NUM        128    // 128K
#define FLASH_START_ADDR        ((uint32_t)0x8000000)
#define FLASH_END_ADDR          ((uint32_t)(0x8000000 + FLASH_SECTOR_NUM * FLASH_SECTOR_SIZE))

#define BOOT_SECTOR_ADDR        0x08000000     // BOOT sector start address 
#define BOOT_SECTOR_SIZE        0x3000         // BOOT sector size
#define SETTING_SECTOR_ADDR     0x08003000     // SETTING sector start address 
#define SETTING_SECTOR_SIZE     0x1000         // SETTING sector size
#define APP_SECTOR_ADDR         0x08004000     // APP sector start address  
#define APP_SECTOR_SIZE         0xE000         // APP sector size
#define DOWNLOAD_SECTOR_ADDR    0x08012000     // Download sector start address
#define DOWNLOAD_SECTOR_SIZE    0xE000         // Download sector size   

2、程序跳轉(zhuǎn)

Bootloader作為引導(dǎo)程序,最重要的工作之一就是通過內(nèi)存跳轉(zhuǎn)進入用戶程序,下面這段代碼可以跳轉(zhuǎn)到任何一個內(nèi)存地址。

uint8_t jump_app(uint32_t app_addr) 
{
    uint32_t jump_addr;
    jump_callback cb;
    if (((*(__IO uint32_t*)app_addr) & 0x2FFE0000 ) == 0x20000000) 
    {  
        jump_addr = *(__IO uint32_t*) (app_addr + 4);  
        cb = (jump_callback)jump_addr;  
        __set_MSP(*(__IO uint32_t*)app_addr);  
        cb();
        return 1;
    } 
    return 0;
}

3、處理函數(shù)

從setting區(qū)里面讀取process狀態(tài)值,然后進行對應(yīng)的處理,如果需要升級則把download區(qū)的固件搬運到app區(qū),然后再運行新APP,如果不需要升級則直接跳轉(zhuǎn)到APP。

process = get_boot_state();
switch (process) 
{
    case START_PROGRAM:
        printf("start app...rn");
        delay_ms(50);
        if (!jump_app(APP_SECTOR_ADDR)) 
        {
            printf("no programrn");
            delay_ms(1000);
        }
        printf("start app failedrn");
        break;
    case UPDATE_PROGRAM:
        printf("update app program...rn");
        app_addr = APP_SECTOR_ADDR;
        down_addr = DOWNLOAD_SECTOR_ADDR;

        printf("app addr: 0x%08X rn", app_addr);
        printf("down addr: 0x%08X rn", down_addr);

        printf("erase mcu flash...rn");
        mcu_flash_erase(app_addr, APP_ERASE_SECTORS);  
        printf("mcu flash erase successrn");
    
        printf("write mcu flash...rn");
        // memset(down_buf, 0, sizeof(down_buf));
        for (i = 0; i < APP_ERASE_SECTORS * 8; i++)
        {
            mcu_flash_read(down_addr, &down_buf[0], 128);
            delay_ms(5);
            mcu_flash_write(app_addr, &down_buf[0], 128);
            delay_ms(5);
            down_addr += 128;
            app_addr += 128;
        }
        printf("mcu flash write successrn");

        set_boot_state(UPDATE_SUCCESS);
        break;
    case UPDATE_SUCCESS:
        printf("update successrn");
        boot_state = UPDATE_SUCCESS_STATE;
        write_setting_boot_state(boot_state);
        set_boot_state(START_PROGRAM);
        break;
    default:
        break;
}

完整代碼下載地址:https://download.csdn.net/download/ShenZhen_zixian/87462312

4 燒錄下載配置

我們的Bootloader做好以后需要燒錄到MCU里面,可以直接用Keil uVison來下載,也可以用J-Flash或者其他,這個都沒關(guān)系,但是要注意內(nèi)存的分配,要把固件燒到對應(yīng)的內(nèi)存地址上。

我這里做出來的bootloader bin只有8K,不過為了方便后續(xù)在這部分增加新功能,我實際分配了12K的空間,地址區(qū)間是0x08000000-0x08003000。

如果是用keil下載的話,需要注意flash的配置,具體如下:

請?zhí)砑訄D片描述
請?zhí)砑訄D片描述

如果是用J-Flash或者STlink的工具燒錄的話注意燒錄的起始地址是0x08000000就好了。

5 運行測試

注:這里我沒講解App部分代碼,你們只看Bootloader部分的log就好了,不影響的,想看APP部分可以看我另外一篇文章,或者下載完整的代碼實際跑一下也行。APP部分講解:STM32 IAP應(yīng)用開發(fā)——通過USB實現(xiàn)固件升級

運行結(jié)果:

不需要升級時直接跳轉(zhuǎn)到App區(qū),如下圖:

請?zhí)砑訄D片描述

需要升級時先從download區(qū)搬運新固件到app區(qū),然后再跳轉(zhuǎn)到App區(qū),如下圖:

請?zhí)砑訄D片描述

結(jié)束語

好了,關(guān)于自制BootLoader的介紹就講到這里,本文只是提供一個思路,不是唯一的方法,關(guān)鍵還是看你自己實際的需求。

還有App那部分這里沒詳細講,我單獨寫了一篇文章,鏈接在下方,合到一起看就比較清晰了?;蛘吣阋部梢韵螺d完整的源碼自己去跑一下,下面的源碼我把BootLoader和APP都上傳了。

APP部分講解:STM32 IAP應(yīng)用開發(fā)——通過USB實現(xiàn)固件升級

完整代碼下載地址:https://download.csdn.net/download/ShenZhen_zixian/87462312

如果你有什么問題或者有更好的方法,歡迎在評論區(qū)留言。

推薦器件

更多器件
器件型號 數(shù)量 器件廠商 器件描述 數(shù)據(jù)手冊 ECAD模型 風(fēng)險等級 參考價格 更多信息
LTC6993IS6-3#TRMPBF 1 Linear Technology LTC6993 - TimerBlox: Monostable Pulse Generator (One Shot); Package: SOT; Pins: 6; Temperature Range: -40&deg;C to 85&deg;C
$3.18 查看
511BBA200M000BAG 1 Silicon Laboratories Inc LVDS Output Clock Oscillator,

ECAD模型

下載ECAD模型
$5.05 查看
HFBR-1414MZ 1 Foxconn Transmitter, 792nm Min, 865nm Max, 160Mbps, ST Connector, DIP, Panel Mount, Through Hole Mount, ROHS COMPLIANT PACKAGE
$26.38 查看

相關(guān)推薦