driver-hdf-servicemanage.md 7.1 KB
Newer Older
W
wenjun 已提交
1 2
# 驱动服务管理<a name="ZH-CN_TOPIC_0000001052777057"></a>

N
NEEN 已提交
3 4 5 6
-   [使用场景](#section14244270117)
-   [接口说明](#section1432412561722)
-   [开发步骤](#section393515164416)

W
wenjun 已提交
7 8
驱动服务是HDF驱动设备对外提供能力的对象,由HDF框架统一管理。驱动服务管理主要包含驱动服务的发布和获取。

N
NEEN 已提交
9
HDF框架定义了驱动对外发布服务的策略,是由配置文件中的policy字段来控制,policy字段的取值范围以及含义如下:
W
wenjun 已提交
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37

```
typedef enum {
    /* 驱动不提供服务 */
    SERVICE_POLICY_NONE = 0,
    /* 驱动对内核态发布服务 */
    SERVICE_POLICY_PUBLIC = 1,
    /* 驱动对内核态和用户态都发布服务 */
    SERVICE_POLICY_CAPACITY = 2,
    /* 驱动服务不对外发布服务,但可以被订阅 */
    SERVICE_POLICY_FRIENDLY = 3,
    /* 驱动私有服务不对外发布服务,也不能被订阅 */
    SERVICE_POLICY_PRIVATE = 4,
    /* 错误的服务策略 */
    SERVICE_POLICY_INVALID
} ServicePolicy;
```

## 使用场景<a name="section14244270117"></a>

当驱动以接口的形式对外提供能力时,可以使用HDF框架的驱动服务管理能力。

## 接口说明<a name="section1432412561722"></a>

针对驱动服务管理功能,HDF框架开放了以下接口供开发者调用,如下表所示:

**表 1**  服务管理接口

N
NEEN 已提交
38 39
<a name="table8431122013592"></a>
<table><thead align="left"><tr id="row13431820135919"><th class="cellrowborder" valign="top" width="50%" id="mcps1.2.3.1.1"><p id="p1670132714592"><a name="p1670132714592"></a><a name="p1670132714592"></a>方法</p>
W
wenjun 已提交
40
</th>
N
NEEN 已提交
41
<th class="cellrowborder" valign="top" width="50%" id="mcps1.2.3.1.2"><p id="p770172785910"><a name="p770172785910"></a><a name="p770172785910"></a>描述</p>
W
wenjun 已提交
42 43 44
</th>
</tr>
</thead>
N
NEEN 已提交
45
<tbody><tr id="row1743112017594"><td class="cellrowborder" valign="top" width="50%" headers="mcps1.2.3.1.1 "><p id="p18601333135911"><a name="p18601333135911"></a><a name="p18601333135911"></a>int32_t (*Bind)(struct HdfDeviceObject *deviceObject);</p>
W
wenjun 已提交
46
</td>
N
NEEN 已提交
47
<td class="cellrowborder" valign="top" width="50%" headers="mcps1.2.3.1.2 "><p id="p46015332591"><a name="p46015332591"></a><a name="p46015332591"></a>需要驱动开发者实现Bind函数,将自己的服务接口绑定到HDF框架中。</p>
W
wenjun 已提交
48 49
</td>
</tr>
N
NEEN 已提交
50
<tr id="row1543212045914"><td class="cellrowborder" valign="top" width="50%" headers="mcps1.2.3.1.1 "><p id="p19601163314590"><a name="p19601163314590"></a><a name="p19601163314590"></a>const struct HdfObject *DevSvcManagerClntGetService(const char *svcName);</p>
W
wenjun 已提交
51
</td>
N
NEEN 已提交
52
<td class="cellrowborder" valign="top" width="50%" headers="mcps1.2.3.1.2 "><p id="p1601123318598"><a name="p1601123318598"></a><a name="p1601123318598"></a>获取驱动的服务。</p>
W
wenjun 已提交
53 54
</td>
</tr>
N
NEEN 已提交
55 56
<tr id="row20432162019594"><td class="cellrowborder" valign="top" width="50%" headers="mcps1.2.3.1.1 "><p id="p960173310590"><a name="p960173310590"></a><a name="p960173310590"></a>int HdfDeviceSubscribeService(</p>
<p id="p126021533165915"><a name="p126021533165915"></a><a name="p126021533165915"></a>struct HdfDeviceObject *deviceObject, const char *serviceName, struct SubscriberCallback callback);</p>
W
wenjun 已提交
57
</td>
N
NEEN 已提交
58
<td class="cellrowborder" valign="top" width="50%" headers="mcps1.2.3.1.2 "><p id="p06029334597"><a name="p06029334597"></a><a name="p06029334597"></a>订阅驱动的服务。</p>
W
wenjun 已提交
59 60 61 62 63 64 65 66 67 68 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
</td>
</tr>
</tbody>
</table>

## 开发步骤<a name="section393515164416"></a>

驱动服务管理的开发包括驱动服务的编写、绑定、获取或者订阅,详细步骤如下。

1.  驱动服务发布。

    ```
    驱动服务结构的定义
    struct ISampleDriverService {
        struct IDeviceIoService ioService;   // 服务结构的首个成员必须是IDeviceIoService类型的成员
        int32_t (*ServiceA)(void);               // 驱动的第一个服务接口
        int32_t (*ServiceB)(uint32_t inputCode); // 驱动的第二个服务接口,有多个可以依次往下累加
    };
    
    驱动服务接口的实现
    int32_t SampleDriverServiceA(void)
    {
        // 驱动开发者实现业务逻辑
        return 0;
    }
    
    int32_t SampleDriverServiceB(uint32_t inputCode)
    {
        // 驱动开发者实现业务逻辑
        return 0;
    }
    ```

2.  驱动服务绑定到HDF框架中,实现HdfDriverEntry中的Bind指针函数。

    ```
    int32_t SampleDriverBind(struct HdfDeviceObject *deviceObject)
    {
        // deviceObject为HDF框架给每一个驱动创建的设备对象,用来保存设备相关的私有数据和服务接口
N
NEEN 已提交
98
        if (deviceObject == NULL) {
W
wenjun 已提交
99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130
            HDF_LOGE("Sample device object is null!");
            return -1;
        }
        static struct ISampleDriverService sampleDriverA = {
            .ServiceA = SampleDriverServiceA,
            .ServiceB = SampleDriverServiceB,
        };
        deviceObject->service = &sampleDriverA.ioService;
        return 0;
    }
    ```

3.  驱动服务获取。

    驱动服务的获取有两种方式,HDF框架提供接口直接获取和HDF框架提供订阅机制获取。

    -   通过HDF接口直接获取

        当明确驱动已经加载完成时,获取该驱动的服务可以通过HDF框架提供的能力接口直接获取,如下所示:

        ```
        const struct ISampleDriverService *sampleService =
                (const struct ISampleDriverService *)DevSvcManagerClntGetService("sample_driver");
        if (sampleService == NULL) {
            return -1;
        }
        sampleService->ServiceA();
        sampleService->ServiceB(5);
        ```

    -   通过HDF提供的订阅机制获取

N
NEEN 已提交
131
        当内核态对驱动(同一个host)加载的时机不感知时,可以通过HDF框架提供的订阅机制来订阅该驱动,当该驱动加载完成时,HDF框架会将被订阅的驱动服务发布给订阅者,实现方式如下所示:
W
wenjun 已提交
132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148

        ```
        // 订阅回调函数的编写,当被订阅的驱动加载完成后,HDF框架会将被订阅驱动的服务发布给订阅者,通过这个回调函数给订阅者使用
        // object为订阅者的私有数据,service为被订阅的服务对象
        int32_t TestDriverSubCallBack(struct HdfDeviceObject *deviceObject, const struct HdfObject *service)
        {
            const struct ISampleDriverService *sampleService =
                (const struct ISampleDriverService *)service;
            if (sampleService == NULL) {
                return -1;
            }
            sampleService->ServiceA();
            sampleService->ServiceB(5);
        }
        // 订阅过程的实现
        int32_t TestDriverInit(struct HdfDeviceObject *deviceObject)
        {
N
NEEN 已提交
149
            if (deviceObject == NULL) {
W
wenjun 已提交
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
                HDF_LOGE("Test driver init failed, deviceObject is null!");
                return -1;
            }
            struct SubscriberCallback callBack;
            callBack.deviceObject = deviceObject;
            callBack.OnServiceConnected = TestDriverSubCallBack;
            int32_t ret = HdfDeviceSubscribeService(deviceObject, "sample_driver", callBack);
            if (ret != 0) {
                HDF_LOGE("Test driver subscribe sample driver failed!");
            }
            return ret;
        }
        ```