openharmony应用开发实例(代码片段)

Bruceoxl Bruceoxl     2022-10-22     321

关键词:

开发环境
开发系统:Ubuntu 20.04
开发板:Pegasus物联网开发板
MCU:Hi3861
OpenHarmony版本:3.0.1-LTS

4.1新建工程及配置

1.新建工程及源码

  1. 新建目录
$ mkdir hello

在applications/sample/myapp中新建src目录以及myapp.c文件,代码如下所示。

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

void app_task(void)

    printf("\\n");
    printf("Hello hi3861!\\n");
    printf("\\n");

SYS_RUN(app_task);
  1. 新建编译组织文件

新建applications/sample/myapp/BUILD.gn文件,内容如下所示:

static_library("myapp") 
    sources = [
        "src/myapp.c"
    ]
    include_dirs = [
        "//utils/native/lite/include"
    ]

static_library中指定业务模块的编译结果,为静态库文件libmyapp.a,开发者根据实际情况完成填写。

sources中指定静态库.a所依赖的.c文件及其路径,若路径中包含"//“则表示绝对路径(此处为代码根路径),若不包含”//"则表示相对路径。

include_dirs中指定source所需要依赖的.h文件路径

新建的工程目录如下:

$ tree

2.添加新组件
修改文件build/lite/components/applications.json,添加组件hello_world_app的配置。


      "component": "my_app",
      "description": "appsamples.",
      "optional": "true",
      "dirs": [
        "applications/sample/myapp"
      ],
      "targets": [
        "//applications/sample/myapp:myapp"
      ],
      "rom": "",
      "ram": "",
      "output": [],
      "adapted_kernel": [ "liteos_m" ],
      "features": [],
      "deps": 
        "components": [],
        "third_party": []
      
    ,

3.修改单板配置文件

修改文件vendor/hisilicon/hispark_pegasus/config.json,新增my_app组件的条目。


     "subsystem": "applications",
      "components": [
 "component": "wifi_iot_sample_app ", "features":[] 
        "component": "my_app", "features":[] 
      ]
,

4.关闭xts测试子系统。
系统每次开机后都要跑xts认证程序,这里先删除该部分内容。

4.2编译下载验证

接下来就可以编译了。

$ hb set

全编译。

$ hb build -f

成功编译后,固件在out/hispark_pegasus/wifiiot_hispark_pegasus目录下。

Hi3861_wifiiot_app_allinone.bin就是需要烧写的固件。

然后把固件下载到板子中。

接下来就可以根据该实例开发自己的应用了。

4.3系统启动流程分析

下面简单分析下系统的启动流程,系统的入口函数是app_main(),在device/hisilicon/hispark_pegasus/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);

    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();
    /* 此处设为TRUE后中断中看门狗复位会显示复位时PC值,但有复位不完全风险,量产版本请务必设为FALSE */
    hi_syserr_watchdog_debug(HI_FALSE);
    /* 默认记录宕机信息到FLASH,根据应用场景,可不记录,避免频繁异常宕机情况损耗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驱动运行日志等,无需初始化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(); /* 释放系统栈内存所使用任务 */

#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();

该函数首先打印SDK的版本信息,然后挂载文件系统,初始化WiFi信息等等一系列初始化,接这就到OHOS_Main(),该函数就是OpenHarmony系统的初始化。OHOS_Main()函数在文件device/hisilicon/hispark_pegasus/sdk_liteos/app/wifiiot_app/src/ohos_main.c中。

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()函数是一个弱函数,其定义如下:

void __attribute__((weak)) OHOS_SystemInit(void)

    return;

因此该函数主要是系统为应用开发者提供的。OHOS_SystemInit()函数在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();

到这里基本就完成了所得初始化,其中我们编写的应用就是MODULE_INIT(run)中完成的。
在base/startup/bootstrap_lite/services/source/core_main.h文件中,有如下定义:
MODULE_INIT定义如下:

#define MODULE_INIT(name)     \\
    do                       \\
        MODULE_CALL(name, 0); \\
     while (0)

