那曲檬骨新材料有限公司

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

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

3天內不再提示

淺析Zephyr在ESP32上的啟動流程

ZephyrProject ? 來源:開源中國 ? 作者:HalfCoder ? 2021-06-07 17:04 ? 次閱讀

理解Zephyr在一款soc上的啟動流程,有利于分析和調試開機過程卡死,驅動異常等的問題。因此在上手一款新的soc時掌握Zephyr在其上面的啟動流程非常必要。本文對Zephyr在ESP32上的啟動流程進行分析,說明ESP32從上電開始如何執行到Zephyr應用的main函數。

Zephyr支持兩種ESP32引導方式:

配置CONFIG_BOOTLOADER_ESP_IDF=n:ROM Boot -》 Zephyr

配置CONFIG_BOOTLOADER_ESP_IDF=y. ROM Boot -》 ESP32 Bootloader -》 Zephyr

本文只分析CONFIG_BOOTLOADER_ESP_IDF=y的流程,在該流程理解另一種也類似,在CONFIG_BOOTLOADER_ESP_IDF=n的情況下相當于是在ESP32的bootloader處放了一個zephyr應用。

ESP32下Zephyr是被當作ESP32的APP被引導,因此有必要簡單了解ESP32的啟動流程

ESP32啟動階段

ESP32是雙核CPU,其中cpu0叫做PRO CPU, cpu1叫做APP CPU,啟動流程如下:

SOC上電, PRO CPU開始運行,跳到ROM 0x40000400 處復位向量代碼處執行

在PRO CPU上運行ROM上一級引導代碼從Flash的0x1000讀出二級引導程序加載到內部IRAM

跳轉到內部IRAM上二級引導程序執行

二級引導程序從 Flash 的 0x8000 偏移地址處讀取分區表, 從分區表中讀到APP的信息

二級引導程序將Zephyr數據和代碼段復制到DRAM和IRAM。對于Zephyr內一些加載地址位于DROM和IROM區域的段,通過配置 Flash MMU 為其提供正確的映射。

二級引導程序會從Zephyr二進制鏡像文件的頭部尋找的入口地址,然后跳轉到該地址處運行。

以上流程中1~3是已經被固化到ESP32的ROM中無法修改,4~6是由modules/hal/espressif/components/bootloader完成,可以做定制修改,但一般不修改。以上1~6都是在PRO CPU中執行。

Zephyr的入口地址就是函數__start,第六步后就會跳轉到__start中執行

Zephyr階段

Zephyr階段運行到main主要步驟:__start-》z_cstart-》bg_thread_main-》main

__start

文件位置zephyr/soc/xtensa/esp32/soc.c, 主要完成下面內容:

搬移中斷向量表

初始化bss段

關閉中斷

確保APP CPU沒有運行(將在后面SMP初始化階段打開)

代碼摘要如下

void __attribute__((section(“.iram1”))) \_\_start(void)

{

//搬移中斷向量表

__asm__ __volatile__ (

“wsr %0, vecbase”

: “r”(&_init_start));

//BSS段初始化

(void)memset(&_bss_start, 0,

(&_bss_end - &_bss_start) * sizeof(_bss_start));

__asm__ __volatile__ (

“”

: “g”(&_bss_start)

: “memory”);

//關閉中斷

__asm__ __volatile__ (

“wsr %0, PS”

: “r”(PS_INTLEVEL(XCHAL_EXCM_LEVEL) | PS_UM | PS_WOE));

//關閉APP CPU

*app_cpu_config_reg &= ~DPORT_APPCPU_CLKGATE_EN;

//初始化cpu指針

__asm__ volatile(“wsr.MISC0 %0; rsync” : : “r”(&_kernel.cpus[0]));

//開始zephyr初始化

z_cstart();

CODE_UNREACHABLE;

}

是否發現跳到zephyr的__start是一個C函數,但之前Zephyr并沒有做C堆棧(SP指針)初始化?這是因為在ESP32的bootloader階段已經做了,Zephyr無需再做。

z_cstart

