OpenHarmony輕量系統(tǒng)開(kāi)發(fā)【4】編寫(xiě)第一個(gè)程序弃甥、啟動(dòng)流程分析

摘要:本文簡(jiǎn)單介紹如何編寫(xiě)第一個(gè)hello world程序,以及程序是被執(zhí)行的

適合群體:適用于Hi3861開(kāi)發(fā)板亿遂,啟動(dòng)流程分析

4.1編寫(xiě)第一個(gè)程序

編寫(xiě)一個(gè)hello world程序比較簡(jiǎn)單卖局,可以參考官網(wǎng):

https://gitee.com/openharmony/docs/blob/master/zh-cn/device-dev/quick-start/quickstart-lite-steps-hi3861-application-framework.md

本文在這里做下總結(jié):

(1)確定目錄結(jié)構(gòu)斧蜕。

開(kāi)發(fā)者編寫(xiě)業(yè)務(wù)時(shí),務(wù)必先在./applications/sample/wifi-iot/app路徑下新建一個(gè)目錄(或一套目錄結(jié)構(gòu))砚偶,用于存放業(yè)務(wù)源碼文件批销。

例如:在app下新增業(yè)務(wù)my_first_app,其中hello_world.c為業(yè)務(wù)代碼染坯,BUILD.gn為編譯腳本风钻,具體規(guī)劃目錄結(jié)構(gòu)如下:

.
└── applications
    └── sample
        └── wifi-iot
            └── app
                │── my_first_app
                │  │── hello_world.c
                │  └── BUILD.gn
                └── BUILD.gn

(2)編寫(xiě)業(yè)務(wù)代碼。

在hello_world.c中新建業(yè)務(wù)入口函數(shù)HelloWorld酒请,并實(shí)現(xiàn)業(yè)務(wù)邏輯骡技。并在代碼最下方,使用 HarmonyOS啟動(dòng)恢復(fù)模塊接口SYS_RUN()啟動(dòng)業(yè)務(wù)羞反。(SYS_RUN定義在ohos_init.h文件中)

#include <stdio.h>
#include "ohos_init.h"
#include "ohos_types.h"

void HelloWorld(void)
{
    printf("___________>>>>>>>>>>>>>>>>>>>> [DEMO] Hello world.\n");
}
SYS_RUN(HelloWorld);

(3)編寫(xiě)用于將業(yè)務(wù)構(gòu)建成靜態(tài)庫(kù)的BUILD.gn文件布朦。

如步驟1所述,BUILD.gn文件由三部分內(nèi)容(目標(biāo)昼窗、源文件是趴、頭文件路徑)構(gòu)成,需由開(kāi)發(fā)者完成填寫(xiě)澄惊。以my_first_app為例唆途,需要?jiǎng)?chuàng)建./applications/sample/wifi-iot/app/my_first_app/BUILD.gn,并完如下配置掸驱。

static_library("myapp") {
    sources = [
        "hello_world.c"
    ]
    include_dirs = [
        "http://utils/native/lite/include"
    ]
}

static_library中指定業(yè)務(wù)模塊的編譯結(jié)果肛搬,為靜態(tài)庫(kù)文件libmyapp.a,開(kāi)發(fā)者根據(jù)實(shí)際情況完成填寫(xiě)毕贼。

sources中指定靜態(tài)庫(kù).a所依賴的.c文件及其路徑温赔,若路徑中包含"http://"則表示絕對(duì)路徑(此處為代碼根路徑),若不包含"http://"則表示相對(duì)路徑鬼癣。

include_dirs中指定source所需要依賴的.h文件路徑陶贼。

(4)編寫(xiě)模塊BUILD.gn文件,指定需參與構(gòu)建的特性模塊待秃。

配置./applications/sample/wifi-iot/app/BUILD.gn文件拜秧,在features字段中增加索引,使目標(biāo)模塊參與編譯章郁。features字段指定業(yè)務(wù)模塊的路徑和目標(biāo)枉氮,以my_first_app舉例,features字段配置如下。

import("http://build/lite/config/component/lite_component.gni")

lite_component("app") {
    features = [
        "my_first_app:myapp",
    ]

}

