精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

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

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

3天內不再提示

STM32 啟動流程的詳細講解

ouxiaolong ? 來源:嵌入式大雜燴 ? 作者:嵌入式大雜燴 ? 2022-03-06 15:26 ? 次閱讀

開發環境:

處理器STM32F103

MDK:5.30

STM32CubeMX:6.0.1

對于我們常用的桌面操作系統而言,我們在開發應用時,并不關心系統的初始化,絕大多數應用程序是在操作系統運行后才開始運行的,操作系統已經提供了一個合適的運行環境,然而對于嵌入式設備而言,在設備上電后,所有的一切都需要由開發者來設置,這里處理器是沒有堆棧,沒有中斷,更沒有外圍設備,這些工作是需要軟件來指定的,而且不同的CPU類型、不同大小的內存和不同種類的外設,其初始化工作都是不同的。本文將以STMF103(基于Cortex-M3)為例進行講解。

在開始正式講解之前,你需要了解ARM寄存器匯編以及反編譯相關的知識,這些可以參考筆者博文。

深入理解ARM寄存器:https://bruceou.blog.csdn.net/article/details/117866186

ARM匯編入門:https://bruceou.blog.csdn.net/article/details/117897496

Keil反編譯入門(一):https://bruceou.blog.csdn.net/article/details/118314875

Keil反編譯入門(二):https://bruceou.blog.csdn.net/article/details/118400368

下面我們就來具體看一下用戶從Flash啟動STM32的過程,主要講解從上電復位到main函數的過程。主要有以下步驟:

1.初始化堆棧指針 SP=_initial_sp,初始化 PC指針=Reset_Handler

2.初始化中斷向量表

3.配置系統時鐘

4.調用 C庫函數_main初始化用戶堆棧,然后進入 main函數。

在開始講解之前,我們需要了解STM32的啟動模式。

1 STM32的啟動模式

首先要講一下STM32的啟動模式,因為啟動模式決定了向量表的位置,STM32有三種啟動模式:

1)主閃存存儲器(Main Flash)啟動:從STM32內置的Flash啟動(0x0800 0000-0x0807 FFFF),一般我們使用JTAG或者SWD模式下載程序時,就是下載到這個里面,重啟后也直接從這啟動程序。以0x08000000對應的內存為例,則該塊內存既可以通過0x00000000操作也可以通過0x08000000操作,且都是操作的同一塊內存。

2)系統存儲器(System Memory)啟動:從系統存儲器啟動(0x1FFFF000 - 0x1FFF F7FF),這種模式啟動的程序功能是由廠家設置的。一般來說,我們選用這種啟動模式時,是為了從串口下載程序,因為在廠家提供的ISP程序中,提供了串口下載程序的固件,可以通過這個ISP程序將用戶程序下載到系統的Flash中。以0x1FFFFFF0對應的內存為例,則該塊內存既可以通過0x00000000操作也可以通過0x1FFFFFF0操作,且都是操作的同一塊內存。

3)片上SRAM啟動:從內置SRAM啟動(0x2000 0000-0x3FFFFFFF),既然是SRAM,自然也就沒有程序存儲的能力了,這個模式一般用于程序調試。SRAM只能通過0x20000000進行操作,與上述兩者不同。從SRAM啟動時,需要在應用程序初始化代碼中重新設置向量表的位置。

用戶可以通過設置BOOT0和BOOT1的引腳電平狀態,來選擇復位后的啟動模式。如下圖所示:

poYBAGGm-NuAD4bcAAB4W9j4wms335.png

啟動模式只決定程序燒錄的位置,加載完程序之后會有一個重映射(映射到0x00000000地址位置);真正產生復位信號的時候,CPU還是從開始位置執行。

值得注意的是STM32上電復位以后,代碼區都是從0x00000000開始的,三種啟動模式只是將各自存儲空間的地址映射到0x00000000中。

2 STM32的啟動文件分析

因為啟動過程主要是由匯編完成的,因此STM32的啟動的大部分內容都是在啟動文件里。筆者的啟動文件是startup_stm32f103xe.s,不管使用標準庫還是使用HAL庫,啟動文件都是差不多的。

2.1堆棧定義

1.Stack棧

