xref: /freebsd/lib/libusb/libusb20_ugen20.c (revision bdafb02fcb88389fd1ab684cfe734cb429d35618)
1 /* $FreeBSD$ */
2 /*-
3  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
4  *
5  * Copyright (c) 2008 Hans Petter Selasky. All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26  * SUCH DAMAGE.
27  */
28 
29 #ifdef LIBUSB_GLOBAL_INCLUDE_FILE
30 #include LIBUSB_GLOBAL_INCLUDE_FILE
31 #else
32 #include <errno.h>
33 #include <fcntl.h>
34 #include <stdio.h>
35 #include <stdlib.h>
36 #include <string.h>
37 #include <unistd.h>
38 #include <time.h>
39 #include <sys/queue.h>
40 #include <sys/types.h>
41 #endif
42 
43 #include <dev/usb/usb.h>
44 #include <dev/usb/usbdi.h>
45 #include <dev/usb/usb_ioctl.h>
46 
47 #include "libusb20.h"
48 #include "libusb20_desc.h"
49 #include "libusb20_int.h"
50 
51 #ifndef	IOUSB
52 #define IOUSB(a) a
53 #endif
54 
55 static libusb20_init_backend_t ugen20_init_backend;
56 static libusb20_open_device_t ugen20_open_device;
57 static libusb20_close_device_t ugen20_close_device;
58 static libusb20_get_backend_name_t ugen20_get_backend_name;
59 static libusb20_exit_backend_t ugen20_exit_backend;
60 static libusb20_dev_get_iface_desc_t ugen20_dev_get_iface_desc;
61 static libusb20_dev_get_info_t ugen20_dev_get_info;
62 static libusb20_root_get_dev_quirk_t ugen20_root_get_dev_quirk;
63 static libusb20_root_get_quirk_name_t ugen20_root_get_quirk_name;
64 static libusb20_root_add_dev_quirk_t ugen20_root_add_dev_quirk;
65 static libusb20_root_remove_dev_quirk_t ugen20_root_remove_dev_quirk;
66 static libusb20_root_set_template_t ugen20_root_set_template;
67 static libusb20_root_get_template_t ugen20_root_get_template;
68 
69 const struct libusb20_backend_methods libusb20_ugen20_backend = {
70 	LIBUSB20_BACKEND(LIBUSB20_DECLARE, ugen20)
71 };
72 
73 /* USB device specific */
74 static libusb20_get_config_desc_full_t ugen20_get_config_desc_full;
75 static libusb20_get_config_index_t ugen20_get_config_index;
76 static libusb20_set_config_index_t ugen20_set_config_index;
77 static libusb20_set_alt_index_t ugen20_set_alt_index;
78 static libusb20_reset_device_t ugen20_reset_device;
79 static libusb20_check_connected_t ugen20_check_connected;
80 static libusb20_set_power_mode_t ugen20_set_power_mode;
81 static libusb20_get_power_mode_t ugen20_get_power_mode;
82 static libusb20_get_power_usage_t ugen20_get_power_usage;
83 static libusb20_kernel_driver_active_t ugen20_kernel_driver_active;
84 static libusb20_detach_kernel_driver_t ugen20_detach_kernel_driver;
85 static libusb20_do_request_sync_t ugen20_do_request_sync;
86 static libusb20_process_t ugen20_process;
87 
88 /* USB transfer specific */
89 static libusb20_tr_open_t ugen20_tr_open;
90 static libusb20_tr_close_t ugen20_tr_close;
91 static libusb20_tr_clear_stall_sync_t ugen20_tr_clear_stall_sync;
92 static libusb20_tr_submit_t ugen20_tr_submit;
93 static libusb20_tr_cancel_async_t ugen20_tr_cancel_async;
94 
95 static const struct libusb20_device_methods libusb20_ugen20_device_methods = {
96 	LIBUSB20_DEVICE(LIBUSB20_DECLARE, ugen20)
97 };
98 
99 static const char *
100 ugen20_get_backend_name(void)
101 {
102 	return ("FreeBSD UGEN 2.0");
103 }
104 
105 static uint32_t
106 ugen20_path_convert_one(const char **pp)
107 {
108 	const char *ptr;
109 	uint32_t temp = 0;
110 
111 	ptr = *pp;
112 
113 	while ((*ptr >= '0') && (*ptr <= '9')) {
114 		temp *= 10;
115 		temp += (*ptr - '0');
116 		if (temp >= 1000000) {
117 			/* catch overflow early */
118 			return (0xFFFFFFFF);
119 		}
120 		ptr++;
121 	}
122 
123 	if (*ptr == '.') {
124 		/* skip dot */
125 		ptr++;
126 	}
127 	*pp = ptr;
128 
129 	return (temp);
130 }
131 
132 static int
133 ugen20_enumerate(struct libusb20_device *pdev, const char *id)
134 {
135 	const char *tmp = id;
136 	struct usb_device_descriptor ddesc;
137 	struct usb_device_info devinfo;
138 	struct usb_device_port_path udpp;
139 	uint32_t plugtime;
140 	char buf[64];
141 	int f;
142 	int error;
143 
144 	pdev->bus_number = ugen20_path_convert_one(&tmp);
145 	pdev->device_address = ugen20_path_convert_one(&tmp);
146 
147 	snprintf(buf, sizeof(buf), "/dev/" USB_GENERIC_NAME "%u.%u",
148 	    pdev->bus_number, pdev->device_address);
149 
150 	f = open(buf, O_RDWR);
151 	if (f < 0) {
152 		return (LIBUSB20_ERROR_OTHER);
153 	}
154 	if (ioctl(f, IOUSB(USB_GET_PLUGTIME), &plugtime)) {
155 		error = LIBUSB20_ERROR_OTHER;
156 		goto done;
157 	}
158 	/* store when the device was plugged */
159 	pdev->session_data.plugtime = plugtime;
160 
161 	if (ioctl(f, IOUSB(USB_GET_DEVICE_DESC), &ddesc)) {
162 		error = LIBUSB20_ERROR_OTHER;
163 		goto done;
164 	}
165 	LIBUSB20_INIT(LIBUSB20_DEVICE_DESC, &(pdev->ddesc));
166 
167 	libusb20_me_decode(&ddesc, sizeof(ddesc), &(pdev->ddesc));
168 
169 	if (pdev->ddesc.bNumConfigurations == 0) {
170 		error = LIBUSB20_ERROR_OTHER;
171 		goto done;
172 	} else if (pdev->ddesc.bNumConfigurations >= 8) {
173 		error = LIBUSB20_ERROR_OTHER;
174 		goto done;
175 	}
176 	if (ioctl(f, IOUSB(USB_GET_DEVICEINFO), &devinfo)) {
177 		error = LIBUSB20_ERROR_OTHER;
178 		goto done;
179 	}
180 	switch (devinfo.udi_mode) {
181 	case USB_MODE_DEVICE:
182 		pdev->usb_mode = LIBUSB20_MODE_DEVICE;
183 		break;
184 	default:
185 		pdev->usb_mode = LIBUSB20_MODE_HOST;
186 		break;
187 	}
188 
189 	switch (devinfo.udi_speed) {
190 	case USB_SPEED_LOW:
191 		pdev->usb_speed = LIBUSB20_SPEED_LOW;
192 		break;
193 	case USB_SPEED_FULL:
194 		pdev->usb_speed = LIBUSB20_SPEED_FULL;
195 		break;
196 	case USB_SPEED_HIGH:
197 		pdev->usb_speed = LIBUSB20_SPEED_HIGH;
198 		break;
199 	case USB_SPEED_VARIABLE:
200 		pdev->usb_speed = LIBUSB20_SPEED_VARIABLE;
201 		break;
202 	case USB_SPEED_SUPER:
203 		pdev->usb_speed = LIBUSB20_SPEED_SUPER;
204 		break;
205 	default:
206 		pdev->usb_speed = LIBUSB20_SPEED_UNKNOWN;
207 		break;
208 	}
209 
210 	/* get parent HUB index and port */
211 
212 	pdev->parent_address = devinfo.udi_hubindex;
213 	pdev->parent_port = devinfo.udi_hubport;
214 
215 	/* generate a nice description for printout */
216 
217 	snprintf(pdev->usb_desc, sizeof(pdev->usb_desc),
218 	    USB_GENERIC_NAME "%u.%u: <%s %s> at usbus%u", pdev->bus_number,
219 	    pdev->device_address, devinfo.udi_vendor,
220 	    devinfo.udi_product, pdev->bus_number);
221 
222 	/* get device port path, if any */
223 	if (ioctl(f, IOUSB(USB_GET_DEV_PORT_PATH), &udpp) == 0 &&
224 	    udpp.udp_port_level < LIBUSB20_DEVICE_PORT_PATH_MAX) {
225 		memcpy(pdev->port_path, udpp.udp_port_no, udpp.udp_port_level);
226 		pdev->port_level = udpp.udp_port_level;
227 	}
228 
229 	error = 0;
230 done:
231 	close(f);
232 	return (error);
233 }
234 
235 struct ugen20_urd_state {
236 	struct usb_read_dir urd;
237 	uint32_t nparsed;
238 	int	f;
239 	uint8_t *ptr;
240 	const char *src;
241 	const char *dst;
242 	uint8_t	buf[256];
243 	uint8_t	dummy_zero[1];
244 };
245 
246 static int
247 ugen20_readdir(struct ugen20_urd_state *st)
248 {
249 	;				/* style fix */
250 repeat:
251 	if (st->ptr == NULL) {
252 		st->urd.urd_startentry += st->nparsed;
253 		st->urd.urd_data = libusb20_pass_ptr(st->buf);
254 		st->urd.urd_maxlen = sizeof(st->buf);
255 		st->nparsed = 0;
256 
257 		if (ioctl(st->f, IOUSB(USB_READ_DIR), &st->urd)) {
258 			return (EINVAL);
259 		}
260 		st->ptr = st->buf;
261 	}
262 	if (st->ptr[0] == 0) {
263 		if (st->nparsed) {
264 			st->ptr = NULL;
265 			goto repeat;
266 		} else {
267 			return (ENXIO);
268 		}
269 	}
270 	st->src = (void *)(st->ptr + 1);
271 	st->dst = st->src + strlen(st->src) + 1;
272 	st->ptr = st->ptr + st->ptr[0];
273 	st->nparsed++;
274 
275 	if ((st->ptr < st->buf) ||
276 	    (st->ptr > st->dummy_zero)) {
277 		/* invalid entry */
278 		return (EINVAL);
279 	}
280 	return (0);
281 }
282 
283 static int
284 ugen20_init_backend(struct libusb20_backend *pbe)
285 {
286 	struct ugen20_urd_state state;
287 	struct libusb20_device *pdev;
288 
289 	memset(&state, 0, sizeof(state));
290 
291 	state.f = open("/dev/" USB_DEVICE_NAME, O_RDONLY);
292 	if (state.f < 0)
293 		return (LIBUSB20_ERROR_OTHER);
294 
295 	while (ugen20_readdir(&state) == 0) {
296 
297 		if ((state.src[0] != 'u') ||
298 		    (state.src[1] != 'g') ||
299 		    (state.src[2] != 'e') ||
300 		    (state.src[3] != 'n')) {
301 			continue;
302 		}
303 		pdev = libusb20_dev_alloc();
304 		if (pdev == NULL) {
305 			continue;
306 		}
307 		if (ugen20_enumerate(pdev, state.src + 4)) {
308 			libusb20_dev_free(pdev);
309 			continue;
310 		}
311 		/* put the device on the backend list */
312 		libusb20_be_enqueue_device(pbe, pdev);
313 	}
314 	close(state.f);
315 	return (0);			/* success */
316 }
317 
318 static void
319 ugen20_tr_release(struct libusb20_device *pdev)
320 {
321 	struct usb_fs_uninit fs_uninit;
322 
323 	if (pdev->nTransfer == 0) {
324 		return;
325 	}
326 	/* release all pending USB transfers */
327 	if (pdev->privBeData != NULL) {
328 		memset(&fs_uninit, 0, sizeof(fs_uninit));
329 		if (ioctl(pdev->file, IOUSB(USB_FS_UNINIT), &fs_uninit)) {
330 			/* ignore any errors of this kind */
331 		}
332 	}
333 	return;
334 }
335 
336 static int
337 ugen20_tr_renew(struct libusb20_device *pdev)
338 {
339 	struct usb_fs_init fs_init;
340 	struct usb_fs_endpoint *pfse;
341 	int error;
342 	uint32_t size;
343 	uint16_t nMaxTransfer;
344 
345 	nMaxTransfer = pdev->nTransfer;
346 	error = 0;
347 
348 	if (nMaxTransfer == 0) {
349 		goto done;
350 	}
351 	size = nMaxTransfer * sizeof(*pfse);
352 
353 	if (pdev->privBeData == NULL) {
354 		pfse = malloc(size);
355 		if (pfse == NULL) {
356 			error = LIBUSB20_ERROR_NO_MEM;
357 			goto done;
358 		}
359 		pdev->privBeData = pfse;
360 	}
361 	/* reset endpoint data */
362 	memset(pdev->privBeData, 0, size);
363 
364 	memset(&fs_init, 0, sizeof(fs_init));
365 
366 	fs_init.pEndpoints = libusb20_pass_ptr(pdev->privBeData);
367 	fs_init.ep_index_max = nMaxTransfer;
368 
369 	if (ioctl(pdev->file, IOUSB(USB_FS_INIT), &fs_init)) {
370 		error = LIBUSB20_ERROR_OTHER;
371 		goto done;
372 	}
373 done:
374 	return (error);
375 }
376 
377 static int
378 ugen20_open_device(struct libusb20_device *pdev, uint16_t nMaxTransfer)
379 {
380 	uint32_t plugtime;
381 	char buf[64];
382 	int f;
383 	int g;
384 	int error;
385 
386 	snprintf(buf, sizeof(buf), "/dev/" USB_GENERIC_NAME "%u.%u",
387 	    pdev->bus_number, pdev->device_address);
388 
389 	/*
390 	 * We need two file handles, one for the control endpoint and one
391 	 * for BULK, INTERRUPT and ISOCHRONOUS transactions due to optimised
392 	 * kernel locking.
393 	 */
394 	g = open(buf, O_RDWR);
395 	if (g < 0) {
396 		return (LIBUSB20_ERROR_NO_DEVICE);
397 	}
398 	f = open(buf, O_RDWR);
399 	if (f < 0) {
400 		close(g);
401 		return (LIBUSB20_ERROR_NO_DEVICE);
402 	}
403 	if (ioctl(f, IOUSB(USB_GET_PLUGTIME), &plugtime)) {
404 		error = LIBUSB20_ERROR_OTHER;
405 		goto done;
406 	}
407 	/* check that the correct device is still plugged */
408 	if (pdev->session_data.plugtime != plugtime) {
409 		error = LIBUSB20_ERROR_NO_DEVICE;
410 		goto done;
411 	}
412 	/* need to set this before "tr_renew()" */
413 	pdev->file = f;
414 	pdev->file_ctrl = g;
415 
416 	/* renew all USB transfers */
417 	error = ugen20_tr_renew(pdev);
418 	if (error) {
419 		goto done;
420 	}
421 	/* set methods */
422 	pdev->methods = &libusb20_ugen20_device_methods;
423 
424 done:
425 	if (error) {
426 		if (pdev->privBeData) {
427 			/* cleanup after "tr_renew()" */
428 			free(pdev->privBeData);
429 			pdev->privBeData = NULL;
430 		}
431 		pdev->file = -1;
432 		pdev->file_ctrl = -1;
433 		close(f);
434 		close(g);
435 	}
436 	return (error);
437 }
438 
439 static int
440 ugen20_close_device(struct libusb20_device *pdev)
441 {
442 	struct usb_fs_uninit fs_uninit;
443 
444 	if (pdev->privBeData) {
445 		memset(&fs_uninit, 0, sizeof(fs_uninit));
446 		if (ioctl(pdev->file, IOUSB(USB_FS_UNINIT), &fs_uninit)) {
447 			/* ignore this error */
448 		}
449 		free(pdev->privBeData);
450 	}
451 	pdev->nTransfer = 0;
452 	pdev->privBeData = NULL;
453 	close(pdev->file);
454 	close(pdev->file_ctrl);
455 	pdev->file = -1;
456 	pdev->file_ctrl = -1;
457 	return (0);			/* success */
458 }
459 
460 static void
461 ugen20_exit_backend(struct libusb20_backend *pbe)
462 {
463 	return;				/* nothing to do */
464 }
465 
466 static int
467 ugen20_get_config_desc_full(struct libusb20_device *pdev,
468     uint8_t **ppbuf, uint16_t *plen, uint8_t cfg_index)
469 {
470 	struct usb_gen_descriptor gen_desc;
471 	struct usb_config_descriptor cdesc;
472 	uint8_t *ptr;
473 	uint16_t len;
474 	int error;
475 
476 	/* make sure memory is initialised */
477 	memset(&cdesc, 0, sizeof(cdesc));
478 	memset(&gen_desc, 0, sizeof(gen_desc));
479 
480 	gen_desc.ugd_data = libusb20_pass_ptr(&cdesc);
481 	gen_desc.ugd_maxlen = sizeof(cdesc);
482 	gen_desc.ugd_config_index = cfg_index;
483 
484 	error = ioctl(pdev->file_ctrl, IOUSB(USB_GET_FULL_DESC), &gen_desc);
485 	if (error) {
486 		return (LIBUSB20_ERROR_OTHER);
487 	}
488 	len = UGETW(cdesc.wTotalLength);
489 	if (len < sizeof(cdesc)) {
490 		/* corrupt descriptor */
491 		return (LIBUSB20_ERROR_OTHER);
492 	}
493 	ptr = malloc(len);
494 	if (!ptr) {
495 		return (LIBUSB20_ERROR_NO_MEM);
496 	}
497 
498 	/* make sure memory is initialised */
499 	memset(ptr, 0, len);
500 
501 	gen_desc.ugd_data = libusb20_pass_ptr(ptr);
502 	gen_desc.ugd_maxlen = len;
503 
504 	error = ioctl(pdev->file_ctrl, IOUSB(USB_GET_FULL_DESC), &gen_desc);
505 	if (error) {
506 		free(ptr);
507 		return (LIBUSB20_ERROR_OTHER);
508 	}
509 	/* make sure that the device doesn't fool us */
510 	memcpy(ptr, &cdesc, sizeof(cdesc));
511 
512 	*ppbuf = ptr;
513 	*plen = len;
514 
515 	return (0);			/* success */
516 }
517 
518 static int
519 ugen20_get_config_index(struct libusb20_device *pdev, uint8_t *pindex)
520 {
521 	int temp;
522 
523 	if (ioctl(pdev->file_ctrl, IOUSB(USB_GET_CONFIG), &temp)) {
524 		return (LIBUSB20_ERROR_OTHER);
525 	}
526 	*pindex = temp;
527 
528 	return (0);
529 }
530 
531 static int
532 ugen20_set_config_index(struct libusb20_device *pdev, uint8_t cfg_index)
533 {
534 	int temp = cfg_index;
535 
536 	/* release all active USB transfers */
537 	ugen20_tr_release(pdev);
538 
539 	if (ioctl(pdev->file_ctrl, IOUSB(USB_SET_CONFIG), &temp)) {
540 		return (LIBUSB20_ERROR_OTHER);
541 	}
542 	return (ugen20_tr_renew(pdev));
543 }
544 
545 static int
546 ugen20_set_alt_index(struct libusb20_device *pdev,
547     uint8_t iface_index, uint8_t alt_index)
548 {
549 	struct usb_alt_interface alt_iface;
550 
551 	memset(&alt_iface, 0, sizeof(alt_iface));
552 
553 	alt_iface.uai_interface_index = iface_index;
554 	alt_iface.uai_alt_index = alt_index;
555 
556 	/* release all active USB transfers */
557 	ugen20_tr_release(pdev);
558 
559 	if (ioctl(pdev->file_ctrl, IOUSB(USB_SET_ALTINTERFACE), &alt_iface)) {
560 		return (LIBUSB20_ERROR_OTHER);
561 	}
562 	return (ugen20_tr_renew(pdev));
563 }
564 
565 static int
566 ugen20_reset_device(struct libusb20_device *pdev)
567 {
568 	int temp = 0;
569 
570 	/* release all active USB transfers */
571 	ugen20_tr_release(pdev);
572 
573 	if (ioctl(pdev->file_ctrl, IOUSB(USB_DEVICEENUMERATE), &temp)) {
574 		return (LIBUSB20_ERROR_OTHER);
575 	}
576 	return (ugen20_tr_renew(pdev));
577 }
578 
579 static int
580 ugen20_check_connected(struct libusb20_device *pdev)
581 {
582 	uint32_t plugtime;
583 	int error = 0;
584 
585 	if (ioctl(pdev->file_ctrl, IOUSB(USB_GET_PLUGTIME), &plugtime)) {
586 		error = LIBUSB20_ERROR_NO_DEVICE;
587 		goto done;
588 	}
589 
590 	if (pdev->session_data.plugtime != plugtime) {
591 		error = LIBUSB20_ERROR_NO_DEVICE;
592 		goto done;
593 	}
594 done:
595 	return (error);
596 }
597 
598 static int
599 ugen20_set_power_mode(struct libusb20_device *pdev, uint8_t power_mode)
600 {
601 	int temp;
602 
603 	switch (power_mode) {
604 	case LIBUSB20_POWER_OFF:
605 		temp = USB_POWER_MODE_OFF;
606 		break;
607 	case LIBUSB20_POWER_ON:
608 		temp = USB_POWER_MODE_ON;
609 		break;
610 	case LIBUSB20_POWER_SAVE:
611 		temp = USB_POWER_MODE_SAVE;
612 		break;
613 	case LIBUSB20_POWER_SUSPEND:
614 		temp = USB_POWER_MODE_SUSPEND;
615 		break;
616 	case LIBUSB20_POWER_RESUME:
617 		temp = USB_POWER_MODE_RESUME;
618 		break;
619 	default:
620 		return (LIBUSB20_ERROR_INVALID_PARAM);
621 	}
622 	if (ioctl(pdev->file_ctrl, IOUSB(USB_SET_POWER_MODE), &temp)) {
623 		return (LIBUSB20_ERROR_OTHER);
624 	}
625 	return (0);
626 }
627 
628 static int
629 ugen20_get_power_mode(struct libusb20_device *pdev, uint8_t *power_mode)
630 {
631 	int temp;
632 
633 	if (ioctl(pdev->file_ctrl, IOUSB(USB_GET_POWER_MODE), &temp)) {
634 		return (LIBUSB20_ERROR_OTHER);
635 	}
636 	switch (temp) {
637 	case USB_POWER_MODE_OFF:
638 		temp = LIBUSB20_POWER_OFF;
639 		break;
640 	case USB_POWER_MODE_ON:
641 		temp = LIBUSB20_POWER_ON;
642 		break;
643 	case USB_POWER_MODE_SAVE:
644 		temp = LIBUSB20_POWER_SAVE;
645 		break;
646 	case USB_POWER_MODE_SUSPEND:
647 		temp = LIBUSB20_POWER_SUSPEND;
648 		break;
649 	case USB_POWER_MODE_RESUME:
650 		temp = LIBUSB20_POWER_RESUME;
651 		break;
652 	default:
653 		temp = LIBUSB20_POWER_ON;
654 		break;
655 	}
656 	*power_mode = temp;
657 	return (0);			/* success */
658 }
659 
660 static int
661 ugen20_get_power_usage(struct libusb20_device *pdev, uint16_t *power_usage)
662 {
663 	int temp;
664 
665 	if (ioctl(pdev->file_ctrl, IOUSB(USB_GET_POWER_USAGE), &temp)) {
666 		return (LIBUSB20_ERROR_OTHER);
667 	}
668 	*power_usage = temp;
669 	return (0);			/* success */
670 }
671 
672 static int
673 ugen20_kernel_driver_active(struct libusb20_device *pdev,
674     uint8_t iface_index)
675 {
676 	int temp = iface_index;
677 
678 	if (ioctl(pdev->file_ctrl, IOUSB(USB_IFACE_DRIVER_ACTIVE), &temp)) {
679 		return (LIBUSB20_ERROR_OTHER);
680 	}
681 	return (0);			/* kernel driver is active */
682 }
683 
684 static int
685 ugen20_detach_kernel_driver(struct libusb20_device *pdev,
686     uint8_t iface_index)
687 {
688 	int temp = iface_index;
689 
690 	if (ioctl(pdev->file_ctrl, IOUSB(USB_IFACE_DRIVER_DETACH), &temp)) {
691 		return (LIBUSB20_ERROR_OTHER);
692 	}
693 	return (0);			/* kernel driver is detached */
694 }
695 
696 static int
697 ugen20_do_request_sync(struct libusb20_device *pdev,
698     struct LIBUSB20_CONTROL_SETUP_DECODED *setup,
699     void *data, uint16_t *pactlen, uint32_t timeout, uint8_t flags)
700 {
701 	struct usb_ctl_request req;
702 
703 	memset(&req, 0, sizeof(req));
704 
705 	req.ucr_data = libusb20_pass_ptr(data);
706 	if (!(flags & LIBUSB20_TRANSFER_SINGLE_SHORT_NOT_OK)) {
707 		req.ucr_flags |= USB_SHORT_XFER_OK;
708 	}
709 	if (libusb20_me_encode(&req.ucr_request,
710 	    sizeof(req.ucr_request), setup)) {
711 		/* ignore */
712 	}
713 	if (ioctl(pdev->file_ctrl, IOUSB(USB_DO_REQUEST), &req)) {
714 		return (LIBUSB20_ERROR_OTHER);
715 	}
716 	if (pactlen) {
717 		/* get actual length */
718 		*pactlen = req.ucr_actlen;
719 	}
720 	return (0);			/* request was successful */
721 }
722 
723 static int
724 ugen20_process(struct libusb20_device *pdev)
725 {
726 	struct usb_fs_complete temp;
727 	struct usb_fs_endpoint *fsep;
728 	struct libusb20_transfer *xfer;
729 
730 	while (1) {
731 
732 	  if (ioctl(pdev->file, IOUSB(USB_FS_COMPLETE), &temp)) {
733 			if (errno == EBUSY) {
734 				break;
735 			} else {
736 				/* device detached */
737 				return (LIBUSB20_ERROR_OTHER);
738 			}
739 		}
740 		fsep = pdev->privBeData;
741 		xfer = pdev->pTransfer;
742 		fsep += temp.ep_index;
743 		xfer += temp.ep_index;
744 
745 		/* update transfer status */
746 
747 		if (fsep->status == 0) {
748 			xfer->aFrames = fsep->aFrames;
749 			xfer->timeComplete = fsep->isoc_time_complete;
750 			xfer->status = LIBUSB20_TRANSFER_COMPLETED;
751 		} else if (fsep->status == USB_ERR_CANCELLED) {
752 			xfer->aFrames = 0;
753 			xfer->timeComplete = 0;
754 			xfer->status = LIBUSB20_TRANSFER_CANCELLED;
755 		} else if (fsep->status == USB_ERR_STALLED) {
756 			xfer->aFrames = 0;
757 			xfer->timeComplete = 0;
758 			xfer->status = LIBUSB20_TRANSFER_STALL;
759 		} else if (fsep->status == USB_ERR_TIMEOUT) {
760 			xfer->aFrames = 0;
761 			xfer->timeComplete = 0;
762 			xfer->status = LIBUSB20_TRANSFER_TIMED_OUT;
763 		} else {
764 			xfer->aFrames = 0;
765 			xfer->timeComplete = 0;
766 			xfer->status = LIBUSB20_TRANSFER_ERROR;
767 		}
768 		libusb20_tr_callback_wrapper(xfer);
769 	}
770 	return (0);			/* done */
771 }
772 
773 static int
774 ugen20_tr_open(struct libusb20_transfer *xfer, uint32_t MaxBufSize,
775     uint32_t MaxFrameCount, uint8_t ep_no, uint16_t stream_id,
776     uint8_t pre_scale)
777 {
778 	union {
779 		struct usb_fs_open fs_open;
780 		struct usb_fs_open_stream fs_open_stream;
781 	} temp;
782 	struct usb_fs_endpoint *fsep;
783 
784 	if (pre_scale)
785 		MaxFrameCount |= USB_FS_MAX_FRAMES_PRE_SCALE;
786 
787 	memset(&temp, 0, sizeof(temp));
788 
789 	fsep = xfer->pdev->privBeData;
790 	fsep += xfer->trIndex;
791 
792 	temp.fs_open.max_bufsize = MaxBufSize;
793 	temp.fs_open.max_frames = MaxFrameCount;
794 	temp.fs_open.ep_index = xfer->trIndex;
795 	temp.fs_open.ep_no = ep_no;
796 
797 	if (stream_id != 0) {
798 		temp.fs_open_stream.stream_id = stream_id;
799 
800 		if (ioctl(xfer->pdev->file, IOUSB(USB_FS_OPEN_STREAM), &temp.fs_open_stream))
801 			return (LIBUSB20_ERROR_INVALID_PARAM);
802 	} else {
803 		if (ioctl(xfer->pdev->file, IOUSB(USB_FS_OPEN), &temp.fs_open))
804 			return (LIBUSB20_ERROR_INVALID_PARAM);
805 	}
806 	/* maximums might have changed - update */
807 	xfer->maxFrames = temp.fs_open.max_frames;
808 
809 	/* "max_bufsize" should be multiple of "max_packet_length" */
810 	xfer->maxTotalLength = temp.fs_open.max_bufsize;
811 	xfer->maxPacketLen = temp.fs_open.max_packet_length;
812 
813 	/* setup buffer and length lists using zero copy */
814 	fsep->ppBuffer = libusb20_pass_ptr(xfer->ppBuffer);
815 	fsep->pLength = libusb20_pass_ptr(xfer->pLength);
816 
817 	return (0);			/* success */
818 }
819 
820 static int
821 ugen20_tr_close(struct libusb20_transfer *xfer)
822 {
823 	struct usb_fs_close temp;
824 
825 	memset(&temp, 0, sizeof(temp));
826 
827 	temp.ep_index = xfer->trIndex;
828 
829 	if (ioctl(xfer->pdev->file, IOUSB(USB_FS_CLOSE), &temp)) {
830 		return (LIBUSB20_ERROR_INVALID_PARAM);
831 	}
832 	return (0);			/* success */
833 }
834 
835 static int
836 ugen20_tr_clear_stall_sync(struct libusb20_transfer *xfer)
837 {
838 	struct usb_fs_clear_stall_sync temp;
839 
840 	memset(&temp, 0, sizeof(temp));
841 
842 	/* if the transfer is active, an error will be returned */
843 
844 	temp.ep_index = xfer->trIndex;
845 
846 	if (ioctl(xfer->pdev->file, IOUSB(USB_FS_CLEAR_STALL_SYNC), &temp)) {
847 		return (LIBUSB20_ERROR_INVALID_PARAM);
848 	}
849 	return (0);			/* success */
850 }
851 
852 static void
853 ugen20_tr_submit(struct libusb20_transfer *xfer)
854 {
855 	struct usb_fs_start temp;
856 	struct usb_fs_endpoint *fsep;
857 
858 	memset(&temp, 0, sizeof(temp));
859 
860 	fsep = xfer->pdev->privBeData;
861 	fsep += xfer->trIndex;
862 
863 	fsep->nFrames = xfer->nFrames;
864 	fsep->flags = 0;
865 	if (!(xfer->flags & LIBUSB20_TRANSFER_SINGLE_SHORT_NOT_OK)) {
866 		fsep->flags |= USB_FS_FLAG_SINGLE_SHORT_OK;
867 	}
868 	if (!(xfer->flags & LIBUSB20_TRANSFER_MULTI_SHORT_NOT_OK)) {
869 		fsep->flags |= USB_FS_FLAG_MULTI_SHORT_OK;
870 	}
871 	if (xfer->flags & LIBUSB20_TRANSFER_FORCE_SHORT) {
872 		fsep->flags |= USB_FS_FLAG_FORCE_SHORT;
873 	}
874 	if (xfer->flags & LIBUSB20_TRANSFER_DO_CLEAR_STALL) {
875 		fsep->flags |= USB_FS_FLAG_CLEAR_STALL;
876 	}
877 	/* NOTE: The "fsep->timeout" variable is 16-bit. */
878 	if (xfer->timeout > 65535)
879 		fsep->timeout = 65535;
880 	else
881 		fsep->timeout = xfer->timeout;
882 
883 	temp.ep_index = xfer->trIndex;
884 
885 	if (ioctl(xfer->pdev->file, IOUSB(USB_FS_START), &temp)) {
886 		/* ignore any errors - should never happen */
887 	}
888 	return;				/* success */
889 }
890 
891 static void
892 ugen20_tr_cancel_async(struct libusb20_transfer *xfer)
893 {
894 	struct usb_fs_stop temp;
895 
896 	memset(&temp, 0, sizeof(temp));
897 
898 	temp.ep_index = xfer->trIndex;
899 
900 	if (ioctl(xfer->pdev->file, IOUSB(USB_FS_STOP), &temp)) {
901 		/* ignore any errors - should never happen */
902 	}
903 	return;
904 }
905 
906 static int
907 ugen20_be_ioctl(uint32_t cmd, void *data)
908 {
909 	int f;
910 	int error;
911 
912 	f = open("/dev/" USB_DEVICE_NAME, O_RDONLY);
913 	if (f < 0)
914 		return (LIBUSB20_ERROR_OTHER);
915 	error = ioctl(f, cmd, data);
916 	if (error == -1) {
917 		if (errno == EPERM) {
918 			error = LIBUSB20_ERROR_ACCESS;
919 		} else {
920 			error = LIBUSB20_ERROR_OTHER;
921 		}
922 	}
923 	close(f);
924 	return (error);
925 }
926 
927 static int
928 ugen20_dev_get_iface_desc(struct libusb20_device *pdev,
929     uint8_t iface_index, char *buf, uint8_t len)
930 {
931 	struct usb_gen_descriptor ugd;
932 
933 	memset(&ugd, 0, sizeof(ugd));
934 
935 	ugd.ugd_data = libusb20_pass_ptr(buf);
936 	ugd.ugd_maxlen = len;
937 	ugd.ugd_iface_index = iface_index;
938 
939 	if (ioctl(pdev->file, IOUSB(USB_GET_IFACE_DRIVER), &ugd)) {
940 		return (LIBUSB20_ERROR_INVALID_PARAM);
941 	}
942 	return (0);
943 }
944 
945 static int
946 ugen20_dev_get_info(struct libusb20_device *pdev,
947     struct usb_device_info *pinfo)
948 {
949 	if (ioctl(pdev->file, IOUSB(USB_GET_DEVICEINFO), pinfo)) {
950 		return (LIBUSB20_ERROR_INVALID_PARAM);
951 	}
952 	return (0);
953 }
954 
955 static int
956 ugen20_root_get_dev_quirk(struct libusb20_backend *pbe,
957     uint16_t quirk_index, struct libusb20_quirk *pq)
958 {
959 	struct usb_gen_quirk q;
960 	int error;
961 
962 	memset(&q, 0, sizeof(q));
963 
964 	q.index = quirk_index;
965 
966 	error = ugen20_be_ioctl(IOUSB(USB_DEV_QUIRK_GET), &q);
967 
968 	if (error) {
969 		if (errno == EINVAL) {
970 			return (LIBUSB20_ERROR_NOT_FOUND);
971 		}
972 	} else {
973 		pq->vid = q.vid;
974 		pq->pid = q.pid;
975 		pq->bcdDeviceLow = q.bcdDeviceLow;
976 		pq->bcdDeviceHigh = q.bcdDeviceHigh;
977 		strlcpy(pq->quirkname, q.quirkname, sizeof(pq->quirkname));
978 	}
979 	return (error);
980 }
981 
982 static int
983 ugen20_root_get_quirk_name(struct libusb20_backend *pbe, uint16_t quirk_index,
984     struct libusb20_quirk *pq)
985 {
986 	struct usb_gen_quirk q;
987 	int error;
988 
989 	memset(&q, 0, sizeof(q));
990 
991 	q.index = quirk_index;
992 
993 	error = ugen20_be_ioctl(IOUSB(USB_QUIRK_NAME_GET), &q);
994 
995 	if (error) {
996 		if (errno == EINVAL) {
997 			return (LIBUSB20_ERROR_NOT_FOUND);
998 		}
999 	} else {
1000 		strlcpy(pq->quirkname, q.quirkname, sizeof(pq->quirkname));
1001 	}
1002 	return (error);
1003 }
1004 
1005 static int
1006 ugen20_root_add_dev_quirk(struct libusb20_backend *pbe,
1007     struct libusb20_quirk *pq)
1008 {
1009 	struct usb_gen_quirk q;
1010 	int error;
1011 
1012 	memset(&q, 0, sizeof(q));
1013 
1014 	q.vid = pq->vid;
1015 	q.pid = pq->pid;
1016 	q.bcdDeviceLow = pq->bcdDeviceLow;
1017 	q.bcdDeviceHigh = pq->bcdDeviceHigh;
1018 	strlcpy(q.quirkname, pq->quirkname, sizeof(q.quirkname));
1019 
1020 	error = ugen20_be_ioctl(IOUSB(USB_DEV_QUIRK_ADD), &q);
1021 	if (error) {
1022 		if (errno == ENOMEM) {
1023 			return (LIBUSB20_ERROR_NO_MEM);
1024 		}
1025 	}
1026 	return (error);
1027 }
1028 
1029 static int
1030 ugen20_root_remove_dev_quirk(struct libusb20_backend *pbe,
1031     struct libusb20_quirk *pq)
1032 {
1033 	struct usb_gen_quirk q;
1034 	int error;
1035 
1036 	memset(&q, 0, sizeof(q));
1037 
1038 	q.vid = pq->vid;
1039 	q.pid = pq->pid;
1040 	q.bcdDeviceLow = pq->bcdDeviceLow;
1041 	q.bcdDeviceHigh = pq->bcdDeviceHigh;
1042 	strlcpy(q.quirkname, pq->quirkname, sizeof(q.quirkname));
1043 
1044 	error = ugen20_be_ioctl(IOUSB(USB_DEV_QUIRK_REMOVE), &q);
1045 	if (error) {
1046 		if (errno == EINVAL) {
1047 			return (LIBUSB20_ERROR_NOT_FOUND);
1048 		}
1049 	}
1050 	return (error);
1051 }
1052 
1053 static int
1054 ugen20_root_set_template(struct libusb20_backend *pbe, int temp)
1055 {
1056 	return (ugen20_be_ioctl(IOUSB(USB_SET_TEMPLATE), &temp));
1057 }
1058 
1059 static int
1060 ugen20_root_get_template(struct libusb20_backend *pbe, int *ptemp)
1061 {
1062 	return (ugen20_be_ioctl(IOUSB(USB_GET_TEMPLATE), ptemp));
1063 }
1064