主要完成kernel初始化,PRE_KERNEL_1和PRE_KERNEL_2級別的驅動初始化,然后啟動main thread:bg_thread_main,剩下的其它初始化和應用程序的main都在bg_thread_main中。

代碼摘要如下

__boot_func

FUNC_NORETURN void z_cstart(void)

{

//架構相關的內核初始化

arch_kernel_init();

// static devices初始化

z_device_state_init();

//初始化PRE_KERNEL_1和PRE_KERNEL_2驅動,大多都是硬件相關

z_sys_init_run_level(_SYS_INIT_LEVEL_PRE_KERNEL_1);

z_sys_init_run_level(_SYS_INIT_LEVEL_PRE_KERNEL_2);

//創建并切換到main thread運行

switch_to_main_thread(prepare_multithreading());

CODE_UNREACHABLE; /* LCOV_EXCL_LINE */

}

__boot_func

static char *prepare_multithreading(void)

{

char *stack_ptr;

//初始化OS調度器

z_sched_init();

//創建main thread

stack_ptr = z_setup_new_thread(&z_main_thread, z_main_stack,

CONFIG_MAIN_STACK_SIZE, bg_thread_main,

NULL, NULL, NULL,

CONFIG_MAIN_THREAD_PRIORITY,

K_ESSENTIAL, “main”);

//將main thread加入到就緒態

z_mark_thread_as_started(&z_main_thread);

z_ready_thread(&z_main_thread);

//為每顆CPU 創建idle thread

for (int i = 0; i 《 CONFIG_MP_NUM_CPUS; i++) {

init_idle_thread(i);

_kernel.cpus[i].idle_thread = &z_idle_threads[i];

_kernel.cpus[i].id = i;

_kernel.cpus[i].irq_stack =

(Z_KERNEL_STACK_BUFFER(z_interrupt_stacks[i]) +

K_KERNEL_STACK_SIZEOF(z_interrupt_stacks[i]));

}

initialize_timeouts();

return stack_ptr;

}

main thread被加入到就緒態,因此下一次調度時bg_thread_main就會被執行

bg_thread_main

在bg_thread_main中完成剩余的驅動初始化,并且啟動esp32的第二顆CPU: APP CPU, 然后運行到應用的main函數。

代碼摘要如下

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

__boot_func

static void bg_thread_main(void *unused1, void *unused2, void *unused3)

{

z_sys_post_kernel = true;

//初始化POST_KERNEL級別驅動

z_sys_init_run_level(_SYS_INIT_LEVEL_POST_KERNEL);

boot_banner();

//初始化APPLICATION級別驅動

z_sys_init_run_level(_SYS_INIT_LEVEL_APPLICATION);

//初始化靜態聲明的thread

z_init_static_threads();

#ifdef CONFIG_SMP

//初始化SMP, 到這里才會啟動ESP32的另一顆CPU

z_smp_init();

//初始SMP級別的驅動,例如跨CPU通信的mailbox, ipm驅動

z_sys_init_run_level(_SYS_INIT_LEVEL_SMP);

#endif

extern void main(void);

//執行應用程序的main

main();

/* Mark nonessenrial since main() has no more work to do */

z_main_thread.base.user_options &= ~K_ESSENTIAL;

}

關于main

這里調用的main函數,是在Zephyr應用程序中實現,最后通過鏈接器鏈接在一起。Zephyr應用程序的main是在main thread中執行,由于main thread的默認優先級比較高0, 因此要注意不要在main中去做while(1),避免導致其它搶占式線程拿不到CPU。

關于SMP

從前面的分析可以看到z_smp_init前,Zephyr上包括main thread的所有代碼都是在PRO CPU上執行,在z_smp_init后Zephyr的代碼才有機會運行到APP CPU上, SMP是一個很大的議題,不是在本文分析范圍內。這里簡單列出esp32 SMP初始化的主要流程供參考:

z_smp_init(smp.c)-》arch_start_cpu(esp32-mp.c)-》appcpu_start-》esp32_rom_ets_set_appcpu_boot_addr-》appcpu_entry1-》z_appcpu_stack_switch-》appcpu_entry2-》smp_init_top(smp.c)

參考

https://docs.espressif.com/projects/esp-idf/zh_CN/latest/esp32/api-guides/startup.html

