91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

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

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

3天內不再提示

bootloader的原理及實現(xiàn)過程詳解

嵌入式應用開發(fā) ? 來源:嵌入式應用開發(fā) ? 作者:嵌入式應用開發(fā) ? 2022-06-18 17:57 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

一、背景

嵌入式操作系統(tǒng)中,BootLoader是在操作系統(tǒng)內核運行之前運行。可以初始化硬件設備、建立內存空間映射圖,從而將系統(tǒng)的軟硬件環(huán)境帶到一個合適狀態(tài),以便為最終調用操作系統(tǒng)內核準備好正確的環(huán)境。在嵌入式系統(tǒng)中,通常并沒有像BIOS那樣的固件程序

二、實現(xiàn)思路

bootloader其實就是一段啟動程序,它在芯片啟動的時候首先被執(zhí)行,它可以用來做一些硬件的初始化,當初始化完成之后跳轉到對應的應用程序中去。

我們可以將內存分為兩個區(qū),一個是啟動程序區(qū)(0x0800 0000 - 0x0800 2000 )大小為8K Bytes,剩下的為應用程序區(qū)(0x0800 2000 - 0x0801 0000)。

芯片上電時先運行啟動程序,然后跳轉到應用程序區(qū)執(zhí)行應用程序。

三、程序跳轉

bootloader一個主要的功能就是首先程序的跳轉。在STM32中只要將要跳轉的地址直接寫入PC寄存器,就可以跳轉到對應的地址中去。

怎么實現(xiàn)呢?

當我們實現(xiàn)一個函數(shù)的時候,這個函數(shù)最終會占用一段內存,而它的函數(shù)名代表的就是這段內存的起始地址。當我們調用這個函數(shù)的時候,單片機會將這段

內存的首地址(函數(shù)名對應的地址)加載到PC寄存器中,從而跳轉到這段代碼來執(zhí)行。那么我們也可以利用這個原理,定義一個函數(shù)指針,將這個指針指向我們

想要跳轉的地址,然后調用這個函數(shù),就可以實現(xiàn)程序的跳轉了。

代碼如下:

#define  APP_ADDR  0x08002000   //應用程序首地址定義 
typedef void (*APP_FUNC)(); //函數(shù)指針類型定義

APP_FUNC jump2app; //定義一個函數(shù)指針

jump2app = ( APP_FUNC )(APP_ADDR + 4); //給函數(shù)指針賦值
jump2app(); //調用函數(shù)指針,實現(xiàn)程序跳轉

上面的代碼實現(xiàn)了我們要的跳轉功能,但是為什么要跳轉到(APP_ADDR + 4) 這個地址,而不是APP_ADDR.

首先我們要了解主控芯片的啟動過程。以STM32為例,在芯片上電的時候,首先會從內存地址位0x0800 0000(由啟動模式決定)的地方加載棧頂?shù)刂罚?字節(jié)),從0x0800 0004的地方加載程序復位地址(4字節(jié)),然后跳轉到對應的復位地址去執(zhí)行。

所以上面的程序會中,jump2app這個函數(shù)指針的地址為(APP_ADDR + 4),調用這個函數(shù)指針的時候,芯片內核會自動跳轉到這個指針指向的內存地址,也即是應用程序的復位地址。

四、加載棧地址

實際運行會發(fā)現(xiàn),上面的程序可能會出現(xiàn)問題。因為我們還缺少了一個棧地址的加載過程,也就是芯片上電的第一個動作。這里要用到一點匯編的知識:

__asm void MSR_MSP(uint32_t addr)
{
    MSR MSP, r0
    BX r14;
}
__asm void MSR_MSP(uint32_t addr) 是MDK嵌入式匯編形式。

MSR MSP, r0 意思是將r0寄存器中的值加載到MSP(主棧寄存器,復位時默認使用)寄存器中,r0中保存的是參數(shù)值,即addr的值

BX r14 跳轉到連接寄存器保存的地址中,即退出函數(shù),跳轉到函數(shù)調用地址

完整的程序如下:

#define APP_ADDR 0x08002000 //應用程序首地址定義 
typedef void (*APP_FUNC)(); //函數(shù)指針類型定義

/**
  * @brief
  * @param
  * @retval
  */
__asm void MSR_MSP(uint32_t addr)
{
    MSR MSP, r0
    BX r14;
}


/**
  * @brief
  * @param
  * @retval
  */
void run_app(uint32_t app_addr)
{
    uint32_t reset_addr = 0;
    APP_FUNC jump2app;
    
    /* 跳轉之前關閉相應的中斷 */
    NVIC_DisableIRQ(SysTick_IRQn);
    NVIC_DisableIRQ(LPUART_IRQ);
    
    /* 棧頂?shù)刂肥欠窈戏?這里sram大小為8k) */
    if(((*(uint32_t *)app_addr)&0x2FFFE000) == 0x20000000)
    {
        /* 設置棧指針 */
        MSR_MSP(app_addr);
        /* 獲取復位地址 */
        reset_addr = *(uint32_t *)(app_addr+4);
        jump2app = ( APP_FUNC )reset_addr;
        jump2app();
    }
    else
    {
        printf("APP Not Found!n");
    }
}

