Niobe開發(fā)板中基于OpenHarmony操作系統(tǒng)進行多線程(多任務(wù))開發(fā)
??51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)??
niobe開發(fā)套件詳情介紹:??Niobe行業(yè)物聯(lián)網(wǎng)開發(fā)板及套件詳解??
線程的基本概念
從系統(tǒng)角度看,線程是競爭系統(tǒng)資源的最小運行單元。線程可以使用或等待CPU、使用內(nèi)存空間等系統(tǒng)資源,并獨立于其它線程運行。
OpenHarmony LiteOS可以給用戶提供多個線程,實現(xiàn)線程間的切換,幫助用戶管理業(yè)務(wù)程序流程。具有如下特性:
- 支持多線程。
- 一個線程代表一個任務(wù)。
- 搶占式調(diào)度機制,高優(yōu)先級的線程可打斷低優(yōu)先級線程,低優(yōu)先級線程必須在高優(yōu)先級線程阻塞或結(jié)束后才能得到調(diào)度。
- 相同優(yōu)先級線程支持時間片輪轉(zhuǎn)調(diào)度方式。
- 共有32個優(yōu)先級[0-31],最高優(yōu)先級為0,最低優(yōu)先級為31。用戶進程可配置的優(yōu)先級有22個 (10~31)。
1、線程的狀態(tài)
線程有多種運行狀態(tài)。系統(tǒng)初始化完成后,創(chuàng)建的線程就可以在系統(tǒng)中競爭一定的資源,由內(nèi)核進行調(diào)度。
線程狀態(tài)通常分為以下四種:
- 就緒(Ready):該線程在就緒隊列中,只等待CPU。
- 運行(Running):該線程正在執(zhí)行。
- 阻塞(Blocked):該線程不在就緒隊列中。包含線程被掛起(suspend狀態(tài))、線程被延時(delay狀態(tài))、線程正在等待信號量、讀寫隊列或者等待事件等。
- 退出態(tài)(Dead):該線程運行結(jié)束,等待系統(tǒng)回收資源。
2、 線程狀態(tài)遷移
就緒態(tài)→運行態(tài): 任務(wù)創(chuàng)建后進入就緒態(tài),發(fā)生任務(wù)切換時,就緒隊列中最高優(yōu)先級的任務(wù)被執(zhí)行,從而進入運行態(tài),同時該任務(wù)從就緒隊列中移出。
運行態(tài)→阻塞態(tài) :正在運行的任務(wù)發(fā)生阻塞(掛起、延時、讀信號量等)時,將該任務(wù)插入到對應(yīng)的阻塞隊列中,任務(wù)狀態(tài)由運行態(tài)變成阻塞態(tài),然后發(fā)生任務(wù)切換,運行就緒隊列中最高優(yōu)先級任務(wù)。
阻塞態(tài)→就緒態(tài)(阻塞態(tài)→運行態(tài)):阻塞的任務(wù)被恢復(fù)后(任務(wù)恢復(fù)、延時時間超時、讀信號量超時或讀到信號量等),此時被恢復(fù)的任務(wù)會被加入就緒隊列,從而由阻塞態(tài)變成就緒態(tài);此時如果被恢復(fù)任務(wù)的優(yōu)先級高于正在運行任務(wù)的優(yōu)先級,則會發(fā)生任務(wù)切換,該任務(wù)由就緒態(tài)變成運行態(tài)。
就緒態(tài)→阻塞態(tài) : 任務(wù)也有可能在就緒態(tài)時被阻塞(掛起),此時任務(wù)狀態(tài)由就緒態(tài)變?yōu)樽枞麘B(tài),該任務(wù)從就緒隊列中刪除,不會參與任務(wù)調(diào)度,直到該任務(wù)被恢復(fù)。
運行態(tài)→就緒態(tài) : 有更高優(yōu)先級任務(wù)創(chuàng)建或者恢復(fù)后,會發(fā)生任務(wù)調(diào)度,此刻就緒隊列中最高優(yōu)先級任務(wù)變?yōu)檫\行態(tài),那么原先運行的任務(wù)由運行態(tài)變?yōu)榫途w態(tài),依然在就緒隊列中。
運行態(tài)→退出態(tài) : 運行中的任務(wù)運行結(jié)束,任務(wù)狀態(tài)由運行態(tài)變?yōu)橥顺鰬B(tài)。退出態(tài)包含任務(wù)運行結(jié)束的正常退出狀態(tài)以及Invalid狀態(tài)。例如,任務(wù)運行結(jié)束但是沒有自刪除,對外呈現(xiàn)的就是Invalid狀態(tài),即退出態(tài)。
阻塞態(tài)→退出態(tài) : 阻塞的任務(wù)調(diào)用刪除接口,任務(wù)狀態(tài)由阻塞態(tài)變?yōu)橥顺鰬B(tài)。
3、線程管理
對于多線程的場景,HarmonyOS內(nèi)核管理線程靠任務(wù)池和就緒隊列,執(zhí)行靠調(diào)度算法。
調(diào)度算法:HarmonyOS內(nèi)核中的線程采用搶占式調(diào)度機制,同時支持SCHED_RR和SCHED_FIFO調(diào)度策略
RR策略能基本保證我們每個任務(wù)都能夠得到有效的執(zhí)行,不會有一些任務(wù)進行長時間等待
FIFO策略優(yōu)點在于任務(wù)的切換比較簡單,而且對于一些時間片不好把握的任務(wù)來說,F(xiàn)IFO能偶更有效的利用我們的cpu。
線程相關(guān)API
此處介紹cmsis2.0的線程接口,頭文件:”//third_party/cmsis/CMSIS/RTOS2/Include/cmsis_os2.h”
創(chuàng)建線程
osThreadId_t osThreadNew (osThreadFunc_t func, void *argument, const osThreadAttr_t *attr);
函數(shù)osThreadNew通過將線程添加到活動線程列表并將其設(shè)置為就緒狀態(tài)來啟動線程函數(shù)。線程函數(shù)的參數(shù)使用參數(shù)指針*argument傳遞。當創(chuàng)建的thread函數(shù)的優(yōu)先級高于當前運行的線程時,創(chuàng)建的thread函數(shù)立即啟動并成為新的運行線程。線程屬性是用參數(shù)指針attr定義的。屬性包括線程優(yōu)先級、堆棧大小或內(nèi)存分配的設(shè)置。可以在RTOS啟動(調(diào)用 osKernelStart)之前安全地調(diào)用該函數(shù),但不能在內(nèi)核初始化 (調(diào)用 osKernelInitialize)之前調(diào)用該函數(shù)。
開發(fā)實例
1、 確定目錄結(jié)構(gòu)
先在路徑./applications/app下新建一個目錄,用于存放業(yè)務(wù)源碼文件。其中“.”表示OpenHarmony源碼的根目錄。
例如:在app下新增業(yè)務(wù)NIOBE_OS_helloworld,其中hello_world.c為業(yè)務(wù)代碼,BUILD.gn為編譯腳本,其目錄結(jié)構(gòu)如下:
.
└── applications
└── app
│── TW002_OS_thread
│ │── os_thread_example.c
│ └── BUILD.gn
└── BUILD.gn
2、 編寫業(yè)務(wù)代碼
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include "ohos_init.h"
#include "cmsis_os2.h"
/*****任務(wù)一*****/
void thread_entry1(void)
{
int sum = 0;
while (1)
{
printf("This is Niobe Thread1----%d\r\n", sum++);
usleep(500000);
}
}
/*****任務(wù)二*****/
void thread_entry2(void)
{
int sum = 0;
while (1)
{
printf("This is Niobe Thread2----%d\r\n", sum++);
usleep(500000);
}
}
/*****任務(wù)創(chuàng)建*****/
static void OS_Thread_example(void)
{
osThreadAttr_t attr;
attr.name = "thread1";
attr.attr_bits = 0U;
attr.cb_mem = NULL;
attr.cb_size = 0U;
attr.stack_mem = NULL;
attr.stack_size = 1024 * 4;
attr.priority = 25;
if (osThreadNew((osThreadFunc_t)thread_entry1, NULL, &attr) == NULL)
{
printf("Falied to create thread1!\n");
}
attr.name = "thread2";
if (osThreadNew((osThreadFunc_t)thread_entry2, NULL, &attr) == NULL)
{
printf("Falied to create thread2!\n");
}
}
3、 編寫將業(yè)務(wù)構(gòu)建成靜態(tài)庫的BUILD.gn
static_library("os_thread_example"){
sources = [
"os_thread_example.c"
]
include_dirs = [
"http://third_party/cmsis/CMSIS/RTOS2/Include"
]
}
4、編寫模塊BUILD.gn文件
import("http://build/lite/config/component/lite_component.gni")
lite_component("app") {
features = [
#"NIOBE_OS_helloworld:helloworld",
"TW002_OS_thread:os_thread_example"
]
}
編譯
用docker編譯,進入OpenHarmony代碼根目錄,運行命令進入docker鏡像,在鏡像中用hb編譯:
sudo docker run -it -v $(pwd):/home/openharmony swr.cn-south-1.myhuaweicloud.com/openharmony-docker/openharmony-docker:0.0.5
hb set
.
//繼續(xù)回車選擇niobe_wifi_iot
hb build -b release -f
等待編譯成功。
燒錄
編譯成功后,bin文件會保存在out/niobe/niobe_wifi_iot目錄下:
用HiBurn.exe將Hi3861_wifiiot_app_allinone.bin文件燒錄到niobe核心板上:
首先用typeC線連接電腦和Niobe核心板,可通過設(shè)備管理確定Niobe連接的端口號,該端口號后續(xù)HiBurn和sscom都需要。
再通過HiBurn.exe工具將固件燒錄到Niobe上,HiBurn工具的獲取和操作可參考燒錄指導(dǎo)
調(diào)試
采用串口調(diào)試工具sscom查看串口打印信息,先對sscom進行配置,設(shè)置端口號、波特率等:
點擊打開串口,按下Niobe核心板上的復(fù)位按鍵,可通過sscom看到串口打印日志如下:
This is Niobe Thread1----2
This is Niobe Thread2----5
This is Niobe Thread1----3
This is Niobe Thread2----6
可以看到線程thread_entry1和線程thread_entry2交替運行。
??51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)??