my_first_app是相對(duì)路徑嘲恍,指向./applications/sample/wifi-iot/app/my_first_app/BUILD.gn。

myapp是目標(biāo)雄驹,指向./applications/sample/wifi-iot/app/my_first_app/BUILD.gn中的static_library("myapp")佃牛。

4.2 Hi3861相關(guān)代碼結(jié)構(gòu)

目前hi3861用的是liteos-m內(nèi)核,但是目前hi3681的liteos-m被芯片rom化了医舆,固化在芯片內(nèi)部了俘侠。所以在harmonyOS代碼是找不到hi3861的內(nèi)核部分。

但是這樣不妨礙我們?nèi)ダ砬録i3861的其他代碼結(jié)構(gòu)蔬将。

hi3861平臺(tái)配置文件位于:

vendor\hisilicon\hispark_pegasus\config.json

可以看到該配置文件有很多內(nèi)容爷速,

第一段這里指定了產(chǎn)品名稱、版本霞怀、使用的內(nèi)核類型

下面這里都是子系統(tǒng):

其中我們重點(diǎn)關(guān)注這幾個(gè)模塊:

(1)applications:應(yīng)用子系統(tǒng)

路徑:applications/sample/wifi-iot/app

作用:這個(gè)路徑下存放了hi3681編寫(xiě)的應(yīng)用程序代碼惫东,例如我們剛剛寫(xiě)得hello world 代碼就放在這個(gè)路徑下。

(2)iot_hardware:硬件驅(qū)動(dòng)子系統(tǒng)

頭文件路徑: base\iot_hardware\peripheral\interfaces\kits

具體代碼路徑毙石,由device\board\hisilicon\hispark_pegasus\liteos_m\config.gni文件中指定:

config.gni文件內(nèi)容較多廉沮,后續(xù)會(huì)一一解讀

作用:存放了 hi3681 芯片相關(guān)的驅(qū)動(dòng)、例如spi徐矩、gpio滞时、uart等。

(3)xts:xts測(cè)試子系統(tǒng)滤灯。

這里我們先不要xts子系統(tǒng)坪稽,不然每次開(kāi)機(jī)后,系統(tǒng)都要跑xts認(rèn)證程序鳞骤,影響我們后面測(cè)試窒百,我們先注刪除,如下:

4.3 Hi3861啟動(dòng)流程

由于hi3681的liteos-m被芯片rom化了豫尽,固化在芯片內(nèi)部了贝咙。所以我們主要看內(nèi)核啟動(dòng)后的第一個(gè)入口函數(shù)。

代碼路徑:

device\soc\hisilicon\hi3861v100\sdk_liteos\app\wifiiot_app\src\app_main.c