棧的作用是用于局部變量,函數調用,函數形參等的開銷,棧的大小不能超過內部SRAM的大小。當程序較大時,需要修改棧的大小,不然可能會出現的HardFault的錯誤。

pYYBAGGm-NyAO9D2AAB5BNYbnlQ498.png

第33行:表示開辟棧的大小為 0X00000400(1KB),EQU是偽指令,相當于C中的 define。

第35行:開辟一段可讀可寫數據空間,ARER偽指令表示下面將開始定義一個代碼段或者數據段。此處是定義數據段。ARER后面的關鍵字表示這個段的屬性。段名為STACK,可以任意命名;NOINIT表示不初始化;READWRITE表示可讀可寫,ALIGN=3,表示按照 8字節對齊。

第36行:SPACE用于分配大小等于 Stack_Size連續內存空間,單位為字節。

第37行: __initial_sp表示棧頂地址。棧是由高向低生長的。

2.Heap堆

堆主要用來動態內存的分配,像 malloc()函數申請的內存就在堆中。

pYYBAGIkYG2AKWA_AAB_1MjIEWw739.png

開辟堆的大小為 0X00000200(512字節),名字為 HEAP,NOINIT即不初始化,可讀可寫,8字節對齊。__heap_base表示對的起始地址,__heap_limit表示堆的結束地址。

2.2向量表

向量表是一個WORD( 32位整數)數組,每個下標對應一種異常,該下標元素的值則是該 ESR的入口地址。向量表在地址空間中的位置是可以設置的,通過 NVIC中的一個重定位寄存器來指出向量表的地址。在復位后,該寄存器的值為 0。因此,在地址 0(即 FLASH地址 0)處必須包含一張向量表,用于初始時的異常分配。

值得注意的是這里有個另類: 0號類型并不是什么入口地址,而是給出了復位后 MSP的初值,后面會具體講解。

pYYBAGIkYIWAalemAAEzAzwzIsM222.png

……

poYBAGGm-OGAYKu-AACYtV0J3wM865.png

第55行:定義一塊代碼段,段名字是RESET,READONLY表示只讀。

第56-58行:使用EXPORT將3個標識符申明為可被外部引用,聲明 __Vectors、__Vectors_End和__Vectors_Size具有全局屬性。

第60行:__Vectors表示向量表起始地址,DCD表示分配 1個 4字節的空間。每行 DCD都會生成一個 4字節的二進制代碼,中斷向量表存放的實際上是中斷服務程序的入口地址。當異常(也即是中斷事件)發生時,CPU的中斷系統會將相應的入口地址賦值給 PC程序計數器,之后就開始執行中斷服務程序。在60行之后,依次定義了中斷服務程序的入口地址。

第138行:__Vectors_End為向量表結束地址。

第139行:__Vectors_Size則是向量表的大小,向量表的大小是通過__Vectors和__Vectors_End相減得到的。

上述向量表可以在《Reference manual》中找到的,筆者這里只截取了部分。

poYBAGIkYJqAea1JAAXNebKDzzY290.png

2.3復位程序

復位程序是系統上電后執行的第一個程序,復位程序也是中斷程序,只是這個程序比較特殊,因此單獨提出來講解。

pYYBAGGm-OSAPptXAACAL8cSWmk466.png

第145行:定義了一個服務程序,PROC表示程序的開始。

第146行:使用EXPORT將Reset_Handler申明為可被外部引用,后面WEAK表示弱定義,如果外部文件定義了該標號則首先引用該標號,如果外部文件沒有聲明也不會出錯。這里表示復位程序可以由用戶在其他文件重新實現,這種寫法在HAL庫中是很常見的。

第147-148行:表示該標號來自外部文件,SystemInit()是一個庫函數,在system_stm32f1xx.c中定義的,__main是一個標準的 C庫函數,主要作用是初始化用戶堆棧,這個是由編譯器完成的,該函數最終會調用我們自己寫的main函數,從而進入C世界中。

第149行:這是一條匯編指令,表示從存儲器中加載SystemInit到一個寄存器R0的地址中。

第150行:匯編指令,表示跳轉到寄存器R0的地址,并根據寄存器的 LSE確定處理器的狀態,還要把跳轉前的下條指令地址保存到 LR。

第151行:和149行是一個意思,表示從存儲器中加載__main到一個寄存器R0的地址中。

