1 /*
2  * Copyright (c) 2022 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include <hdf_base.h>
17 #include <hdf_device_desc.h>
18 #include <hdf_log.h>
19 #include <hdf_sbuf_ipc.h>
20 
21 #include "dcamera_provider.h"
22 #include "v1_1/dcamera_provider_stub.h"
23 
24 #include <shared_mutex>
25 using namespace OHOS::HDI::DistributedCamera::V1_1;
26 
27 namespace {
28     std::shared_mutex mutex_;
29 }
30 
31 struct HdfDCameraProviderHost {
32     struct IDeviceIoService ioService;
33     OHOS::sptr<OHOS::IRemoteObject> stub;
34 };
35 
DCameraProviderDriverDispatch(struct HdfDeviceIoClient * client,int cmdId,struct HdfSBuf * data,struct HdfSBuf * reply)36 static int32_t DCameraProviderDriverDispatch(struct HdfDeviceIoClient *client, int cmdId, struct HdfSBuf *data,
37     struct HdfSBuf *reply)
38 {
39     OHOS::MessageParcel *dataParcel = nullptr;
40     OHOS::MessageParcel *replyParcel = nullptr;
41     OHOS::MessageOption option;
42 
43     if (SbufToParcel(data, &dataParcel) != HDF_SUCCESS) {
44         HDF_LOGE("%{public}s:invalid data sbuf object to dispatch", __func__);
45         return HDF_ERR_INVALID_PARAM;
46     }
47     if (SbufToParcel(reply, &replyParcel) != HDF_SUCCESS) {
48         HDF_LOGE("%{public}s:invalid reply sbuf object to dispatch", __func__);
49         return HDF_ERR_INVALID_PARAM;
50     }
51 
52     std::shared_lock lock(mutex_);
53     if (client == nullptr || client->device == nullptr || client->device->service == nullptr) {
54         HDF_LOGE("%{public}s: client or client.device or service is nullptr", __func__);
55         return HDF_FAILURE;
56     }
57     auto *hdfDCameraProviderHost = CONTAINER_OF(client->device->service, struct HdfDCameraProviderHost, ioService);
58     if (hdfDCameraProviderHost == NULL || hdfDCameraProviderHost->stub == NULL) {
59         HDF_LOGE("%{public}s:invalid hdfDCameraProviderHost", __func__);
60         return HDF_ERR_INVALID_PARAM;
61     }
62     return hdfDCameraProviderHost->stub->SendRequest(cmdId, *dataParcel, *replyParcel, option);
63 }
64 
HdfDCameraProviderDriverInit(struct HdfDeviceObject * deviceObject)65 static int HdfDCameraProviderDriverInit(struct HdfDeviceObject *deviceObject)
66 {
67     HDF_LOGI("HdfDCameraProviderDriverInit enter");
68     if (deviceObject == nullptr) {
69         HDF_LOGE("HdfDCameraProviderDriverInit:: HdfDeviceObject is NULL !");
70         return HDF_FAILURE;
71     }
72 
73     if (!HdfDeviceSetClass(deviceObject, DEVICE_CLASS_CAMERA)) {
74         HDF_LOGE("HdfDCameraProviderDriverInit set camera class failed");
75         return HDF_FAILURE;
76     }
77     return HDF_SUCCESS;
78 }
79 
HdfDCameraProviderDriverBind(struct HdfDeviceObject * deviceObject)80 static int HdfDCameraProviderDriverBind(struct HdfDeviceObject *deviceObject)
81 {
82     HDF_LOGI("HdfDCameraProviderDriverBind enter");
83 
84     auto *hdfDCameraProviderHost = new (std::nothrow) HdfDCameraProviderHost;
85     if (hdfDCameraProviderHost == nullptr) {
86         HDF_LOGE("%{public}s: failed to create create HdfDCameraProviderHost object", __func__);
87         return HDF_FAILURE;
88     }
89 
90     hdfDCameraProviderHost->ioService.Dispatch = DCameraProviderDriverDispatch;
91     hdfDCameraProviderHost->ioService.Open = NULL;
92     hdfDCameraProviderHost->ioService.Release = NULL;
93 
94     auto serviceImpl = OHOS::DistributedHardware::DCameraProvider::GetInstance();
95     if (serviceImpl == nullptr) {
96         HDF_LOGE("%{public}s: failed to get of implement service", __func__);
97         delete hdfDCameraProviderHost;
98         return HDF_FAILURE;
99     }
100 
101     hdfDCameraProviderHost->stub = OHOS::HDI::ObjectCollector::GetInstance().GetOrNewObject(serviceImpl,
102         IDCameraProvider::GetDescriptor());
103     if (hdfDCameraProviderHost->stub == nullptr) {
104         HDF_LOGE("%{public}s: failed to get stub object", __func__);
105         delete hdfDCameraProviderHost;
106         return HDF_FAILURE;
107     }
108 
109     deviceObject->service = &hdfDCameraProviderHost->ioService;
110     return HDF_SUCCESS;
111 }
112 
HdfDCameraProviderDriverRelease(struct HdfDeviceObject * deviceObject)113 static void HdfDCameraProviderDriverRelease(struct HdfDeviceObject *deviceObject)
114 {
115     HDF_LOGI("HdfDCameraProviderDriverRelease enter");
116     if (deviceObject == nullptr || deviceObject->service == nullptr) {
117         HDF_LOGE("%{public}s: params invalid.", __func__);
118         return;
119     }
120 
121     std::unique_lock lock(mutex_);
122     auto *hdfDCameraProviderHost = CONTAINER_OF(deviceObject->service, struct HdfDCameraProviderHost, ioService);
123     if (hdfDCameraProviderHost != nullptr) {
124         hdfDCameraProviderHost->stub = nullptr;
125     }
126     delete hdfDCameraProviderHost;
127     hdfDCameraProviderHost = nullptr;
128     if (deviceObject != nullptr) {
129         deviceObject->service = nullptr;
130     }
131 }
132 
133 static struct HdfDriverEntry g_dcameraproviderDriverEntry = {
134     .moduleVersion = 1,
135     .moduleName = "distributed_camera_provider_service",
136     .Bind = HdfDCameraProviderDriverBind,
137     .Init = HdfDCameraProviderDriverInit,
138     .Release = HdfDCameraProviderDriverRelease,
139 };
140 
141 #ifdef __cplusplus
142 extern "C" {
143 #endif /* __cplusplus */
144 HDF_INIT(g_dcameraproviderDriverEntry);
145 #ifdef __cplusplus
146 }
147 #endif /* __cplusplus */
148