hi_void app_main(hi_void)
{
#ifdef CONFIG_FACTORY_TEST_MODE
        printf("factory test mode!\r\n");
#endif

    const hi_char* sdk_ver = hi_get_sdk_version();
    printf("sdk ver:%s\r\n", sdk_ver);
    printf("_____>>>>>>> lza %s %d\r\n", __FILE__, __LINE__);
    hi_flash_partition_table *ptable = HI_NULL;

    peripheral_init();
    peripheral_init_no_sleep();

#ifndef CONFIG_FACTORY_TEST_MODE
    hi_lpc_register_wakeup_entry(peripheral_init);
#endif

    hi_u32 ret = hi_factory_nv_init(HI_FNV_DEFAULT_ADDR, HI_NV_DEFAULT_TOTAL_SIZE, HI_NV_DEFAULT_BLOCK_SIZE);
    if (ret != HI_ERR_SUCCESS) {
        printf("factory nv init fail\r\n");
    }

    /* partion table should init after factory nv init. */
    ret = hi_flash_partition_init();
    if (ret != HI_ERR_SUCCESS) {
        printf("flash partition table init fail:0x%x \r\n", ret);
    }
    ptable = hi_get_partition_table();

    ret = hi_nv_init(ptable->table[HI_FLASH_PARTITON_NORMAL_NV].addr, ptable->table[HI_FLASH_PARTITON_NORMAL_NV].size,
        HI_NV_DEFAULT_BLOCK_SIZE);
    if (ret != HI_ERR_SUCCESS) {
        printf("nv init fail\r\n");
    }

#ifndef CONFIG_FACTORY_TEST_MODE
    hi_upg_init();
#endif

    /* if not use file system, there is no need init it */
    hi_fs_init();

    (hi_void)hi_event_init(APP_INIT_EVENT_NUM, HI_NULL);
    hi_sal_init();
    /* 此處設(shè)為T(mén)RUE后中斷中看門(mén)狗復(fù)位會(huì)顯示復(fù)位時(shí)PC值拂募,但有復(fù)位不完全風(fēng)險(xiǎn)庭猩,量產(chǎn)版本請(qǐng)務(wù)必設(shè)為FALSE */
    hi_syserr_watchdog_debug(HI_FALSE);
    /* 默認(rèn)記錄宕機(jī)信息到FLASH,根據(jù)應(yīng)用場(chǎng)景陈症,可不記錄蔼水,避免頻繁異常宕機(jī)情況損耗FLASH壽命 */
    hi_syserr_record_crash_info(HI_TRUE);

    hi_lpc_init();
    hi_lpc_register_hw_handler(config_before_sleep, config_after_sleep);

#if defined(CONFIG_AT_COMMAND) || defined(CONFIG_FACTORY_TEST_MODE)
    ret = hi_at_init();
    if (ret == HI_ERR_SUCCESS) {
        hi_at_sys_cmd_register();
    }
#endif

    /* 如果不需要使用Histudio查看WIFI驅(qū)動(dòng)運(yùn)行日志等,無(wú)需初始化diag */
    /* if not use histudio for diagnostic, diag initialization is unnecessary */
    /* Shell and Diag use the same uart port, only one of them can be selected */
#ifndef CONFIG_FACTORY_TEST_MODE

#ifndef ENABLE_SHELL_DEBUG
#ifdef CONFIG_DIAG_SUPPORT
    (hi_void)hi_diag_init();
#endif
#else
    (hi_void)hi_shell_init();
#endif

    tcpip_init(NULL, NULL);
#endif

    ret = hi_wifi_init(APP_INIT_VAP_NUM, APP_INIT_USR_NUM);
    if (ret != HISI_OK) {
        printf("wifi init failed!\n");
    } else {
        printf("wifi init success!\n");
    }
    app_demo_task_release_mem(); /* 釋放系統(tǒng)棧內(nèi)存所使用任務(wù) */

#ifndef CONFIG_FACTORY_TEST_MODE
    app_demo_upg_init();
#ifdef CONFIG_HILINK
    ret = hilink_main();
    if (ret != HISI_OK) {
        printf("hilink init failed!\n");
    } else {
        printf("hilink init success!\n");
    }
#endif
#endif
    OHOS_Main();
}

app_main一開(kāi)始打印了 SDK版本號(hào)录肯,中間還會(huì)有一些初始化動(dòng)作趴腋,最后一行會(huì)調(diào)用OHOS_Main();

該函數(shù)原型如下:

void OHOS_Main()
{
#if defined(CONFIG_AT_COMMAND) || defined(CONFIG_FACTORY_TEST_MODE)
    hi_u32 ret;
    ret = hi_at_init();
    if (ret == HI_ERR_SUCCESS) {
        hi_u32 ret2 = hi_at_register_cmd(G_OHOS_AT_FUNC_TBL, OHOS_AT_FUNC_NUM);
        if (ret2 != HI_ERR_SUCCESS) {
            printf("Register ohos failed!\n");
        }
    }
#endif
    OHOS_SystemInit();
}

最后,OHOS_SystemInit函數(shù)進(jìn)行鴻蒙系統(tǒng)的初始化。我們進(jìn)去看下初始化做了哪些動(dòng)作优炬。

路徑:base\startup\bootstrap_lite\services\source\system_init.c

void OHOS_SystemInit(void)
{

    MODULE_INIT(bsp);

    MODULE_INIT(device);

    MODULE_INIT(core);

    SYS_INIT(service);

    SYS_INIT(feature);

    MODULE_INIT(run);

    SAMGR_Bootstrap();

} 