第152行:和150稍微不同,這里跳轉到至指定寄存器的地址后,不會返回。

第153行:和PROC是對應的,表示程序的結束。

2.4中斷服務程序

我們平時要使用哪個中斷,就需要編寫相應的中斷服務程序,只是啟動文件把這些函數留出來了,但是內容都是空的,真正的中斷復服務程序需要我們在外部的 C文件里面重新實現,這里只是提前占了一個位置罷了。

pYYBAGIkYMeAUS-_AADPgbuSoi0636.png

這部分沒啥好說的,和服務程序類似的,只需要注意‘B .’語句,B表示跳轉,這里跳轉到一個‘.’,即表示無線循環。

2.5堆棧初始化

堆棧初始化是由一個IF條件來實現的,MICROLIB的定義與否決定了堆棧的初始化方式。

這個定義是在Options->Target中設置的。

poYBAGIkYNuAYXyDAAGMgB83ta4038.png

如果沒有定義__MICROLIB,則會使用雙段存儲器模式,且聲明了__user_initial_stackheap 具有全局屬性,這需要開發者自己來初始化堆棧。

pYYBAGIkYOqALSSLAAEVrV020ZY840.png

這部分也沒啥講的,需要注意的是,ALIGN表示對指令或者數據存放的地址進行對齊,缺省表示4字節對齊。

2.6其他

poYBAGGm-OqAGc6jAAASCQvb2jQ569.png

第50行:PRESERVE8用于指定當前文件的堆棧按照 8字節對齊。

第51行:THUMB表示后面指令兼容 THUMB指令。現在 Cortex-M系列的都使用 THUMB-2指令集,THUMB-2是 32位的,兼容 16位和 32位的指令,是 THUMB的超集。

3 STM32的啟動流程實例分析

有了前面的分析,接下來就來具體看看STM32啟動流程的具體內容。

3.1初始化SP、PC、向量表

當系統復位后,處理器首先讀取向量表中的前兩個字(8個字節),第一個字存入 MSP,第二個字為復位向量,也就是程序執行的起始地址。

pYYBAGGm-OuAQF1vAAByWiO3WGs335.jpg

這里通過J-Flash打開hex文件。

pYYBAGIkYSWAP1kUAAK0V6w8j5Y782.png

硬件這時自動從0x0800 0000位置處讀取數據賦給棧指針SP,然后自動從0x0800 0004位置處讀取數據賦給PC,完成了復位操作,SP= 0x0200 0410,PC = 0x0800 0145。

初始化SP、PC緊接著就初始化向量表,如果感覺看HEX文件抽象,我們看看反匯編文件吧。

poYBAGIkYV-ACQ07AAI1iaHeQ28403.png

是不是更容易些,是不是和《Reference manual》中的向量表對應起來了。其實看反匯編文件更好理解STM32的啟動流程,只是有些抽象。

3.2設置系統時鐘

細心的朋友可能發現,PC=0x08000145的地址是沒有對齊的。然后在反匯編文件中卻是這樣的:

pYYBAGIkYXGAF4_rAAHcuom8E2I251.png

這里是硬件自動對齊到 0x08000144,并執行SystemInit函數初始化系統時鐘。

當然也可通過硬件調試來確認上面的分析:

pYYBAGIkYYCAPxBeAALPXJh0jEs254.png

接下來就會進入SystemInit函數中。

poYBAGGm-POAQf9bAADXhfB5j74822.png

SystemInit函數內容如下:

/**
 * @brief Setup the microcontroller system
 *        Initialize the Embedded Flash Interface, the PLL and update the 
 *        SystemCoreClock variable.
 * @note  This function should be used only after reset.
 * @param None
 * @retval None
 */