五、編譯設置

我們需要在設置界面將默認(0x8000000)改為我們的應用程序地址(0x8002000)

poYBAGKtoIGAOtpZAAGKkMPFU-E907.png

六、中斷向量表重映射

完成了上面的工作,實際測試發(fā)現(xiàn)程序還是無法正確運行。原因是我們沒有進行中斷向量表的重映射。向量表映射?什么時候有做過這個工作,我們來看一下:

.s文件里有如下代碼:

; Reset handler routine
Reset_Handler    PROC
                 EXPORT  Reset_Handler                 [WEAK]
        IMPORT  __main
        IMPORT  SystemInit  
                 LDR     R0, =SystemInit
                 BLX     R0
                 LDR     R0, =__main
                 BX      R0
                 ENDP

這代碼表示,程序在執(zhí)行main函數(shù)之前,會先執(zhí)行SystemInit這個函數(shù)。下面看看這個函數(shù):

/**
  * @brief  Setup the microcontroller system.
  * @param  None
  * @retval None
  */
void SystemInit (void)
{
/*!< Set MSION bit */
  RCC->CR |= (uint32_t)0x00000100U;

  /*!< Reset SW[1:0], HPRE[3:0], PPRE1[2:0], PPRE2[2:0], MCOSEL[2:0] and MCOPRE[2:0] bits */
  RCC->CFGR &= (uint32_t) 0x88FF400CU;

  /*!< Reset HSION, HSIDIVEN, HSEON, CSSON and PLLON bits */
  RCC->CR &= (uint32_t)0xFEF6FFF6U;

  /*!< Reset HSI48ON  bit */
  RCC->CRRCR &= (uint32_t)0xFFFFFFFEU;

  /*!< Reset HSEBYP bit */
  RCC->CR &= (uint32_t)0xFFFBFFFFU;

  /*!< Reset PLLSRC, PLLMUL[3:0] and PLLDIV[1:0] bits */
  RCC->CFGR &= (uint32_t)0xFF02FFFFU;

  /*!< Disable all interrupts */
  RCC->CIER = 0x00000000U;

  /* Configure the Vector Table location add offset address ------------------*/
#ifdef VECT_TAB_SRAM
  SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal SRAM */
#else
  SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal FLASH */
#endif
}

從上面的代碼可以看到,這個函數(shù)主要是做了時鐘的初始化和中斷初始化,還有就是中斷向量表的映射,就是最后那一段代碼

poYBAGKtoMKAO1bzAAD8OXA5lO8440.png

再看看FLASH_BASE 和 VECT_TAB_OFFSET的定義:

poYBAGKtoOCAIUjEAAEZB3-GFsQ012.png

這里默認映射地址就是FLASH的初始地址,所以只要將其改成我們程序的起始地址就行了: SCB->VTOR = 0x08002000

編譯,運行,下載.

七、總結

程序跳轉完成,對于bootloader來說也就完成了一大半。剩下的就是根據(jù)自己的需求去完善相應功能了,比如我的在線升級功能,就要在bootloader里做固件接收和校驗的功能。這里有一點需要特別注意的是,跳轉程序之前最好把你用到的中斷都關了,不然跳轉之后的程序沒有對應的中斷處理函數(shù),那就又可能使得程序進入死循環(huán)中。