我們可以看到主要是初始化了 一些相關(guān)模塊颁井、系統(tǒng),包括有bsp蠢护、device(設(shè)備)雅宾。其中最終的是MODULE_INIT(run);

它負(fù)責(zé)調(diào)用了,所有run段的代碼葵硕,那么run段的代碼是哪些呢眉抬?

事實(shí)上就是我們前面application中使用SYS_RUN() 宏設(shè)置的函數(shù)名。

還記得我們前面寫(xiě)的hello world應(yīng)用程序嗎懈凹?

#include "ohos_init.h"
#include "ohos_types.h"

void HelloWorld(void)
{
    printf("[DEMO] Hello world.\n");
}

SYS_RUN(HelloWorld);

也就是說(shuō)所有用SYS_RUN() 宏設(shè)置的函數(shù)都會(huì)在使用MODULE_INIT(run); 的時(shí)候被調(diào)用蜀变。

為了驗(yàn)證這一點(diǎn),我們可以加一些打印信息介评,如下:

我們重新編譯后燒錄库北。打開(kāi)串口查看打印信息,如下:

可以看到在27行之后们陆,就打印 hello world的信息贤惯。符合預(yù)期。

寫(xiě)在最后

如果你覺(jué)得這篇內(nèi)容對(duì)你還蠻有幫助棒掠,我想邀請(qǐng)你幫我三個(gè)小忙

  • 點(diǎn)贊孵构,轉(zhuǎn)發(fā),有你們的 『點(diǎn)贊和評(píng)論』烟很,才是我創(chuàng)造的動(dòng)力颈墅。
  • 關(guān)注小編,同時(shí)可以期待后續(xù)文章ing??雾袱,不定期分享原創(chuàng)知識(shí)恤筛。
  • 想要獲取更多完整鴻蒙最新學(xué)習(xí)知識(shí)點(diǎn),請(qǐng)移步前往小編:https://gitee.com/MNxiaona/733GH/blob/master/jianshu
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末芹橡,一起剝皮案震驚了整個(gè)濱河市毒坛,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌林说,老刑警劉巖煎殷,帶你破解...
    沈念sama閱讀 218,682評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異腿箩,居然都是意外死亡豪直,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,277評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)珠移,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)弓乙,“玉大人末融,你說(shuō)我怎么就攤上這事∠救停” “怎么了勾习?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,083評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)懈玻。 經(jīng)常有香客問(wèn)我巧婶,道長(zhǎng),這世上最難降的妖魔是什么酪刀? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,763評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮钮孵,結(jié)果婚禮上骂倘,老公的妹妹穿的比我還像新娘。我一直安慰自己巴席,他們只是感情好历涝,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,785評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著漾唉,像睡著了一般荧库。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上赵刑,一...
    開(kāi)封第一講書(shū)人閱讀 51,624評(píng)論 1 305
  • 那天分衫,我揣著相機(jī)與錄音,去河邊找鬼般此。 笑死蚪战,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的铐懊。 我是一名探鬼主播邀桑,決...
    沈念sama閱讀 40,358評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼科乎!你這毒婦竟也來(lái)了壁畸?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,261評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤茅茂,失蹤者是張志新(化名)和其女友劉穎捏萍,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體空闲,經(jīng)...
    沈念sama閱讀 45,722評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡照弥,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,900評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了进副。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片这揣。...
    茶點(diǎn)故事閱讀 40,030評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡悔常,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出给赞,到底是詐尸還是另有隱情机打,我是刑警寧澤,帶...
    沈念sama閱讀 35,737評(píng)論 5 346
  • 正文 年R本政府宣布片迅,位于F島的核電站残邀,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏柑蛇。R本人自食惡果不足惜芥挣,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,360評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望耻台。 院中可真熱鬧空免,春花似錦、人聲如沸盆耽。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,941評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)摄杂。三九已至坝咐,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間析恢,已是汗流浹背墨坚。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,057評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留映挂,地道東北人框杜。 一個(gè)月前我還...
    沈念sama閱讀 48,237評(píng)論 3 371
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像袖肥,于是被迫代替她去往敵國(guó)和親咪辱。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,976評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容