1 /*
2 * Copyright (c) 2021-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 "devmgr_service_proxy.h"
17 #include "devhost_service_stub.h"
18 #include "devmgr_service_if.h"
19 #include "devmgr_service_stub.h"
20 #include "devsvc_manager_clnt.h"
21 #include "hdf_base.h"
22 #include "hdf_log.h"
23 #include "hdf_sbuf.h"
24 #include "osal_mem.h"
25
26 #define HDF_LOG_TAG devmgr_service_proxy
27
DevmgrServiceProxyAttachDeviceHost(struct IDevmgrService * inst,uint16_t hostId,struct IDevHostService * service)28 int DevmgrServiceProxyAttachDeviceHost(struct IDevmgrService *inst, uint16_t hostId, struct IDevHostService *service)
29 {
30 int status = HDF_FAILURE;
31 struct HdfSBuf *data = HdfSbufTypedObtain(SBUF_IPC);
32 struct HdfSBuf *reply = HdfSbufTypedObtain(SBUF_IPC);
33 struct HdfRemoteDispatcher *dipatcher = NULL;
34 struct HdfRemoteService *remoteService = NULL;
35 struct DevmgrServiceProxy *serviceProxy = (struct DevmgrServiceProxy *)inst;
36 struct DevHostServiceStub *hostStub = (struct DevHostServiceStub *)service;
37 if ((serviceProxy->remote == NULL) || (data == NULL) || (reply == NULL) || (hostStub == NULL)) {
38 HDF_LOGE("DevmgrServiceProxyAttachDeviceHost failed, host id is %{public}u", hostId);
39 goto FINISHED;
40 }
41 remoteService = serviceProxy->remote;
42 dipatcher = remoteService->dispatcher;
43 if (!HdfRemoteServiceWriteInterfaceToken(remoteService, data) || !HdfSbufWriteInt32(data, hostId) ||
44 HdfSbufWriteRemoteService(data, hostStub->remote) != HDF_SUCCESS) {
45 HDF_LOGE("failed to attach host, write sbuf error");
46 goto FINISHED;
47 }
48 status = dipatcher->Dispatch(remoteService, DEVMGR_SERVICE_ATTACH_DEVICE_HOST, data, reply);
49 HDF_LOGI("Attach device host dispatch finish, status is %{public}d", status);
50 FINISHED:
51 HdfSbufRecycle(reply);
52 HdfSbufRecycle(data);
53 return status;
54 }
55
DevmgrServiceProxyAttachDevice(struct IDevmgrService * inst,struct IHdfDeviceToken * token)56 int DevmgrServiceProxyAttachDevice(struct IDevmgrService *inst, struct IHdfDeviceToken *token)
57 {
58 int status = HDF_FAILURE;
59 struct HdfSBuf *data = HdfSbufTypedObtain(SBUF_IPC);
60 struct HdfSBuf *reply = HdfSbufTypedObtain(SBUF_IPC);
61 struct DevmgrServiceProxy *serviceProxy = (struct DevmgrServiceProxy *)inst;
62 if (serviceProxy == NULL || serviceProxy->remote == NULL || data == NULL || reply == NULL || token == NULL) {
63 HDF_LOGE("DevmgrServiceProxyAttachDevice failed");
64 goto FINISHED;
65 }
66 struct HdfRemoteService *remoteService = serviceProxy->remote;
67 const char *srvName = (token->servName == NULL) ? "" : token->servName;
68 const char *deviceName = (token->deviceName == NULL) ? "" : token->deviceName;
69 if (!HdfRemoteServiceWriteInterfaceToken(remoteService, data) || !HdfSbufWriteInt32(data, token->devid) ||
70 !HdfSbufWriteString(data, srvName) || !HdfSbufWriteString(data, deviceName)) {
71 goto FINISHED;
72 }
73
74 status = remoteService->dispatcher->Dispatch(remoteService, DEVMGR_SERVICE_ATTACH_DEVICE, data, reply);
75 FINISHED:
76 HdfSbufRecycle(reply);
77 HdfSbufRecycle(data);
78 return status;
79 }
80
DevmgrServiceProxyDetachDevice(struct IDevmgrService * inst,devid_t devid)81 int DevmgrServiceProxyDetachDevice(struct IDevmgrService *inst, devid_t devid)
82 {
83 int status = HDF_FAILURE;
84 struct HdfSBuf *data = HdfSbufTypedObtain(SBUF_IPC);
85 struct HdfSBuf *reply = HdfSbufTypedObtain(SBUF_IPC);
86 struct DevmgrServiceProxy *serviceProxy = (struct DevmgrServiceProxy *)inst;
87 if (serviceProxy == NULL || serviceProxy->remote == NULL || data == NULL || reply == NULL) {
88 HDF_LOGE("DevmgrServiceProxyDetachDevice failed");
89 goto FINISHED;
90 }
91 struct HdfRemoteService *remoteService = serviceProxy->remote;
92 if (!HdfRemoteServiceWriteInterfaceToken(remoteService, data) || !HdfSbufWriteInt32(data, devid)) {
93 goto FINISHED;
94 }
95
96 status = remoteService->dispatcher->Dispatch(remoteService, DEVMGR_SERVICE_DETACH_DEVICE, data, reply);
97 FINISHED:
98 HdfSbufRecycle(reply);
99 HdfSbufRecycle(data);
100 return status;
101 }
102
DevmgrServiceProxyProcessLoad(struct IDevmgrService * inst,const char * svcName,int code)103 static int DevmgrServiceProxyProcessLoad(struct IDevmgrService *inst, const char *svcName, int code)
104 {
105 struct DevmgrServiceProxy *serviceProxy = (struct DevmgrServiceProxy *)inst;
106 if (serviceProxy == NULL || serviceProxy->remote == NULL || svcName == NULL) {
107 HDF_LOGE("DevmgrServiceProxyProcessLoad code: %{public}d failed", code);
108 return HDF_ERR_INVALID_PARAM;
109 }
110
111 struct HdfSBuf *data = HdfSbufTypedObtain(SBUF_IPC);
112 struct HdfRemoteService *remoteService = serviceProxy->remote;
113 if (!HdfRemoteServiceWriteInterfaceToken(remoteService, data) || !HdfSbufWriteString(data, svcName)) {
114 HdfSbufRecycle(data);
115 return HDF_FAILURE;
116 }
117
118 int status = remoteService->dispatcher->Dispatch(remoteService, code, data, NULL);
119 HdfSbufRecycle(data);
120 return status;
121 }
122
DevmgrServiceProxyLoadDevice(struct IDevmgrService * inst,const char * svcName)123 int DevmgrServiceProxyLoadDevice(struct IDevmgrService *inst, const char *svcName)
124 {
125 return DevmgrServiceProxyProcessLoad(inst, svcName, DEVMGR_SERVICE_LOAD_DEVICE);
126 }
127
DevmgrServiceProxyUnLoadDevice(struct IDevmgrService * inst,const char * svcName)128 int DevmgrServiceProxyUnLoadDevice(struct IDevmgrService *inst, const char *svcName)
129 {
130 return DevmgrServiceProxyProcessLoad(inst, svcName, DEVMGR_SERVICE_UNLOAD_DEVICE);
131 }
132
DevmgrServiceProxyConstruct(struct DevmgrServiceProxy * inst)133 static void DevmgrServiceProxyConstruct(struct DevmgrServiceProxy *inst)
134 {
135 struct IDevmgrService *pvtbl = (struct IDevmgrService *)inst;
136 pvtbl->AttachDeviceHost = DevmgrServiceProxyAttachDeviceHost;
137 pvtbl->AttachDevice = DevmgrServiceProxyAttachDevice;
138 pvtbl->DetachDevice = DevmgrServiceProxyDetachDevice;
139 pvtbl->LoadDevice = DevmgrServiceProxyLoadDevice;
140 pvtbl->UnloadDevice = DevmgrServiceProxyUnLoadDevice;
141 pvtbl->StartService = NULL;
142 }
143
DevmgrServiceProxyObtain(struct HdfRemoteService * service)144 static struct IDevmgrService *DevmgrServiceProxyObtain(struct HdfRemoteService *service)
145 {
146 if (service != NULL) {
147 struct DevmgrServiceProxy *demgrServicProxy =
148 (struct DevmgrServiceProxy *)OsalMemCalloc(sizeof(struct DevmgrServiceProxy));
149 if (demgrServicProxy != NULL) {
150 demgrServicProxy->remote = service;
151 DevmgrServiceProxyConstruct(demgrServicProxy);
152 return &demgrServicProxy->super;
153 }
154 }
155 HDF_LOGE("DevmgrServiceProxyObtain failed");
156 return NULL;
157 }
158
DevmgrServiceProxyCreate(void)159 struct HdfObject *DevmgrServiceProxyCreate(void)
160 {
161 static struct IDevmgrService *instance = NULL;
162 if (instance != NULL) {
163 return (struct HdfObject *)instance;
164 }
165 struct IDevSvcManager *serviceManagerIf = NULL;
166 struct DevSvcManagerClnt *serviceManager = (struct DevSvcManagerClnt *)DevSvcManagerClntGetInstance();
167 if ((serviceManager == NULL) || (serviceManager->devSvcMgrIf == NULL)) {
168 HDF_LOGE("Fail to Create Service Manager Client");
169 return NULL;
170 }
171 serviceManagerIf = serviceManager->devSvcMgrIf;
172 if (serviceManagerIf->GetService == NULL) {
173 HDF_LOGE("Get Service is not implement!!!");
174 return NULL;
175 }
176 struct HdfRemoteService *remote =
177 (struct HdfRemoteService *)serviceManagerIf->GetService(serviceManagerIf, DEVICE_MANAGER_SERVICE);
178 if (remote != NULL) {
179 if (!HdfRemoteServiceSetInterfaceDesc(remote, "HDI.IDeviceManager.V1_0")) {
180 HDF_LOGE("%{public}s: failed to init interface desc", __func__);
181 HdfRemoteServiceRecycle(remote);
182 return NULL;
183 }
184 instance = DevmgrServiceProxyObtain(remote);
185 }
186
187 return (struct HdfObject *)instance;
188 }
189
DevmgrServiceProxyRelease(struct HdfObject * object)190 void DevmgrServiceProxyRelease(struct HdfObject *object)
191 {
192 struct DevmgrServiceProxy *instance = (struct DevmgrServiceProxy *)object;
193 if (instance != NULL) {
194 if (instance->remote != NULL) {
195 HdfRemoteServiceRecycle(instance->remote);
196 instance->remote = NULL;
197 }
198 OsalMemFree(instance);
199 }
200 }
201