driver-hdf-development.md 11.3 KB
Newer Older
W
wenjun 已提交
1 2
# 驱动开发<a name="ZH-CN_TOPIC_0000001051930361"></a>

N
NEEN 已提交
3 4 5
-   [驱动模型介绍](#section157425168112)
-   [驱动开发步骤](#section1969312275533)

W
wenjun 已提交
6 7
## 驱动模型介绍<a name="section157425168112"></a>

D
duangavin123 已提交
8
HDF框架以组件化的驱动模型作为核心设计思路,为开发者提供更精细化的驱动管理,让驱动开发和部署更加规范。HDF框架将一类设备驱动放在同一个host里面,开发者也可以将驱动功能分层独立开发和部署,支持一个驱动多个node,HDF驱动模型如下图所示:
W
wenjun 已提交
9

D
duangavin123 已提交
10 11
**图 1**  HDF驱动模型<a name="fig3580184214210"></a>  
![](figures/HDF驱动模型.png "HDF驱动模型")
W
wenjun 已提交
12 13 14 15 16 17 18 19 20 21 22 23 24 25 26

## 驱动开发步骤<a name="section1969312275533"></a>

基于HDF框架进行驱动的开发主要分为两个部分,驱动实现和驱动配置,详细开发流程如下所示:

1.  <a name="li35182436435"></a>驱动实现

    驱动实现包含驱动业务代码和驱动入口注册,具体写法如下:

    -   驱动业务代码

        ```
        #include "hdf_device_desc.h"  // HDF框架对驱动开放相关能力接口的头文件
        #include "hdf_log.h"          // HDF 框架提供的日志接口头文件
        
N
NEEN 已提交
27
        #define HDF_LOG_TAG "sample_driver"   // 打印日志所包含的标签,如果不定义则用默认定义的HDF_TAG标签
W
wenjun 已提交
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
        
        //驱动对外提供的服务能力,将相关的服务接口绑定到HDF框架
        int32_t HdfSampleDriverBind(struct HdfDeviceObject *deviceObject)
        {
            HDF_LOGD("Sample driver bind success");
            return 0;
        }
        
        // 驱动自身业务初始的接口
        int32_t HdfSampleDriverInit(struct HdfDeviceObject *deviceObject)
        {
            HDF_LOGD("Sample driver Init success");
            return 0;
        }
        
        // 驱动资源释放的接口
        void HdfSampleDriverRelease(struct HdfDeviceObject *deviceObject)
        {
            HDF_LOGD("Sample driver release success");
            return;
        }
        ```

    -   驱动入口注册到HDF框架

        ```
        // 定义驱动入口的对象,必须为HdfDriverEntry(在hdf_device_desc.h中定义)类型的全局变量
        struct HdfDriverEntry g_sampleDriverEntry = {
            .moduleVersion = 1,
            .moduleName = "sample_driver",
            .Bind = HdfSampleDriverBind,
            .Init = HdfSampleDriverInit,
            .Release = HdfSampleDriverRelease,
        };
        
        // 调用HDF_INIT将驱动入口注册到HDF框架中,在加载驱动时HDF框架会先调用Bind函数,再调用Init函数加载该驱动,当Init调用异常时,HDF框架会调用Release释放驱动资源并退出。
        HDF_INIT(g_sampleDriverEntry);
        ```

Z
zhang 已提交
67 68
2. 驱动编译

Z
zhang 已提交
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
   - liteos

     涉及makefile和BUILD.gn修改:

     * makefile部分:

       驱动代码的编译必须要使用HDF框架提供的Makefile模板进行编译。

       ```
       include $(LITEOSTOPDIR)/../../drivers/adapter/khdf/liteos/lite.mk #导入hdf预定义内容,必需
       MODULE_NAME :=    #生成的结果文件
       LOCAL_INCLUDE :=  #本驱动的头文件目录
       LOCAL_SRCS :=     #本驱动的源代码文件
       LOCAL_CFLAGS :=  #自定义的编译选项
       include $(HDF_DRIVER) #导入模板makefile完成编译
       ```

       编译结果文件链接到内核镜像,添加到drivers/adapter/khdf/liteos目录下的hdf_lite.mk里面,示例如下:

       ```
       LITEOS_BASELIB +=  -lxxx  #链接生成的静态库
       LIB_SUBDIRS    +=         #驱动代码Makefile的目录
       ```

     * BUILD.gn部分:

       添加模块BUILD.gn参考定义如下内容:

       ```
       import("//build/lite/config/component/lite_component.gni")
       import("//drivers/adapter/khdf/liteos/hdf.gni")
Z
zhang 已提交
100
       module_switch = defined(LOSCFG_DRIVERS_HDF_xxx)
Z
zhang 已提交
101 102
       module_name = "xxx"
       hdf_driver(module_name) {
Z
zhang 已提交
103 104 105 106
           sources = [
               "xxx/xxx/xxx.c", #模块要编译的源码文件
           ]
           public_configs = [ ":public" ] #使用依赖的头文件配置
Z
zhang 已提交
107
       }
Z
zhang 已提交
108 109
       config("public") {  #定义依赖的头文件配置
           include_dirs = [
110
               "xxx/xxx/xxx", #依赖的头文件目录
Z
zhang 已提交
111
           ]
Z
zhang 已提交
112 113 114
       }
       ```

Z
zhang 已提交
115
       把新增模块的BUILD.gn所在的目录添加到/drivers/adapter/khdf/liteos/BUILD.gn里面:
Z
zhang 已提交
116

Z
zhang 已提交
117 118
       ```
       group("liteos") {
119 120 121 122
           public_deps = [ ":$module_name" ]
           deps = [
               "xxx/xxx",   #新增模块BUILD.gn所在的目录,目录结构相对于/drivers/adapter/khdf/liteos
           ]
Z
zhang 已提交
123 124
       }
       ```
Z
zhang 已提交
125

Z
zhang 已提交
126
   - linux
Z
zhang 已提交
127

Z
zhang 已提交
128
     如果需要定义模块控制宏,需要在模块目录xxx里面添加Kconfig文件,并把Kconfig文件路径添加到drivers/adapter/khdf/linux/Kconfig里面:
Z
zhang 已提交
129

Z
zhang 已提交
130
     ```
Z
zhang 已提交
131
     source "drivers/hdf/khdf/xxx/Kconfig" #目录为hdf模块软链接到kernel里面的目录
Z
zhang 已提交
132
     ```
Z
zhang 已提交
133

Z
zhang 已提交
134
     添加模块目录到drivers/adapter/khdf/linux/Makefile:
Z
zhang 已提交
135

Z
zhang 已提交
136 137 138
     ```
     obj-$(CONFIG_DRIVERS_HDF)  += xxx/
     ```
Z
zhang 已提交
139

Z
zhang 已提交
140
     在模块目录xxx里面添加Makefile文件,在Makefile文件里面添加模块代码编译规则:
Z
zhang 已提交
141

Z
zhang 已提交
142 143 144
     ```
     obj-y  += xxx.o
     ```
Z
zhang 已提交
145 146 147



W
wenjun 已提交
148 149 150

3.  驱动配置

D
duangavin123 已提交
151
    HDF使用HCS作为配置描述源码,HCS详细介绍参考[配置管理](driver-hdf-manage.md)介绍。
W
wenjun 已提交
152 153 154 155 156

    驱动配置包含两部分,HDF框架定义的驱动设备描述和驱动的私有配置信息,具体写法如下:

    -   驱动设备描述(必选)

N
NEEN 已提交
157
        HDF框架加载驱动所需要的信息来源于HDF框架定义的驱动设备描述,因此基于HDF框架开发的驱动必须要在HDF框架定义的device\_info.hcs配置文件中添加对应的设备描述,驱动的设备描述填写如下所示:
W
wenjun 已提交
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217

        ```
        root {
            device_info {
                match_attr = "hdf_manager";
                template host {       // host模板,继承该模板的节点(如下sample_host)如果使用模板中的默认值,则节点字段可以缺省
                    hostName = "";
                    priority = 100;
                    template device {
                        template deviceNode {
                            policy = 0;
                            priority = 100;
                            preload = 0;
                            permission = 0664;
                            moduleName = "";
                            serviceName = "";
                            deviceMatchAttr = "";
                        }
                    }
                }
                sample_host :: host{
                    hostName = "host0";    // host名称,host节点是用来存放某一类驱动的容器
                    priority = 100;        // host启动优先级(0-200),值越大优先级越低,建议默认配100,优先级相同则不保证host的加载顺序
                    device_sample :: device {        // sample设备节点
                        device0 :: deviceNode {      // sample驱动的DeviceNode节点
                            policy = 1;              // policy字段是驱动服务发布的策略,在驱动服务管理章节有详细介绍
                            priority = 100;          // 驱动启动优先级(0-200),值越大优先级越低,建议默认配100,优先级相同则不保证device的加载顺序
                            preload = 0;             // 驱动按需加载字段,在本章节最后的说明有详细介绍
                            permission = 0664;       // 驱动创建设备节点权限
                            moduleName = "sample_driver";   // 驱动名称,该字段的值必须和驱动入口结构的moduleName值一致
                            serviceName = "sample_service";    // 驱动对外发布服务的名称,必须唯一
                            deviceMatchAttr = "sample_config"; // 驱动私有数据匹配的关键字,必须和驱动私有数据配置表中的match_attr值相等
                        }
                    }
                }
            }
        }
        ```

    -   驱动私有配置信息(可选)

        如果驱动有私有配置,则可以添加一个驱动的配置文件,用来填写一些驱动的默认配置信息,HDF框架在加载驱动的时候,会将对应的配置信息获取并保存在HdfDeviceObject 中的property里面,通过Bind和Init(参考[驱动开发](#li35182436435))传递给驱动,驱动的配置信息示例如下:

        ```
        root {
            SampleDriverConfig {
                sample_version = 1;
                sample_bus = "I2C_0";
                match_attr = "sample_config";   //该字段的值必须和device_info.hcs中的deviceMatchAttr值一致
            }
        }
        ```

        配置信息定义之后,需要将该配置文件添加到板级配置入口文件hdf.hcs(这一块可以通过OpenHarmony驱动子系统在DevEco集成驱动开发套件工具一键式配置,具体使用方法参考驱动开发套件中的介绍),示例如下:

        ```
        #include "device_info/device_info.hcs"
        #include "sample/sample_config.hcs"
        ```

218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236
4. 用户态驱动服务启动配置

   用户态需要把驱动服务配置到文件drivers/adapter/uhdf2/host/hdf_devhostmusl.cfg中,如下:

   ```
   {
       "name" : "sample_host", //驱动服务进程名字,和device_info.hcs中配置的hostName对应
       "dynamic" : true, //动态加载,目前驱动服务只支持动态加载,即由hdf_devmgr在初始化时调用init模块接口启动
       "path" : ["/vendor/bin/hdf_devhost"],//hdf_devhost所在的目录
       "uid" : "sample_host",//进程的用户ID
       "gid" : ["sample_host"],//进程的组ID
       "caps" : ["DAC_OVERRIDE", "DAC_READ_SEARCH"]//进程的Linux capabilities配置
   }
   ```

   进程的用户ID在文件base/startup/init_lite/services/etc/passwd中配置,进程的组ID在文件base/startup/init_lite/services/etc/group中配置,进程用户ID和组ID配置参考:[系统服务用户组添加方法](https://gitee.com/openharmony/startup_init_lite/wikis)



W
wenjun 已提交
237 238


D
duangavin123 已提交
239
>![](../public_sys-resources/icon-note.gif) **说明:** 
W
wenjun 已提交
240 241 242 243 244
>驱动加载方式支持按需加载和按序加载两种方式,具体使用方法如下:
>-   按需加载
>    ```
>    typedef enum {
>        DEVICE_PRELOAD_ENABLE = 0,
M
mamingshuai 已提交
245
>        DEVICE_PRELOAD_ENABLE_STEP2,
W
wenjun 已提交
246 247 248 249
>        DEVICE_PRELOAD_DISABLE,
>        DEVICE_PRELOAD_INVALID
>    } DevicePreload;
>    ```
D
duangavin123 已提交
250
>    配置文件中preload 字段配成 0 (DEVICE\_PRELOAD\_ENABLE ),则系统启动过程中默认加载;配成1(DEVICE\_PRELOAD\_ENABLE\_STEP2),当系统支持快启的时候,则在系统完成之后再加载这一类驱动,否则和DEVICE\_PRELOAD\_ENABLE 含义相同;配成2(DEVICE\_PRELOAD\_DISABLE),则系统启动过程中默认不加载,支持后续动态加载,当用户态获取驱动服务(参考[消息机制](driver-hdf-message-management.md))时,如果驱动服务不存在时,HDF框架会尝试动态加载该驱动。
W
wenjun 已提交
251 252 253
>-   按序加载(需要驱动为默认加载)
>    配置文件中的priority(取值范围为整数0到200)是用来表示host和驱动的优先级,不同的host内的驱动,host的priority值越小,驱动加载优先级越高;同一个host内驱动的priority值越小,加载优先级越高。