隨著科技的發(fā)展,人們對嵌入式設備的性能和運行效率要求越來越苛刻,傳統(tǒng)嵌入式設備存在非常單一的流向、極低的CPU利用率等缺點,實時性對于汽車、航天領域至關重要,可能會因為處理器沒有及時響應任務而造成極大的損失。針對這一不足,提出將嵌入式操作系統(tǒng)移植到微處理器中以提高運行效率和穩(wěn)定性。RT Thread作為一款實時、搶占式多任務操作系統(tǒng),在智能家居、航天、安防、可穿戴電子產(chǎn)品方面應用廣泛,勢必會成為未來AIoT(人工智能物聯(lián)網(wǎng))平臺主流的操作系統(tǒng)。
為了搶先一步占領市場,將RT Thread移植到各大廠商的芯片上具有極其重要的現(xiàn)實意義,但是目前并沒有將RT Thread移植到BL602的成熟方案,并且移植具有一定的難度。針對這一現(xiàn)象,本文提出了將RT Thread移植到BL602的方法和關鍵步驟。
1 軟硬件環(huán)境概述
1.1 硬件概述
本文所采用的BL602 芯片使用RISCV 架構的CPU,其超低功耗的電源管理單元可用于高性能應用開發(fā),采用外部時鐘源可以獲得高精度和穩(wěn)定的時鐘,另外還可以選配閃存(Flash),高速緩存可以加快CPU 訪問存儲器的速率,大大提高CPU 利用率和程序執(zhí)行速率。
1.2 軟件準備
本文采用RT Thread 3.1.1標準版本的源碼進行裁剪移植,為了避免從頭開始編寫代碼,提前在博流官網(wǎng)下載bl_mcu_sdk代碼用作參考。
2 RT Thread工程框架啟動原理
2.1 總體啟動流程
整個工程啟動流程分為BL602芯片啟動和RT Thread實時內核啟動。該過程與U boot啟動過程類似。對于前者,芯片上電后會首先從start.S匯編代碼處開始執(zhí)行,最后通過調用entry()進入應用程序入口,進而調用內核啟動函數(shù)rtthread_startup()將系統(tǒng)控制權轉移給RT Thread,具體流程如圖1所示。
圖1 總體啟動流程
2.2 RT Thread實時內核啟動流程
RT Thread實時內核啟動主要由rtthread_startup()函數(shù)完成,主要進行板級初始化、打印RT Thread版本信息、定時器初始化、調度器初始化、應用程序初始化、定時器線程初始化、空閑線程初始化、調度器啟動。
2.2.1 板級初始化
板級初始化過程就是通過調用rt_hw_board_init()函數(shù)實現(xiàn),對于不同型號的芯片,該函數(shù)所要處理的任務大同小異,一般都是初始化芯片引腳和串口、配置系統(tǒng)時鐘、初始化堆空間。對于BL602芯片,因其內部自帶了一個64位的定時器mtimer,將會對該定時器進行初始化以設置系統(tǒng)定時時間,其作用與STM32芯片中的嘀嗒時鐘類似。
2.2.2 定時器初始化
調用rt_system_timer_init()函數(shù)初始化的是硬件中斷模式下的定時器,RT Thread的硬件定時器由一個靜態(tài)的雙向鏈表構成,剛開始初始化定時器隊列為空,即L>next=L >pre=L,具體結構如圖2所示。
圖2 初始化后的定時器結構
2.2.3 調度器初始化
調度器用于調度線程運行,在當前系統(tǒng)的就緒隊列中找到優(yōu)先級最高的就緒線程來執(zhí)行,RTThread中線程一共有32個等級的優(yōu)先級,數(shù)值越大,優(yōu)先級越低。調度器初始化主要對線程優(yōu)先隊列進行初始化為NULL,設置當前線程優(yōu)先級:RT_THREAD_PRIORITY_MAX-1=31為最低的優(yōu)先級,初始化線程控制塊指針為NULL,初始化線程就緒優(yōu)先級組為0。
2.2.4 應用程序初始化
通過調用rt_application_init()來動態(tài)創(chuàng)建main主線程,在線程函數(shù)main_thread_entry()調用main(),從而進入用戶主函數(shù)。不過此時線程并沒有啟動,而是根據(jù)優(yōu)先級插入到了線程就緒優(yōu)先隊列指定位置,等到調度器啟動后才會真正運行主線程。
2.2.5 調度器啟動
通過調用rt_system_scheduler_start()啟動調度器,main線程、idle線程才會真正被調度,系統(tǒng)會從線程就緒隊列中選擇優(yōu)先級最高的一個線程并執(zhí)行。如果系統(tǒng)的就緒隊列中還有線程1(優(yōu)先級為7)、線程2(優(yōu)先級為10)、線程3(優(yōu)先級為7)、tshell(優(yōu)先級為20)四個線程。當調度器啟動后,系統(tǒng)中各個線程指向關系如圖3所示。
圖3 線程指向關系
3 RT Thread移植過程
RT Thread移植就是讓RTThread實時操作系統(tǒng)能夠在BL602芯片上跑起來,并且可以實現(xiàn)任務管理、任務創(chuàng)建、任務調度等功能。移植過程主要分為以下幾個部分:啟動入口、系統(tǒng)時鐘配置、mtimer定時器配置、注冊中斷回調函數(shù)、實現(xiàn)Finsh功能、rt_kprintf實現(xiàn)、任務創(chuàng)建和任務調度。下面將對具體步驟進行詳細介紹。
3.1 添加單板BL602和修改源碼目錄
為了移植最小內核到BL602芯片上,有必要對RT Thread源碼進行裁剪。保留src、include目錄文件不變,components目錄下面只保留finsh文件夾,bsp目錄下添加BL602單板目錄,在libcpu/risc_v目錄下新建一個bl602文件夾,向libcpu/risc_v/bl602中添加對應的 interrupt_gcc.S、cpuport.c、context_gcc.S,這幾個匯編和C文件可以從其他RISC V 架構下的芯片中復制,CPU 架構移植主要實現(xiàn)如下功能:中斷使能/失能、任務切換、線程棧初始化、中斷處理等。把bl_mcu_sdk中的drivers/soc/bl602/startup/start.S 啟動文件復制到libcpu/risc_v/bl602目錄下,作為芯片上電啟動文件。rtconfig.h頭文件中定義了一些預編譯宏,只需定義一些基本的宏就行,至此源碼裁剪工作基本完成。
3.2 修改入口函數(shù)
裸機程序中啟動文件start.S中最后幾行匯編指令如下所示(現(xiàn)在需要修改jal main為 jalentry,從而跳轉到應用程序入口調用rtthread_startup()啟動函數(shù)啟動內核,至此系統(tǒng)控制權就轉交給了操作系統(tǒng)):
//系統(tǒng)初始化,保存默認配置、清除所有中斷
jal SystemInit
/*從ITCM、DTCM、RAM 復制數(shù)據(jù)到TCM 中,并且清空.bss區(qū)*/
jal start_load
//設置PDS、HBN時鐘
jal System_Post_Init
……
//跳轉到用戶主程序
jal main 修改為jalentry
3.3 適配板級初始化函數(shù)
對開發(fā)板的初始化操作一般都在rt_hw_board_init()接口進行,比如系統(tǒng)時鐘初始化、外設時鐘初始化、定時器初始化、串口初始化、GPIO初始化、堆初始化等。
3.3.1 系統(tǒng)時鐘初始化
視系統(tǒng)需求,外部晶振時鐘可選24、32、38.4、40 MHz,可以設置外部晶振XTAL為40 MHz,配置系統(tǒng)時鐘為最高的192 MHz。通過配置clk_cfg0寄存器(地址為0x40000000)的bit[0:3]為1,使能PLLCLK、BCLK、FCLK、HCLK;通過設置bit[4:5]都為1,選擇PLL輸出時鐘為192 MHz;設置bit[6:7]都為1,選擇root clock來自RC32M(RC振蕩器頻率為32 MHz);最后選擇PLL 輸出192MHz作為system clock。
3.3.2 外設時鐘初始化
首先使能外圍設備時鐘,然后再進行配置。BL602芯片的外圍設備時鐘包括Flash、UART、I2C、GPIO 等,這里只用到了UART 外圍設備,bl_mcu_sdk在peripheral_clock_init()只需要執(zhí)行兩個操作:一是使能UART 時鐘(通過寫0x4000 0024的第16位為1);二是配置UART時鐘為160 MHz(通過配置0x40000008的bit7為選擇時鐘源,通過0x4000 0008的bit[0:2]設置分頻系數(shù)為0,設置bit4為1使能串口時鐘)。
3.3.3 配置mtimer定時器
mtimer定時器是RISC V 內核自帶的一個64位定時器,可以通過配置0x4000 0090寄存器來使能mtimer時鐘和選擇時鐘類型,還可以選擇分頻系數(shù),與STM32芯片里面的嘀嗒時鐘類似。bl_mcu_sdk的中斷采用vector mode,一共有18個中斷源,中斷號0~15為RISCV保留中斷,而mtimer中斷號是7,UART0中斷號是(IRQ_NUM_BASE+29),其中IRQ_NUM_BASE為16,可以在rt_hw_board_init()中調用bflb_mtimer_config(1000000,SysTick_Handler),從而將mtimer中斷號與中斷服務函數(shù)SysTick_Handler綁定在一起,假如時鐘經(jīng)過分頻以后為 1 MHz,經(jīng)過以上設置后,mtimer定時時間即為1 s。
3.3.4 串口初始化
主要對引腳和UART 外設進行初始化,bl_mcu_sdk中有個函數(shù)已經(jīng)實現(xiàn)該功能,可以直接在rt_hw_board_init()里調用console_init(),該函數(shù)里面調用bflb_gpio_uart_init()實現(xiàn)了GPIO輸入輸出引腳的初始化以及調用bflb_uart_init()實現(xiàn)了uart0初始化。
3.3.5 堆初始化
和其他操作系統(tǒng)不同的是,RT Thread堆空間大小由程序員自定義的一個靜態(tài)數(shù)組heap_buf[]決定,數(shù)組大小不能超過SRAM 最大值,然后通過rt_system_heap_init(begin_addr, end_addr)函數(shù)對堆空間的起始和終止地址進行初始化,此處應為heap_buf和(heap_buf + sizeof(heap_buf) 1),并對begin_addr進行向上4字節(jié)對齊操作,對end_addr進行向下4字節(jié)對齊操作。在rtconfig.h頭文件中定義RT_USING_HEAP 宏后才能動態(tài)創(chuàng)建main主線程,否則只能通過靜態(tài)方法創(chuàng)建線程。
3.4 實現(xiàn)rt_kprintf功能
要想實現(xiàn)RT Thread的打印功能,其實就是實現(xiàn)rt_hw_console_output()函數(shù)向串口輸出數(shù)據(jù),BL602參考手冊里面有兩個寄存器需要設置:一個是uart_fifo_config_1,地址為:0x4000 a084,需要設置bit[0:5]位用于TX FIFO可用計數(shù);另外一個是uart_fifo_wdata,地址為:0x4000a088,用來將一個字符寫進0x4000 a088地址里,bl_mcu_sdk里面已經(jīng)實現(xiàn)過該部分,直接在rt_hw_console_output()里面調用即可,至此可以通過串口uart0輸出數(shù)據(jù)。
3.5 移植Finsh
移植Finsh組件,實現(xiàn)命令行交互功能,首先需要添加Finsh代碼,在rtconfig.h頭文件中定義RT_USING_FINSH以及與Finsh線程相關的一些宏定義,然后對接rt_hw_console_getchar()函數(shù)即可,其中獲取字符有兩種方式:采用查詢方式或者中斷方式,建議采用中斷方式獲取字符。
3.5.1 查詢方式
查詢方式下不能使能串口接收中斷,且此方法相較于中斷方式較為簡單,不過效率很低,需要對兩個寄存器進行配置:一個是uart_fifo_config_1,地址為0x4000 a084,需要設置bit[8:13]用于RX FIFO 可用計數(shù);另外一個是uart_fifo_rdata,地址為0x4000 a08c,用于從0x4000 a08c地址中讀取一個字符到串口。
3.5.2 中斷方式
采取中斷方式必須使能串口接收中斷,參考BL602數(shù)據(jù)手冊得知,可以通過配置uart_int_en寄存器bit3使能串口接收中斷,寄存器地址為0x4000 a02c。中斷方式獲取字符流程如下:定義一個buffer緩沖區(qū)和一個信號量,當uart接收產(chǎn)生中斷時,會在中斷服務函數(shù)中將讀取到的數(shù)據(jù)緩存到buffer緩沖區(qū),然后釋放信號量來通知finsh線程接收數(shù)據(jù),finsh線程將從緩沖區(qū)中讀取數(shù)據(jù)。所以還需要實現(xiàn)串口接收中斷服務函數(shù)Uart_IRQHandler,并通過bflb_irq_attach(45, Uart_IRQHandler)將UART0中斷號與之綁定在一起,UART0中斷號為(IRQ_NUM_BASE+29),IRQ_NUM_BASE為16。
4 測試結果
操作系統(tǒng)啟動后,會自動創(chuàng)建main、idle、tshell線程。下面將在main函數(shù)中動態(tài)創(chuàng)建3個線程,將線程1、3優(yōu)先級設置為7,線程2優(yōu)先級設置為10,tick都設置為10,在線程函數(shù)中都同時延時1 s,觀察打印出的當前線程數(shù)量是否正確以及各個線程是否按照優(yōu)先級正確被調度執(zhí)行。實驗結果表明,RT Thread操作系統(tǒng)能夠在BL602上穩(wěn)定運行,并且可以正常進行線程創(chuàng)建和調度,測試結果如圖4所示。
圖4 測試結果
5 結 語
本文在介紹了RT Thread實時操作系統(tǒng)基礎上講解了如何在BL602芯片上成功移植RT Thread實時操作系統(tǒng)的方法。最后在BL602上進行測試,可以正常運行,為其他RTOS移植到芯片上提供了參考和借鑒。
(本文由《單片機與嵌入式系統(tǒng)應用》雜志授權發(fā)表,原文刊發(fā)在2023年第10期)
-
嵌入式
+關注
關注
5082文章
19104瀏覽量
304791 -
操作系統(tǒng)
+關注
關注
37文章
6801瀏覽量
123282 -
微處理器
+關注
關注
11文章
2258瀏覽量
82402 -
移植
+關注
關注
1文章
379瀏覽量
28124 -
RTThread
+關注
關注
8文章
132瀏覽量
40859
原文標題:基于BL602的RT Thread移植方法研究
文章出處:【微信號:麥克泰技術,微信公眾號:麥克泰技術】歡迎添加關注!文章轉載請注明出處。
發(fā)布評論請先 登錄
相關推薦
評論