審核編輯:符乾江

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

    關注

    5198

    文章

    20436

    瀏覽量

    333946
  • Boot
    +關注

    關注

    0

    文章

    154

    瀏覽量

    37738
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    瑞薩RX MCU啟動文件詳解

    本文詳細介紹了RX MCU從復位到進入main函數(shù)的過程,有助于讀者對RX MCU體系結構的理解,RAM和ROM的初始化,以及bootloader程序的開發(fā)。
    的頭像 發(fā)表于 02-11 08:43 ?1.5w次閱讀
    瑞薩RX MCU啟動文件<b class='flag-5'>詳解</b>

    USBISP/USBasp編程器給Atmega32U4下載Arduino bootloader引導程序

    對于新出廠的ATmega32U4芯片內部是沒有arduino引導程序的,需要用戶預先下載bootloader后才能用串口下載自己的應用程序.在某些罕見情況下舊的bootloader會導致一些bug
    的頭像 發(fā)表于 01-31 14:38 ?666次閱讀
    USBISP/USBasp編程器給Atmega32U4下載Arduino <b class='flag-5'>bootloader</b>引導程序

    深入解析rk平臺Android Bootloader核心代碼:從啟動流程到AVB驗證

    U-Boot中Android Bootloader的核心實現(xiàn),核心作用是 銜接硬件初始化與內核啟動 ,主要負責: ?讀寫Bootloader控制塊(BCB),判斷設備啟動
    的頭像 發(fā)表于 01-22 07:06 ?246次閱讀
    深入解析rk平臺Android <b class='flag-5'>Bootloader</b>核心代碼:從啟動流程到AVB驗證

    LAT1171+STM32F745 USART1 Bootloader 失敗原因分析與解決

    STM32 的 Bootloader 可以支持多種協(xié)議的,比如 USART,I2C,DFU 等等,USARTBootloader 是客戶使用 STM32 的時候常常會用到的協(xié)議??蛻粼?/div>
    發(fā)表于 01-11 17:33 ?0次下載

    深入解析RK平臺Android/Linux Bootloader核心文件:android_bootloader.c

    Bootloader是Android設備啟動的第一道“關卡”,負責初始化硬件、加載系統(tǒng)鏡像并完成內核啟動的前置準備。在基于U-Boot的Android設備中,android_bootloader
    的頭像 發(fā)表于 01-09 10:58 ?1194次閱讀
    深入解析RK平臺Android/Linux <b class='flag-5'>Bootloader</b>核心文件:android_<b class='flag-5'>bootloader</b>.c

    ESP32 編譯過程bootloader 配置階段的 CMake 緩存沖突錯誤,記錄

    你遇到的是 ESP32 編譯過程bootloader 配置階段的 CMake 緩存沖突錯誤,核心原因是系統(tǒng)中混合了 ESP-IDF v5.5.1 和 v5.4.3 兩個版本的路徑,導致
    發(fā)表于 12-23 07:07

    如何在CW32F030上實現(xiàn)IAP功能實現(xiàn)遠程升級?

    區(qū)域進行燒寫,目的是為了在產品發(fā)布后可以方便地通過預留的通信口對產品中的固件程序進行更新升級。 IAP 功能的實現(xiàn),一般將程序分為兩個部分,即:Bootloader 和 APP。Bootloader
    發(fā)表于 12-11 06:15

    Bootloader固件升級的步驟

    文件則下載到 ROM 或 Flash 中BootLoader 后面的地址中。也就是說,存在 ROM/Flash 中的內容是分為兩部分的。 4、要實現(xiàn)在同一個 ROM/Flash 中保存兩段程序
    發(fā)表于 11-19 07:41

    是否可以僅使用 Bootloader Host 來實現(xiàn)可引導加載項目的相同編程結果?

    你好 我想使用 Bootloader Host 在我的 CY8CKIT-059 上對 CY8C5888LTI-LP097 芯片進行編程,并將項目類型設置為可引導加載。我的目標是實現(xiàn)與通過使用 SWD
    發(fā)表于 07-18 07:39

    瑞薩MCU方案:瑞薩RZ/G2L Bootloader單獨編譯方法詳解

    會提出需要單獨編譯Bootloader的需求。為了滿足用戶需求,本文將介紹一種RZ/G2L上脫離Yocto的單獨編譯Bootloader的方法,從而能夠快
    的頭像 發(fā)表于 07-08 14:47 ?2622次閱讀
    瑞薩MCU方案:瑞薩RZ/G2L <b class='flag-5'>Bootloader</b>單獨編譯方法<b class='flag-5'>詳解</b>

    SMA接頭制造工藝詳解:精密加工技術與實現(xiàn)策略

    SMA接頭制造工藝詳解:精密加工技術與實現(xiàn)策略
    的頭像 發(fā)表于 04-26 09:22 ?775次閱讀
    SMA接頭制造工藝<b class='flag-5'>詳解</b>:精密加工技術與<b class='flag-5'>實現(xiàn)</b>策略

    倒裝芯片鍵合技術的特點和實現(xiàn)過程

    本文介紹了倒裝芯片鍵合技術的特點和實現(xiàn)過程以及詳細工藝等。
    的頭像 發(fā)表于 04-22 09:38 ?2845次閱讀
    倒裝芯片鍵合技術的特點和<b class='flag-5'>實現(xiàn)</b><b class='flag-5'>過程</b>

    求助,關于S32G3 Multi-Boot Bootloader創(chuàng)建的問題求解

    使用Bootloader_S32G3XX_ASR_4.4_M7 示例,我根據(jù)以下內容修改了引導源信息: A53: Reset Handler Address: 0x34302000 Load Image Address: 0x342F8000 Image
    發(fā)表于 04-09 08:30

    U-Boot 和 Bootloader,99% 的工程師都分不清?

    嵌入式軟件工程師聽說過 u-boot 和 bootloader,但很多工程師依然不知道他們到底是啥。 ? 今天就來簡單講講?u-boot 和 bootloader?的內容以及區(qū)別
    的頭像 發(fā)表于 03-25 20:47 ?1778次閱讀

    自定義RISC V的bootloader-v3

    在生成SoC時,會生成一個預定義bootloader .bin文件,用于指定soc的工程運行的地址,這包括在flash的存儲地址 ,加載到外存中的運行地址及在外存中分配的存儲空間的大小 。下面我們
    的頭像 發(fā)表于 03-10 09:05 ?1712次閱讀
    自定義RISC V的<b class='flag-5'>bootloader</b>-v3