MODULE_CALL定义如下:

#define MODULE_CALL(name, step)                                      \\
    do                                                              \\
        InitCall *initcall = (InitCall *)(MODULE_BEGIN(name, step)); \\
        InitCall *initend = (InitCall *)(MODULE_END(name, step));    \\
        for (; initcall < initend; initcall++)                      \\
            (*initcall)();                                           \\
                                                                    \\
     while (0)

模块的名字定义如下:
#define MODULE_NAME(name, step) “.zinitcall.” #name #step “.init”
而SYS_RUN在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")

而LAYER_INITCALL_DEF定义如下:

#define LAYER_INITCALL(func, layer, clayer, priority)                                     \\
    static __attribute__((constructor(CTOR_VALUE_##layer + LAYER_INIT_LEVEL_##priority))) \\
        void BOOT_##layer##priority##func() func();
#else
#define LAYER_INITCALL(func, layer, clayer, priority)            \\
    static const InitCall USED_ATTR __zinitcall_##layer##_##func \\
        __attribute__((section(".zinitcall." clayer #priority ".init"))) = func
#endif
// Default priority is 2, priority range is [0, 4]
#define LAYER_INITCALL_DEF(func, layer, clayer) \\
    LAYER_INITCALL(func, layer, clayer, 2)

可以看到最终SYS_RUN宏定义都是定义在.zinitcall中,因此SYS_RUN()宏设置的函数都会在MODULE_INIT(run)完成调用。

好了,最后看看应用启动的调用流程:

官方文档



欢迎访问我的网站

BruceOu的哔哩哔哩
BruceOu的主页
BruceOu的博客
BruceOu的CSDN博客
BruceOu的简书
BruceOu的知乎


欢迎订阅我的微信公众号

关注公众号[嵌入式实验楼]]获取更多资讯

openharmony音频流管理开发(代码片段)

...1 音频流管理调用关系图         说明:在进行应用开发的过程中,开发者需要使用getStreamManager()创建一个AudioStreamManager实例,进而通过该实例管理音频流。开发者可通过调用on('audioRendererChange')、on('audi... 查看详情

openharmony音频焦点模式开发(代码片段)

1、OpenHarmony音频焦点模式开发简介        音频焦点模式指的是应用内,允许对多个声音的播放进行控制。音频应用可以在AudioRenderer下设置独立焦点模式、共享焦点模式。当设置在共享的模式下,多个音频共用一个会... 查看详情

openharmony音量管理开发(代码片段)

1、OpenHarmony音量管理开发简介        AudioVolumeManager提供了音量管理的方法。开发者可以通过本博文了解应用如何通过AudioVolumeManager获取指定流音量信息、监听铃声模式变化、设置麦克风静音等。 2、OpenHarmony音量管理运行机... 查看详情

#打卡不停更#-openharmony/docs开发入门(代码片段)

...armony这个系统是什么,能干什么,今日分享从设备开发和应用开发两个方向学习OpenHarmony/docs文档。一、设备开发介绍1.环境搭建与源码获取环境搭建搭建OpenHarmony的系统环境,需要Windows系统与 查看详情

openharmony开发环境搭建(代码片段)

开发环境:开发系统:Ubuntu20.04开发板:Pegasus物联网开发板MCU:Hi3861OpenHarmony版本:3.0.1-LTS本文将介绍如何搭建OpenHarmony开发环境,OpenHarmony主要在Linux平台上开发,因此需要使用Linux环境开发,要想... 查看详情

openharmony相机开发(代码片段)

1、OpenHarmony相机开发简介         OpenHarmony相机模块支持相机业务的开发,开发者可以通过已开放的接口实现相机硬件的访问、操作和新功能开发,最常见的操作如:预览、拍照和录像等。开发者也可以通过合适... 查看详情

openharmony音频开发(代码片段)

1、OpenHarmony音频开发简介        音频播放开发的主要工作是将音频数据转码为可听见的音频模拟信号,并通过输出设备进行播放,同时对播放任务进行管理,包括开始播放、暂停播放、停止播放、释放资源、设... 查看详情

openharmony视频开发(代码片段)

1.简介    OpenHarmony视频播放的主要工作是将视频数据转码并输出到设备进行播放,同时管理播放任务,包括开始播放、暂停播放、停止播放、资源释放、音量设置、跳转播放位置、设置倍数、获取轨道信息等功能控制... 查看详情

openharmony音频录制开发(代码片段)

1、OpenHarmony音频录制开发简介        音频录制的主要工作是捕获音频信号,完成音频编码并保存到文件中,帮助开发者轻松实现音频录制功能。该模块允许调用者指定音频录制的采样率、声道数、编码格式、封装格... 查看详情

openharmony数据转码应用开发实战(下)(代码片段)

...帮助的,本文将以一个小项目——数据转码应用,来讲解应用开发全流程。在《OpenHarmony数据转码应用开发实战(中)》我们讲述了核心解转码工具包的实现,以及UI组件数据绑定,那么接下来将讲述项目的国际化适配、调测和... 查看详情

#打卡不停更#openharmony-应用开发入门指南(代码片段)

作者:余香鑫前言了解OpenHarmony能够开发一些简单的OpenHarmony应用一、了解OpenHarmonyOpenHarmony是由开放原子开源基金会(OpenAtomFoundation)孵化及运营的开源项目,目标是面向全场景、全连接、全智能时代,搭建一个智能终端设备操作系统... 查看详情

openharmonyopensles音频录制(代码片段)

1、OpenHarmonyOpenSlES音频录制开发简介        开发者可以通过本博文了解在OpenHarmony中如何使用OpenSLES进行录音相关操作;当前仅实现了部分OpenSLES接口,因此调用未实现接口后会返回。SL_RESULT_FEATURE_UNSUPPORTED。 2、OpenHa... 查看详情

openharmony藏头诗应用(代码片段)

今天我们将做一个OpenHarmony趣味应用——OpenHarmony藏头诗应用,是通过AI接口来做。通过调用指定的AI接口来做,接口会返回藏头诗或者继续完成诗的后面几句。我要实现的功能主要有:生成藏头诗,生成整首诗ÿ... 查看详情

openharmony视频录制开发(代码片段)

1、OpenHarmony视频录制开发简介    视频录制的主要工作是捕获音视频信号,完成音视频编码并保存到文件中,帮助开发者轻松实现音视频录制功能,包括开始录制、暂停录制、恢复录制、停止录制、释放资源等功能... 查看详情

openharmony音频开发(代码片段)

1、OpenHarmony音频开发简介        音频播放开发的主要工作是将音频数据转码为可听见的音频模拟信号,并通过输出设备进行播放,同时对播放任务进行管理,包括开始播放、暂停播放、停止播放、释放资源、设... 查看详情

openharmony源码编译与下载(代码片段)

开发环境:开发系统:Ubuntu20.04开发板:Pegasus物联网开发板MCU:Hi3861OpenHarmony版本:3.0.1-LTS3.1OpenHarmony源码编译Pegasus物联网开发板基于华为海思Hi3861芯片,对应的产品名字是wifiiot_hispark_pegasus。接下来编译wi... 查看详情

openharmony源码编译与下载(代码片段)

开发环境:开发系统:Ubuntu20.04开发板:Pegasus物联网开发板MCU:Hi3861OpenHarmony版本:3.0.1-LTS3.1OpenHarmony源码编译Pegasus物联网开发板基于华为海思Hi3861芯片,对应的产品名字是wifiiot_hispark_pegasus。接下来编译wi... 查看详情

#盲盒+码##跟着小白一起学鸿蒙#如何编译openharmony自带app(代码片段)

...armony的主干代码是开源社区的重要学习资源,对于想进行应用开发和熟悉OpenHarmony能力的同学主干代码是非常重要的资源,在主干代码的applications目录里聚集了很多原生的应用实现,那么如何编译这些代码就是我们这篇文章的主... 查看详情