【鴻蒙開發(fā)板試用報(bào)告】從點(diǎn)燈開始理解鴻蒙OS的項(xiàng)目結(jié)構(gòu)與啟動(dòng)流程
51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)
https://harmonyos.51cto.com/#zz
和大家一樣,拿到板子后,就急不可耐的按照老師們的教程開始各種操作了。但是一段時(shí)間后,我突然發(fā)現(xiàn),我對(duì)項(xiàng)目的結(jié)構(gòu)和啟動(dòng)流程還都一知半解。為了能更深入的理解HarmonyOS的代碼,我決定從基礎(chǔ)開始,再?gòu)念^學(xué)習(xí)。
一、整體情況
首先,咱們HarmonyOS是用C語(yǔ)言寫的(廢話),編譯用gcc。項(xiàng)目構(gòu)建上,沒有用傳統(tǒng)的make,而是用的GN。什么是GN?
- Generate Ninja,是Google為Ninja專門開發(fā)的上層編譯框架,可以生成Ninja可以識(shí)別的輸入文件。GN由c++編譯,相比于基于python的gyp,速度快接近20倍。
什么是Ninja?
- Ninja 是Google的一名程序員推出的注重速度的構(gòu)建工具,一般在Unix/Linux上的程序通過make/makefile來構(gòu)建編譯,而Ninja通過將編譯任務(wù)并行組織,大大提高了構(gòu)建速度。
重點(diǎn)突出一個(gè)“快”字??偠灾?,有了這倆先進(jìn)工具的加持,咱這個(gè)鴻蒙編譯速度那是飛快。相信大家都深有體會(huì)。
二、項(xiàng)目結(jié)構(gòu)
1.applications,自然就是用戶的各種應(yīng)用代碼了,這里是咱們的主戰(zhàn)場(chǎng)。具體來說,applications/sample/wifi-iot/app/,這個(gè)app目錄里是咱們的業(yè)務(wù)代碼。
2.base,OS的基礎(chǔ)代碼。主要包含全球化(global),DFX(hiviewdfx),公共基礎(chǔ)(iot_hardware),安全(security),啟動(dòng)恢復(fù)(startup)等若干模塊。
3.build,構(gòu)建目錄。編譯過程中的文件存放目錄。
4.docs,文檔。很多新手往往忽略了自帶的文檔。
5.domains,領(lǐng)域??礃幼邮菐讉€(gè)demo。
6.drivers,驅(qū)動(dòng)。OpenHarmony驅(qū)動(dòng)子系統(tǒng)采用C面向?qū)ο缶幊棠P蜆?gòu)建,通過平臺(tái)解耦、內(nèi)核解耦,兼容不同內(nèi)核,提供了歸一化的驅(qū)動(dòng)平臺(tái)底座,旨在為開發(fā)者提供更精準(zhǔn)、更高效的開發(fā)環(huán)境,力求做到一次開發(fā),多系統(tǒng)部署。
7.foundation,基礎(chǔ)模塊。內(nèi)容很復(fù)雜,包含Ability、ACE、Graphics等等很多模塊。
8.kernel,內(nèi)核代碼。
9.out,輸出目錄。生成的固件文件就在這里。
10.prebuilts,LiteOS預(yù)先編譯好的文件。一些LiteOS的.o和.a文件放在這里,可用來加快編譯速度。
11.test,測(cè)試目錄。具體都是干嘛的暫時(shí)沒有搞清楚。
12.third_party,第三方代碼。
13.utils,工具模塊。像文件訪問、timer、task什么的。
14.vendor,制造商提供的代碼。這里有程序啟動(dòng)的入口代碼,應(yīng)給予一定的關(guān)注。有時(shí)間可以研究一下。
15.build.py,編譯腳本。基本用法:python build.py wifiiot
三、啟動(dòng)流程
HelloWorld的教程我就不再重復(fù)了,推薦參考連老師的文章。關(guān)鍵弄懂一個(gè)地方:
- SYS_RUN(HelloWorld);
這個(gè)SYS_RUN是系統(tǒng)自帶的宏,是告訴項(xiàng)目,咱們的業(yè)務(wù)代碼的入口函數(shù)是HelloWorld。SYS_RUN宏的定義在ohos_init.h頭文件中,位置在\utils\native\lite\include\ohos_init.h,定義如下:
- /**
- * @brief Identifies the entry for initializing and starting a system running phase by the
- * priority 2.
- *
- * This macro is used to identify the entry called at the priority 2 in the system startup
- * phase of the startup process. \n
- *
- * @param func Indicates the entry function for initializing and starting a system running phase.
- * The type is void (*)(void).
- */
- #define SYS_RUN(func) LAYER_INITCALL_DEF(func, run, "run")
定義了系統(tǒng)啟動(dòng)階段的初始化和啟動(dòng)入口,類型必須是void (*)(void),即不能有參數(shù),也沒有返回值。LAYER_INITCALL_DEF也是宏,是為了方便靈活調(diào)整啟動(dòng)階段和優(yōu)先級(jí)而設(shè)定的,具體讀者可以自行研究。
回到咱們的HelloWorld中,這里說一下線程。一般業(yè)務(wù)代碼都會(huì)通過一個(gè)主循環(huán)來執(zhí)行各項(xiàng)任務(wù),最佳的方法是啟動(dòng)一個(gè)線程,這樣入口函數(shù)不會(huì)阻塞導(dǎo)致一系列問題。啟動(dòng)線程的方法如下:
- osThreadAttr_t attr;
- attr.name = "HelloTask";
- attr.attr_bits = 0U;
- attr.cb_mem = NULL;
- attr.cb_size = 0U;
- attr.stack_mem = NULL;
- attr.stack_size = 10240;
- attr.priority = osPriorityNormal;
- if (osThreadNew(HelloTaskFunc, NULL, &attr) == NULL) {
- printf("[HelloTaskDemo] Falied to create HelloTask!\n");
- }
至此,已經(jīng)可以順利完成HelloWorld,且對(duì)項(xiàng)目結(jié)構(gòu)和啟動(dòng)流程有了一個(gè)初步的理解。
51CTO和華為官方合作共建的鴻蒙技術(shù)社區(qū)
https://harmonyos.51cto.com/#zz