編輯:jq

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

    關注

    68

    文章

    10902

    瀏覽量

    213001
  • soc
    soc
    +關注

    關注

    38

    文章

    4204

    瀏覽量

    219090
  • SMP
    SMP
    +關注

    關注

    0

    文章

    76

    瀏覽量

    19746
  • ESP32
    +關注

    關注

    18

    文章

    978

    瀏覽量

    17525

原文標題:Zephyr ESP32啟動流程

文章出處:【微信號:ZephyrProject,微信公眾號:ZephyrProject】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    esp32如何接入豆包

    要在 ESP32 上接入豆包工具,本質是讓 ESP32 設備與豆包的 API 進行通信,以獲取相關服務
    的頭像 發表于 02-05 13:40 ?125次閱讀
    <b class='flag-5'>esp32</b>如何接入豆包

    【AI技術支持】ESP32模組接大電容無法正常啟動處理

    啟明云端/01問題描述客戶工控產品使用樂鑫ESP32-WROOM-32E-N4模組,ESP-IDF版本為idf4.46時,項目應用時掉
    的頭像 發表于 01-20 18:03 ?142次閱讀
    【AI技術支持】<b class='flag-5'>ESP32</b>模組接大電容無法正常<b class='flag-5'>啟動</b>處理

    【AI技術支持】ESP32-WROVER-IE-N16R8模組啟動失敗問題處理

    你是否曾遇到過?無線視頻監控設備,使用ESP32-WROVER-IE-N16R8模組,ESP-IDF版本為idf5.0.1時,模組需與PHY芯片通信,主板
    的頭像 發表于 11-20 01:00 ?643次閱讀
    【AI技術支持】<b class='flag-5'>ESP32</b>-WROVER-IE-N16R8模組<b class='flag-5'>上</b>電<b class='flag-5'>啟動</b>失敗問題處理

    【AI技術支持】ESP32模組PSRAM的CS引腳上拉導致功耗上升處理

    啟明云端/01你是否曾遇到過?游戲機產品,使用ESP32-WROOM-32E/ESP32-WROOM-32UE模組,ESP-IDF版本為
    的頭像 發表于 10-31 08:01 ?730次閱讀
    【AI技術支持】<b class='flag-5'>ESP32</b>模組PSRAM的CS引腳上拉導致功耗上升處理

    esp32使用chatGPT做一些有意思的事情

    ChatGPT獲得響應,我們需要進行以下步驟:1、OpenAI網站上注冊,并在ESP32安裝必要的庫。 2、OpenAI API
    的頭像 發表于 10-18 10:04 ?715次閱讀

    esp32esp8266代碼共用嗎

    本文將介紹ESP32ESP8266兩款流行的微控制器代碼共用性方面的可能性與差異性。 一、引言 隨著物聯網技術的飛速發展,越來越多的智能設備開始進入我們的生活。其中,ESP32
    的頭像 發表于 08-19 18:23 ?1430次閱讀

    esp8266和esp32區別是什么

    的內存選項,包括最小512KB的RAM和最大16MB的閃存。 Wi-Fi和藍牙 : ESP8266 :支持2.4GHz Wi-Fi。 ESP32 :支持2.4GHz Wi-Fi和藍牙5.0,這使得ESP32
    的頭像 發表于 08-19 18:16 ?5952次閱讀

    esp32用什么軟件編程

    ESP32是一款由樂鑫(Espressif)推出的低功耗、高性能的Wi-Fi和藍牙雙模微控制器,廣泛應用于物聯網、智能家居、智能穿戴等領域。要對ESP32進行編程,需要選擇合適的編程軟件和開發環境
    的頭像 發表于 08-19 17:24 ?4320次閱讀

    ESP32-WROOM-32E、ESP32-WROOM-32D、ESP32-WROOM-32U 有什么區別?ESP32-WROOM-32 后綴字母代表的意思是?

    相信很多人心里都有這樣的疑問,今天就教大家怎么區分它們。 32D和32U是同一個芯片ESP32-D0WD的模組,主要區別的天線模式,分別是板載和IPEX外接天線。 32E是用的升級版的芯片
    的頭像 發表于 07-17 10:09 ?1.1w次閱讀
    <b class='flag-5'>ESP32</b>-WROOM-32E、<b class='flag-5'>ESP32</b>-WROOM-32D、<b class='flag-5'>ESP32</b>-WROOM-32U  有什么區別?<b class='flag-5'>ESP32</b>-WROOM-32 后綴字母代表的意思是?

    ESP32能取代STM32嗎?哪個更好?

    不能!首先二者不存在哪個更好的問題,因為這兩個芯片使用場景各不相同,嵌入式系統領域,ESP32和STM32都是常見的單片機系列,它們各自具有一定的優勢和適用場景。本文主要探討ESP32是否能夠取代
    的頭像 發表于 07-06 08:04 ?1.4w次閱讀
    <b class='flag-5'>ESP32</b>能取代STM32嗎?哪個更好?

    esp32s3的啟動時間遠大于esp32,為什么?

    我自己畫了一塊esp32s3的開發板,使用esp32s3-wroom-1 n8r2的模組,最近測試時發現,這個開發板幾乎沒有添加額外程序時啟動需要70ms,而
    發表于 07-01 06:25

    樂鑫esp32系列睡眠模式下保持藍牙連接的功耗測試

    本教程適用于ESP32-S3、ESP32-C3、ESP32-C6睡眠模式介紹ESP32系列常見的休眠方式有三種,分別為Modem-sleep、Light-sleep和Deep-slee
    的頭像 發表于 06-21 08:04 ?3519次閱讀
    樂鑫<b class='flag-5'>esp32</b>系列<b class='flag-5'>在</b>睡眠模式下保持藍牙連接的功耗測試

    如何在zephyr使能esp32-c3內置的USB串口?

    如題,請問如何在zephyr使能esp32-c3內置的 USB 串口。
    發表于 06-12 08:11

    樂鑫科技發布全新ESP32-H4 SoC

    樂鑫信息科技最新發布了ESP32-H4,這一創新產品是對其ESP32-H2系列的完美補充,進一步鞏固了公司802.15.4和Bluetooth LE領域的產品布局。ESP32-H4作
    的頭像 發表于 05-06 15:18 ?1135次閱讀

    淺析ESP32運行MQTT客戶端進行主題的發布和訂閱的方法

    ESP32 MQTT的庫有很多,凌順實驗室(lingshunlab.com)這次主要使用AsyncMQTT_ESP32,以后有機會再更多的MQTT其他庫的使用方法。
    的頭像 發表于 03-18 10:05 ?4171次閱讀
    <b class='flag-5'>淺析</b><b class='flag-5'>ESP32</b>運行MQTT客戶端進行主題的發布和訂閱的方法
    百家乐官网开户就送现金| 水果机技巧规律| 百家乐知识技巧玩法| 网页百家乐官网游戏| 战神国际娱乐城| KK娱乐| www.18lk.com| 大玩家百家乐游戏| 百家乐游戏玩法技巧| 百家乐破解的方法| 澳门百家乐真人娱乐城| 宝博百家乐娱乐城| 百家乐利来| 叶氏百家乐平注技巧| 怎么赢百家乐的玩法技巧和规则| 三国百家乐的玩法技巧和规则| 威尼斯人娱乐注册| 威尼斯人娱乐城游戏lm0| 大发888娱乐场大发888娱乐场| 盛大69棋牌游戏| 疯狂水果机怎么玩| 大发888安装需要多久| tt线上娱乐城| 哪个百家乐官网网站最大| 娱乐场百家乐官网大都| 真人百家乐官网游戏网| 赌博百家乐赢不了| 百家乐视频官方下载| 迪威百家乐娱乐场| 大发888捕鱼游戏| 巴登娱乐城开户| 百家乐官网洗码软件| 百家乐官网对付抽水| 百家乐官网游戏大厅下| 至尊百家乐官网| 百家乐系统足球博彩通| 百家乐一代龙虎机| 百家乐开户投注| 威尼斯人娱乐备用622| 爱玩棋牌官方下载| 百家乐官网天上人间|