• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2013-2023, Huawei Technologies Co., Ltd. All rights reserved.
3  * Copyright (c) 2020, Huawei Device Co., Ltd. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without modification,
6  * are permitted provided that the following conditions are met:
7  *
8  * 1. Redistributions of source code must retain the above copyright notice, this list of
9  *    conditions and the following disclaimer.
10  *
11  * 2. Redistributions in binary form must reproduce the above copyright notice, this list
12  *    of conditions and the following disclaimer in the documentation and/or other materials
13  *    provided with the distribution.
14  *
15  * 3. Neither the name of the copyright holder nor the names of its contributors may be used
16  *    to endorse or promote products derived from this software without specific prior written
17  *    permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
21  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
23  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
24  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
25  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
26  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
27  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
28  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
29  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  */
31 
32 #include "usb_init.h"
33 #include "usb_api_pri.h"
34 #include "devmgr_service.h"
35 #include "devsvc_manager_clnt.h"
36 #include "hdf_device_object.h"
37 
38 typedef struct usb_info {
39 	bool b_init;
40 	controller_type ctype;
41 	device_type dtype;
42 } usb_info_t;
43 
44 static char *dev_name = NULL;
45 static bool uvc_enable = false;
46 static usb_info_t usb_info = { false, (controller_type)0xFFFF, (device_type)0xFFFF };
47 static struct mtx usb_mtx = PTHREAD_MUTEX_INITIALIZER;
48 
49 static struct driver_module_data* usb_driver_mode_list[] = {
50 #ifdef LOSCFG_DRIVERS_USB_HOST_EHCI
51 	/* xxx controller modules */
52 	/* usb generial controller modules */
53 	&usbus_ehci_driver_mod,
54 #endif
55 
56 #if defined (LOSCFG_DRIVERS_USB_HOST_XHCI)
57 	&usbus_xhci_driver_mod,
58 #endif
59 
60 #ifdef LOSCFG_DRIVERS_USB_HOST_DRIVER
61 	/* xxx driver modules */
62 	&uhub_uhub_driver_mod,
63 	&uhub_usbus_driver_mod,
64 #endif
65 
66 #ifdef LOSCFG_DRIVERS_USB_4G_MODEM
67 #ifndef LOSCFG_DRIVERS_HDF_USB_DDK_HOST
68 	&cdce_uhub_driver_mod,
69 #endif
70 	//&bsd_u3g_uhub_driver_mod,
71 #endif
72 
73 #if defined (LOSCFG_DRIVERS_USB_SERIAL) || defined (LOSCFG_DRIVERS_USB_4G_MODEM)
74 	&u3g_uhub_driver_mod,
75 #endif
76 
77 #ifdef LOSCFG_DRIVERS_USB_MASS_STORAGE
78 	&umass_uhub_driver_mod,
79 #endif
80 
81 #ifdef LOSCFG_DRIVERS_USB_ETHERNET
82 	&axe_uhub_driver_mod,
83 	&axge_uhub_driver_mod,
84 #endif
85 
86 #ifdef LOSCFG_DRIVERS_USB_RNDIS_HOST
87 	&urndis_uhub_driver_mod,
88 #endif
89 
90 #ifdef LOSCFG_DRIVERS_USB_WIRELESS
91 	&usb_linux_uhub_driver_mod,
92 #endif
93 
94 #if defined (LOSCFG_DRIVERS_USB_HID_CLASS) && defined (LOSCFG_DRIVERS_HDF_INPUT)
95 	&uhid_uhub_driver_mod,
96 #endif
97 
98 #ifdef LOSCFG_DRIVERS_HDF_USB_DDK_DEVICE
99 	&composite_hiudc3_driver_mod,
100 #endif
101 	NULL
102 };
103 
104 extern device_t
105 bus_get_device(device_t dev, const char *name);
106 
107 void
usbinfo_clean(void)108 usbinfo_clean(void)
109 {
110 	dev_name = NULL;
111 	uvc_enable = false;
112 	usb_info.ctype = (controller_type)0xFFFF;
113 	usb_info.dtype = (device_type)0xFFFF;
114 	usb_info.b_init = false;
115 	return;
116 }
117 
118 char *
dev_name_get(void)119 dev_name_get(void)
120 {
121 	return (dev_name);
122 }
123 
124 static uint32_t
usb_loadonce(void)125 usb_loadonce(void)
126 {
127 	struct driver_module_data *data;
128 	uint32_t i;
129 
130 	dprintf("usb %s\n",fetach_usbversion());
131 
132 #ifdef LOSCFG_DRIVERS_USB
133 	/* init quirk */
134 	usb_quirk_init(NULL);
135 
136 	for (i = 0; (data = usb_driver_mode_list[i]) && (data != NULL); i++) {
137 		driver_module_handler(NULL, MOD_LOAD, data);
138 	}
139 #endif
140 
141 #ifdef LOSCFG_USB_DEBUG
142 	usb_debug_module_regsiter();
143 #endif
144 
145 	return (0);
146 }
147 
148 static void
usb_unloadonce(void)149 usb_unloadonce(void)
150 {
151 	uint32_t i;
152 
153 #ifdef LOSCFG_USB_DEBUG
154 	usb_debug_module_unregsiter();
155 #endif
156 
157 #ifdef LOSCFG_DRIVERS_USB
158 	for (i = 0; usb_driver_mode_list[i] != NULL; i++) {
159 		driver_module_handler(NULL, MOD_UNLOAD, usb_driver_mode_list[i]);
160 	}
161 #endif
162 
163 	return;
164 }
165 #ifdef LOSCFG_DRIVERS_HDF_USB_DDK_DEVICE
HdfLoadUsbDevice(const char * serv_name)166 static struct HdfDeviceObject *HdfLoadUsbDevice(const char *serv_name)
167 {
168 	struct IDevmgrService *devmgr = DevmgrServiceGetInstance();
169 	if(devmgr== NULL || devmgr->LoadDevice(devmgr, serv_name) != HDF_SUCCESS) {
170 		dprintf("failed to load %s", serv_name);
171 		return NULL;
172 	}
173 
174 	return DevSvcManagerClntGetDeviceObject(serv_name);
175 }
176 
HdfRegisterUsbDevice(struct HdfDeviceObject * usb_fn_master,const char * driver_name,const char * serv_name)177 struct HdfDeviceObject *HdfRegisterUsbDevice(struct HdfDeviceObject *usb_fn_master,
178 	const char *driver_name, const char *serv_name)
179 {
180 	struct HdfDeviceObject *dev = HdfDeviceObjectAlloc(usb_fn_master, driver_name);
181 	if (dev == NULL) {
182 		dprintf("%s: failed to alloc device object", __func__);
183 		return NULL;
184 	}
185 
186 	if (HdfDeviceObjectRegister(dev) != HDF_SUCCESS) {
187 		dprintf("%s: failed to regitst device %s", __func__, serv_name);
188 		HdfDeviceObjectRelease(dev);
189 		return NULL;
190 	}
191 
192 	if (HdfDeviceObjectPublishService(dev, serv_name, SERVICE_POLICY_PUBLIC, 0664) != HDF_SUCCESS) {
193 		dprintf("%s: failed to regitst device %s", __func__, serv_name);
194 		HdfDeviceObjectRelease(dev);
195 		return NULL;
196 	}
197 
198 	return dev;
199 }
200 
composite_add(void)201 static int composite_add(void)
202 {
203 	device_t udc;
204 	device_t composite;
205 
206 	udc = bus_get_device(nexus, "hiudc3");
207 	if (udc == NULL) {
208 		return -1;
209 	}
210 
211 	composite = device_add_child(udc, "composite", -1);
212 	if (composite == NULL) {
213 		return -1;
214 	}
215 
216 	if (device_probe_and_attach(composite)) {
217 		device_printf(composite, "WARNING: Probe and attach failed!\n");
218 		return -1;
219 	}
220 	struct HdfDeviceObject *usb_fn_dev = HdfLoadUsbDevice("usbfn");
221 	if (usb_fn_dev == NULL) {
222 		dprintf("%s register usbfn may failed\n", __func__);
223 	}
224 
225 	struct HdfDeviceObject *devobj = HdfRegisterUsbDevice(usb_fn_dev, "usbfn_cdcacm", "usbfn_cdcacm");
226 	if (devobj == NULL) {
227 		dprintf("%s register usbfn_cdcacm may failed\n", __func__);
228 	}
229 
230 	devobj = HdfRegisterUsbDevice(usb_fn_dev, "usbfn_cdcecm", "usbfn_cdcecm");
231 	if (devobj == NULL) {
232 		dprintf("%s register usbfn_cdcecm may failed\n", __func__);
233 	}
234 	dprintf("%s success\n", __func__);
235 	return 0;
236 }
237 #endif
238 /*
239  * step1: modify DRIVER_MODULE,register all driver module
240  * step2: make ehci/ohci device (direct skip pci bus)
241  * step3: insert ehci/ohci device into usb controller
242  * step4: create ehci/ohci root hub device
243  * step5: ehci/ohci transfer setup/start
244  */
245 uint32_t
usb_init(controller_type ctype,device_type dtype)246 usb_init(controller_type ctype, device_type dtype)
247 {
248 	uint32_t ret;
249 	static int usb_loaded = 0;
250 
251 	dprintf("\n******** %s in **********\n", __FUNCTION__);
252 
253 	mtx_lock(&usb_mtx);
254 	if (usb_info.b_init) {
255 		dprintf("\n duplicate usb_init %s, ctype:%d dtype:%d\n", __FUNCTION__, usb_info.ctype, usb_info.dtype);
256 		mtx_unlock(&usb_mtx);
257 		return (LOS_NOK);
258 	}
259 
260 	if (usb_loaded == 0) {
261 		ret = usb_loadonce();
262 		if (ret) {
263 			goto err;
264 		}
265 		usb_loaded = 1;
266 	}
267 
268 	usb_dev_init(NULL);
269 
270 	if (ctype == HOST) {
271 #if defined (LOSCFG_DRIVERS_USB_HOST_XHCI)
272 		ret = hixhci_init();
273 #endif
274 #ifdef LOSCFG_DRIVERS_USB_HOST_EHCI
275 		ret = hiehci_init();
276 #endif
277 	} else {
278 #ifdef LOSCFG_DRIVERS_HDF_USB_DDK_DEVICE
279 		ret = usbd_load_driver();
280 		if (ret != LOS_OK) {
281 			dprintf("usbd_load_driver failed ,ret = %d\n", ret);
282 			goto err;
283 		}
284 		ret = composite_add();
285 		if (ret != LOS_OK) {
286 			dprintf("composite_add failed ,ret = %d\n", ret);
287 			goto err;
288 		}
289 		ret = usbd_enable_interrupt();
290 		if (ret != LOS_OK) {
291 			dprintf("usbd_enable_interrupt failed, ret = %d\n", ret);
292 			goto err;
293 		}
294 #endif
295 	}
296 
297 	usb_info.b_init = true;
298 	usb_info.ctype = ctype;
299 	usb_info.dtype = dtype;
300 
301 	mtx_unlock(&usb_mtx);
302 	dprintf("******** %s ok**********\n\n", __FUNCTION__);
303 	return (LOS_OK);
304 
305 err:
306 	mtx_unlock(&usb_mtx);
307 	if (!usb_loaded) {
308 		usb_unloadonce();
309 	}
310 	dprintf("******** %s fail**********\n\n", __FUNCTION__);
311 
312 	return (LOS_NOK);
313 }
314 
315 uint32_t
usb_deinit(void)316 usb_deinit(void)
317 {
318 	uint32_t ret = LOS_OK;
319 
320 	dprintf("******** %s in **********\n\n", __FUNCTION__);
321 	mtx_lock(&usb_mtx);
322 
323 	if (usb_info.b_init == false) {
324 		dprintf("******** %s out, no init **********\n\n", __FUNCTION__);
325 		goto err;
326 	}
327 
328 	if (usb_info.ctype == HOST) {
329 		dprintf("******** %s fail, host not support **********\n\n", __FUNCTION__);
330 		goto err;
331 	}
332 
333 	if (ret) {
334 		dprintf("******** %s fail, %d **********\n\n", __FUNCTION__, ret);
335 		goto err;
336 	}
337 
338 	usb_dev_uninit(NULL);
339 
340 	dprintf(" ** %s uninit success **\n", dev_name_get());
341 	usbinfo_clean();
342 	mtx_unlock(&usb_mtx);
343 	return (LOS_OK);
344 
345 err:
346 	mtx_unlock(&usb_mtx);
347 	return (LOS_NOK);
348 }
349 
350 bool
usb_is_devicemode(void)351 usb_is_devicemode(void)
352 {
353 	return (HiUsbIsDeviceMode());
354 }
355 
356