void SystemInit (void)
{
 /* Reset the RCC clock configuration to the default reset state(for debug purpose) */
 /* Set HSION bit */
 RCC->CR |= 0x00000001U;

 /* Reset SW, HPRE, PPRE1, PPRE2, ADCPRE and MCO bits */
#if !defined(STM32F105xC) && !defined(STM32F107xC)
 RCC->CFGR &= 0xF8FF0000U;
#else
 RCC->CFGR &= 0xF0FF0000U;
#endif /* STM32F105xC */  
 
 /* Reset HSEON, CSSON and PLLON bits */
 RCC->CR &= 0xFEF6FFFFU;

 /* Reset HSEBYP bit */
 RCC->CR &= 0xFFFBFFFFU;

 /* Reset PLLSRC, PLLXTPRE, PLLMUL and USBPRE/OTGFSPRE bits */
 RCC->CFGR &= 0xFF80FFFFU;

#if defined(STM32F105xC) || defined(STM32F107xC)
 /* Reset PLL2ON and PLL3ON bits */
 RCC->CR &= 0xEBFFFFFFU;

 /* Disable all interrupts and clear pending bits */
 RCC->CIR = 0x00FF0000U;

 /* Reset CFGR2 register */
 RCC->CFGR2 = 0x00000000U;
#elif defined(STM32F100xB) || defined(STM32F100xE)
 /* Disable all interrupts and clear pending bits */
 RCC->CIR = 0x009F0000U;

 /* Reset CFGR2 register */
 RCC->CFGR2 = 0x00000000U;    
#else
 /* Disable all interrupts and clear pending bits */
 RCC->CIR = 0x009F0000U;
#endif /* STM32F105xC */
   
#if defined(STM32F100xE) || defined(STM32F101xE) || defined(STM32F101xG) || defined(STM32F103xE) || defined(STM32F103xG)
 #ifdef DATA_IN_ExtSRAM
    SystemInit_ExtMemCtl(); 
 #endif /* DATA_IN_ExtSRAM */
#endif 

#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 
}

前面部分是配置時鐘的,具體參考手冊吧,這里需要注意以下代碼:

#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 

默認是沒有開啟VECT_TAB_SRAM,則從FLASH中啟動,VTOR寄存器存放的是中斷向量表的起始地址,在IAP升級會修改這里的偏移量,后面講解IAP升級在細講吧。

3.3初始化堆棧并進入main

執行指令LDR R0, =__main,然后就跳轉到__main程序段運行,當然這里指標準庫的__main函數。

pYYBAGIkYcWABomfAAFzw58xdfQ653.png

這中間初始化了棧區。

poYBAGIkYcqAcd5VAAHfb4agIdk973.png

這段代碼是個循環(BCC 0x08000192),實際運行時候循環了兩次。第一次運行的時候,讀取“加載數據段的函數”的地址并跳轉到該函數處運行(注意加載已初始化數據段和未初始化數據段用的是同一個函數);第二次運行的時候,讀取“初始化棧的函數”的地址并跳轉到該函數處運行。

最后就進入C文件的main函數中,至此,啟動過程到此結束。

最后,總結下STM32從flash的啟動流程。

MCU上電后從0x0800 0000處讀取棧頂地址并保存,然后從0x0800 0004讀取中斷向量表的起始地址,這就是復位程序的入口地址,接著跳轉到復位程序入口處,初始向量表,然后設置時鐘,設置堆棧,最后跳轉到C空間的main函數,即進入用戶程序。

pYYBAGIkYduAHd6EAAIBGqGQGI0878.png

審核編輯:符乾江

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

    關注

    2266

    文章

    10871

    瀏覽量

    354813
  • 啟動流程
    +關注

    關注

    0

    文章

    14

    瀏覽量

    6467
收藏 人收藏

    評論

    相關推薦

    嵌入式系統的啟動流程

    嵌入式系統的啟動流程是一個復雜但有序的過程,它涉及從系統上電到操作系統內核及應用程序啟動的多個階段。
    的頭像 發表于 10-05 17:44 ?291次閱讀

    AMD Versal自適應SoC CPM5 QDMA的Tandem PCIe啟動流程介紹

    本文將從硬件設計和驅動使用兩個方面介紹基于 CPM5 QDMA 的 AMD Versal 自適應 SoC 的 Tandem 設計和啟動流程
    的頭像 發表于 09-18 10:07 ?423次閱讀
    AMD Versal自適應SoC CPM5 QDMA的Tandem PCIe<b class='flag-5'>啟動</b><b class='flag-5'>流程</b>介紹

    電感技術的講解

    詳細講解電感的原理及計算
    的頭像 發表于 09-06 02:07 ?1936次閱讀
    電感技術的<b class='flag-5'>講解</b>

    第12章-ADC采集電壓和顯示 基于STM32的ADC—電壓采集(詳細講解+HAL庫)

    第12章-ADC采集電壓和顯示 基于STM32的ADC—電壓采集(詳細講解+HAL庫)
    的頭像 發表于 08-21 16:31 ?2168次閱讀
    第12章-ADC采集電壓和顯示 基于<b class='flag-5'>STM32</b>的ADC—電壓采集(<b class='flag-5'>詳細</b><b class='flag-5'>講解</b>+HAL庫)

    什么是軟啟動器?它有哪些優缺點?

    隨著現代工業技術的飛速發展,電動機作為工業生產中的核心動力設備,其啟動和運行效率對整體生產流程具有重大影響。在這個過程中,軟啟動器作為一種重要的電機控制裝置,以其獨特的優勢逐漸受到廣泛關注。本文將對軟
    的頭像 發表于 06-13 15:29 ?3335次閱讀

    STM32啟動腳本詳解

    我從Keil和STM32CubeIDE開始我的嵌入式開發,然后慢慢過渡到只使用文本編輯器而沒有任何集成開發環境(IDE)。我一直使用這些IDE提供的啟動文件。在這篇文章中,我將解釋如何用C++編寫
    發表于 05-31 14:17 ?1次下載

    求助,關于TRAVEO MCU閃存啟動流程的問題求解

    我有一個關于 TRAVEO MCU 閃存啟動流程的問題。 Q) 架構 TRM - 34.3.3 閃存啟動流程,(6)、(8)、(9)、(10)、(11)、(12) 和 (13) 可以在
    發表于 05-30 06:25

    Linux啟動流程基本分析

    Linux啟動流程總的來說可以分成三個階段 Linux啟動流程圖 ? 第一步:上電 ?在 x86 系統中,將 1M 空間最上面的 0xF0000 到 0xFFFFF 這 64K 映射給
    的頭像 發表于 05-11 08:49 ?613次閱讀
    Linux<b class='flag-5'>啟動</b><b class='flag-5'>流程</b>基本分析

    STM32F7的雙bank啟動模式有什么應用場景?

    看了STM32中文官網的文件《STM32F769 雙 Bank 啟動 》,感覺還是不太明白是這種雙bank模式的實際應用場景,麻煩各位大神講解一下哈!謝謝啦~
    發表于 04-28 06:42

    詳細講解Altium Designer 23的安裝教程

    在PCB設計中,軟件的安裝是我們邁出的第一步,接下來將詳細講解Altium Designer 23安裝教程。
    的頭像 發表于 01-09 10:02 ?8646次閱讀
    <b class='flag-5'>詳細</b><b class='flag-5'>講解</b>Altium Designer 23的安裝教程

    Rockchip 3588 系統啟動流程

    Rockchip 3588是一款高性能、低功耗的ARM處理器,廣泛應用于智能電視、智能音箱和車載娛樂系統等領域。在本文中,我們將深入探討Rockchip 3588的系統啟動流程。 系統啟動流程
    的頭像 發表于 01-07 16:08 ?2378次閱讀

    Rockchip 系統啟動流程

    Rockchip 系統的啟動流程,包括 Boot ROM、Mask ROM、U-Boot、Linux 內核等。 Boot ROM 階段: 在 Rockchip 系統啟動的第一階段,處理器復位后會首先執行
    的頭像 發表于 01-04 15:54 ?1230次閱讀

    stm32中斷怎么處理的

    STM32是一款非常強大的微控制器系列,具有豐富的外設和功能。中斷是STM32中非常重要的部分,能夠幫助我們提高系統的響應速度和效率。本文將詳細介紹STM32中斷的處理方法。 一、中斷
    的頭像 發表于 01-02 17:35 ?2472次閱讀

    stm32啟動過程詳解

    啟動過程。本文將詳細介紹STM32啟動過程。 一、復位電路 對于STM32微控制器來說,復位是啟動
    的頭像 發表于 12-08 15:47 ?1524次閱讀

    SMP多核secondary cpu啟動流程

    圖,詳細分析可以參考上篇博文。其中以下流程執行secondary cpu啟動相關的一些初始化工作: 在初始化完成且hotplug線程創建完成后,就可通過以下流程喚醒cpu hotplu
    的頭像 發表于 12-05 17:41 ?783次閱讀
    SMP多核secondary cpu<b class='flag-5'>啟動</b><b class='flag-5'>流程</b>