xref: /freebsd/lib/libusb/libusb20.c (revision eb6d21b4ca6d668cf89afd99eef7baeafa712197)
1 /* $FreeBSD$ */
2 /*-
3  * Copyright (c) 2008-2009 Hans Petter Selasky. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  */
26 
27 #include <stdio.h>
28 #include <stdlib.h>
29 #include <string.h>
30 #include <poll.h>
31 #include <ctype.h>
32 #include <sys/queue.h>
33 
34 #include "libusb20.h"
35 #include "libusb20_desc.h"
36 #include "libusb20_int.h"
37 
38 static int
39 dummy_int(void)
40 {
41 	return (LIBUSB20_ERROR_NOT_SUPPORTED);
42 }
43 
44 static void
45 dummy_void(void)
46 {
47 	return;
48 }
49 
50 static void
51 dummy_callback(struct libusb20_transfer *xfer)
52 {
53 	;				/* style fix */
54 	switch (libusb20_tr_get_status(xfer)) {
55 	case LIBUSB20_TRANSFER_START:
56 		libusb20_tr_submit(xfer);
57 		break;
58 	default:
59 		/* complete or error */
60 		break;
61 	}
62 	return;
63 }
64 
65 #define	dummy_get_config_desc_full (void *)dummy_int
66 #define	dummy_get_config_index (void *)dummy_int
67 #define	dummy_set_config_index (void *)dummy_int
68 #define	dummy_set_alt_index (void *)dummy_int
69 #define	dummy_reset_device (void *)dummy_int
70 #define	dummy_set_power_mode (void *)dummy_int
71 #define	dummy_get_power_mode (void *)dummy_int
72 #define	dummy_kernel_driver_active (void *)dummy_int
73 #define	dummy_detach_kernel_driver (void *)dummy_int
74 #define	dummy_do_request_sync (void *)dummy_int
75 #define	dummy_tr_open (void *)dummy_int
76 #define	dummy_tr_close (void *)dummy_int
77 #define	dummy_tr_clear_stall_sync (void *)dummy_int
78 #define	dummy_process (void *)dummy_int
79 #define	dummy_dev_info (void *)dummy_int
80 #define	dummy_dev_get_iface_driver (void *)dummy_int
81 
82 #define	dummy_tr_submit (void *)dummy_void
83 #define	dummy_tr_cancel_async (void *)dummy_void
84 
85 static const struct libusb20_device_methods libusb20_dummy_methods = {
86 	LIBUSB20_DEVICE(LIBUSB20_DECLARE, dummy)
87 };
88 
89 void
90 libusb20_tr_callback_wrapper(struct libusb20_transfer *xfer)
91 {
92 	;				/* style fix */
93 
94 repeat:
95 
96 	if (!xfer->is_pending) {
97 		xfer->status = LIBUSB20_TRANSFER_START;
98 	} else {
99 		xfer->is_pending = 0;
100 	}
101 
102 	xfer->callback(xfer);
103 
104 	if (xfer->is_restart) {
105 		xfer->is_restart = 0;
106 		goto repeat;
107 	}
108 	if (xfer->is_draining &&
109 	    (!xfer->is_pending)) {
110 		xfer->is_draining = 0;
111 		xfer->status = LIBUSB20_TRANSFER_DRAINED;
112 		xfer->callback(xfer);
113 	}
114 	return;
115 }
116 
117 int
118 libusb20_tr_close(struct libusb20_transfer *xfer)
119 {
120 	int error;
121 
122 	if (!xfer->is_opened) {
123 		return (LIBUSB20_ERROR_OTHER);
124 	}
125 	error = xfer->pdev->methods->tr_close(xfer);
126 
127 	if (xfer->pLength) {
128 		free(xfer->pLength);
129 	}
130 	if (xfer->ppBuffer) {
131 		free(xfer->ppBuffer);
132 	}
133 	/* clear some fields */
134 	xfer->is_opened = 0;
135 	xfer->maxFrames = 0;
136 	xfer->maxTotalLength = 0;
137 	xfer->maxPacketLen = 0;
138 	return (error);
139 }
140 
141 int
142 libusb20_tr_open(struct libusb20_transfer *xfer, uint32_t MaxBufSize,
143     uint32_t MaxFrameCount, uint8_t ep_no)
144 {
145 	uint32_t size;
146 	int error;
147 
148 	if (xfer->is_opened) {
149 		return (LIBUSB20_ERROR_BUSY);
150 	}
151 	if (MaxFrameCount == 0) {
152 		return (LIBUSB20_ERROR_INVALID_PARAM);
153 	}
154 	xfer->maxFrames = MaxFrameCount;
155 
156 	size = MaxFrameCount * sizeof(xfer->pLength[0]);
157 	xfer->pLength = malloc(size);
158 	if (xfer->pLength == NULL) {
159 		return (LIBUSB20_ERROR_NO_MEM);
160 	}
161 	memset(xfer->pLength, 0, size);
162 
163 	size = MaxFrameCount * sizeof(xfer->ppBuffer[0]);
164 	xfer->ppBuffer = malloc(size);
165 	if (xfer->ppBuffer == NULL) {
166 		free(xfer->pLength);
167 		return (LIBUSB20_ERROR_NO_MEM);
168 	}
169 	memset(xfer->ppBuffer, 0, size);
170 
171 	error = xfer->pdev->methods->tr_open(xfer, MaxBufSize,
172 	    MaxFrameCount, ep_no);
173 
174 	if (error) {
175 		free(xfer->ppBuffer);
176 		free(xfer->pLength);
177 	} else {
178 		xfer->is_opened = 1;
179 	}
180 	return (error);
181 }
182 
183 struct libusb20_transfer *
184 libusb20_tr_get_pointer(struct libusb20_device *pdev, uint16_t trIndex)
185 {
186 	if (trIndex >= pdev->nTransfer) {
187 		return (NULL);
188 	}
189 	return (pdev->pTransfer + trIndex);
190 }
191 
192 uint32_t
193 libusb20_tr_get_actual_frames(struct libusb20_transfer *xfer)
194 {
195 	return (xfer->aFrames);
196 }
197 
198 uint16_t
199 libusb20_tr_get_time_complete(struct libusb20_transfer *xfer)
200 {
201 	return (xfer->timeComplete);
202 }
203 
204 uint32_t
205 libusb20_tr_get_actual_length(struct libusb20_transfer *xfer)
206 {
207 	uint32_t x;
208 	uint32_t actlen = 0;
209 
210 	for (x = 0; x != xfer->aFrames; x++) {
211 		actlen += xfer->pLength[x];
212 	}
213 	return (actlen);
214 }
215 
216 uint32_t
217 libusb20_tr_get_max_frames(struct libusb20_transfer *xfer)
218 {
219 	return (xfer->maxFrames);
220 }
221 
222 uint32_t
223 libusb20_tr_get_max_packet_length(struct libusb20_transfer *xfer)
224 {
225 	/*
226 	 * Special Case NOTE: If the packet multiplier is non-zero for
227 	 * High Speed USB, the value returned is equal to
228 	 * "wMaxPacketSize * multiplier" !
229 	 */
230 	return (xfer->maxPacketLen);
231 }
232 
233 uint32_t
234 libusb20_tr_get_max_total_length(struct libusb20_transfer *xfer)
235 {
236 	return (xfer->maxTotalLength);
237 }
238 
239 uint8_t
240 libusb20_tr_get_status(struct libusb20_transfer *xfer)
241 {
242 	return (xfer->status);
243 }
244 
245 uint8_t
246 libusb20_tr_pending(struct libusb20_transfer *xfer)
247 {
248 	return (xfer->is_pending);
249 }
250 
251 void   *
252 libusb20_tr_get_priv_sc0(struct libusb20_transfer *xfer)
253 {
254 	return (xfer->priv_sc0);
255 }
256 
257 void   *
258 libusb20_tr_get_priv_sc1(struct libusb20_transfer *xfer)
259 {
260 	return (xfer->priv_sc1);
261 }
262 
263 void
264 libusb20_tr_stop(struct libusb20_transfer *xfer)
265 {
266 	if (!xfer->is_opened) {
267 		/* transfer is not opened */
268 		return;
269 	}
270 	if (!xfer->is_pending) {
271 		/* transfer not pending */
272 		return;
273 	}
274 	if (xfer->is_cancel) {
275 		/* already cancelling */
276 		return;
277 	}
278 	xfer->is_cancel = 1;		/* we are cancelling */
279 
280 	xfer->pdev->methods->tr_cancel_async(xfer);
281 	return;
282 }
283 
284 void
285 libusb20_tr_drain(struct libusb20_transfer *xfer)
286 {
287 	if (!xfer->is_opened) {
288 		/* transfer is not opened */
289 		return;
290 	}
291 	/* make sure that we are cancelling */
292 	libusb20_tr_stop(xfer);
293 
294 	if (xfer->is_pending) {
295 		xfer->is_draining = 1;
296 	}
297 	return;
298 }
299 
300 void
301 libusb20_tr_clear_stall_sync(struct libusb20_transfer *xfer)
302 {
303 	xfer->pdev->methods->tr_clear_stall_sync(xfer);
304 	return;
305 }
306 
307 void
308 libusb20_tr_set_buffer(struct libusb20_transfer *xfer, void *buffer, uint16_t frIndex)
309 {
310 	xfer->ppBuffer[frIndex] = buffer;
311 	return;
312 }
313 
314 void
315 libusb20_tr_set_callback(struct libusb20_transfer *xfer, libusb20_tr_callback_t *cb)
316 {
317 	xfer->callback = cb;
318 	return;
319 }
320 
321 void
322 libusb20_tr_set_flags(struct libusb20_transfer *xfer, uint8_t flags)
323 {
324 	xfer->flags = flags;
325 	return;
326 }
327 
328 uint32_t
329 libusb20_tr_get_length(struct libusb20_transfer *xfer, uint16_t frIndex)
330 {
331 	return (xfer->pLength[frIndex]);
332 }
333 
334 void
335 libusb20_tr_set_length(struct libusb20_transfer *xfer, uint32_t length, uint16_t frIndex)
336 {
337 	xfer->pLength[frIndex] = length;
338 	return;
339 }
340 
341 void
342 libusb20_tr_set_priv_sc0(struct libusb20_transfer *xfer, void *sc0)
343 {
344 	xfer->priv_sc0 = sc0;
345 	return;
346 }
347 
348 void
349 libusb20_tr_set_priv_sc1(struct libusb20_transfer *xfer, void *sc1)
350 {
351 	xfer->priv_sc1 = sc1;
352 	return;
353 }
354 
355 void
356 libusb20_tr_set_timeout(struct libusb20_transfer *xfer, uint32_t timeout)
357 {
358 	xfer->timeout = timeout;
359 	return;
360 }
361 
362 void
363 libusb20_tr_set_total_frames(struct libusb20_transfer *xfer, uint32_t nFrames)
364 {
365 	if (nFrames > xfer->maxFrames) {
366 		/* should not happen */
367 		nFrames = xfer->maxFrames;
368 	}
369 	xfer->nFrames = nFrames;
370 	return;
371 }
372 
373 void
374 libusb20_tr_setup_bulk(struct libusb20_transfer *xfer, void *pBuf, uint32_t length, uint32_t timeout)
375 {
376 	xfer->ppBuffer[0] = pBuf;
377 	xfer->pLength[0] = length;
378 	xfer->timeout = timeout;
379 	xfer->nFrames = 1;
380 	return;
381 }
382 
383 void
384 libusb20_tr_setup_control(struct libusb20_transfer *xfer, void *psetup, void *pBuf, uint32_t timeout)
385 {
386 	uint16_t len;
387 
388 	xfer->ppBuffer[0] = psetup;
389 	xfer->pLength[0] = 8;		/* fixed */
390 	xfer->timeout = timeout;
391 
392 	len = ((uint8_t *)psetup)[6] | (((uint8_t *)psetup)[7] << 8);
393 
394 	if (len != 0) {
395 		xfer->nFrames = 2;
396 		xfer->ppBuffer[1] = pBuf;
397 		xfer->pLength[1] = len;
398 	} else {
399 		xfer->nFrames = 1;
400 	}
401 	return;
402 }
403 
404 void
405 libusb20_tr_setup_intr(struct libusb20_transfer *xfer, void *pBuf, uint32_t length, uint32_t timeout)
406 {
407 	xfer->ppBuffer[0] = pBuf;
408 	xfer->pLength[0] = length;
409 	xfer->timeout = timeout;
410 	xfer->nFrames = 1;
411 	return;
412 }
413 
414 void
415 libusb20_tr_setup_isoc(struct libusb20_transfer *xfer, void *pBuf, uint32_t length, uint16_t frIndex)
416 {
417 	if (frIndex >= xfer->maxFrames) {
418 		/* should not happen */
419 		return;
420 	}
421 	xfer->ppBuffer[frIndex] = pBuf;
422 	xfer->pLength[frIndex] = length;
423 	return;
424 }
425 
426 uint8_t
427 libusb20_tr_bulk_intr_sync(struct libusb20_transfer *xfer,
428     void *pbuf, uint32_t length, uint32_t *pactlen,
429     uint32_t timeout)
430 {
431 	struct libusb20_device *pdev = xfer->pdev;
432 	uint32_t transfer_max;
433 	uint32_t transfer_act;
434 	uint8_t retval;
435 
436 	/* set some sensible default value */
437 	if (pactlen != NULL)
438 		*pactlen = 0;
439 
440 	/* check for error condition */
441 	if (libusb20_tr_pending(xfer))
442 		return (LIBUSB20_ERROR_OTHER);
443 
444 	do {
445 		/* compute maximum transfer length */
446 		transfer_max =
447 		    libusb20_tr_get_max_total_length(xfer);
448 
449 		if (transfer_max > length)
450 			transfer_max = length;
451 
452 		/* setup bulk or interrupt transfer */
453 		libusb20_tr_setup_bulk(xfer, pbuf,
454 		    transfer_max, timeout);
455 
456 		/* start the transfer */
457 		libusb20_tr_start(xfer);
458 
459 		/* wait for transfer completion */
460 		while (libusb20_dev_process(pdev) == 0) {
461 
462 			if (libusb20_tr_pending(xfer) == 0)
463 				break;
464 
465 			libusb20_dev_wait_process(pdev, -1);
466 		}
467 
468 		transfer_act = libusb20_tr_get_actual_length(xfer);
469 
470 		/* update actual length, if any */
471 		if (pactlen != NULL)
472 			pactlen[0] += transfer_act;
473 
474 		/* check transfer status */
475 		retval = libusb20_tr_get_status(xfer);
476 		if (retval)
477 			break;
478 
479 		/* check for short transfer */
480 		if (transfer_act != transfer_max)
481 			break;
482 
483 		/* update buffer pointer and length */
484 		pbuf = ((uint8_t *)pbuf) + transfer_max;
485 		length = length - transfer_max;
486 
487 	} while (length != 0);
488 
489 	return (retval);
490 }
491 
492 void
493 libusb20_tr_submit(struct libusb20_transfer *xfer)
494 {
495 	if (!xfer->is_opened) {
496 		/* transfer is not opened */
497 		return;
498 	}
499 	if (xfer->is_pending) {
500 		/* should not happen */
501 		return;
502 	}
503 	xfer->is_pending = 1;		/* we are pending */
504 	xfer->is_cancel = 0;		/* not cancelling */
505 	xfer->is_restart = 0;		/* not restarting */
506 
507 	xfer->pdev->methods->tr_submit(xfer);
508 	return;
509 }
510 
511 void
512 libusb20_tr_start(struct libusb20_transfer *xfer)
513 {
514 	if (!xfer->is_opened) {
515 		/* transfer is not opened */
516 		return;
517 	}
518 	if (xfer->is_pending) {
519 		if (xfer->is_cancel) {
520 			/* cancelling - restart */
521 			xfer->is_restart = 1;
522 		}
523 		/* transfer not pending */
524 		return;
525 	}
526 	/* get into the callback */
527 	libusb20_tr_callback_wrapper(xfer);
528 	return;
529 }
530 
531 /* USB device operations */
532 
533 int
534 libusb20_dev_close(struct libusb20_device *pdev)
535 {
536 	struct libusb20_transfer *xfer;
537 	uint16_t x;
538 	int error = 0;
539 
540 	if (!pdev->is_opened) {
541 		return (LIBUSB20_ERROR_OTHER);
542 	}
543 	for (x = 0; x != pdev->nTransfer; x++) {
544 		xfer = pdev->pTransfer + x;
545 
546 		if (!xfer->is_opened) {
547 			/* transfer is not opened */
548 			continue;
549 		}
550 
551 		libusb20_tr_drain(xfer);
552 
553 		libusb20_tr_close(xfer);
554 	}
555 
556 	if (pdev->pTransfer != NULL) {
557 		free(pdev->pTransfer);
558 		pdev->pTransfer = NULL;
559 	}
560 	error = pdev->beMethods->close_device(pdev);
561 
562 	pdev->methods = &libusb20_dummy_methods;
563 
564 	pdev->is_opened = 0;
565 
566 	/*
567 	 * The following variable is only used by the libusb v0.1
568 	 * compat layer:
569 	 */
570 	pdev->claimed_interface = 0;
571 
572 	return (error);
573 }
574 
575 int
576 libusb20_dev_detach_kernel_driver(struct libusb20_device *pdev, uint8_t ifaceIndex)
577 {
578 	int error;
579 
580 	error = pdev->methods->detach_kernel_driver(pdev, ifaceIndex);
581 	return (error);
582 }
583 
584 struct LIBUSB20_DEVICE_DESC_DECODED *
585 libusb20_dev_get_device_desc(struct libusb20_device *pdev)
586 {
587 	return (&(pdev->ddesc));
588 }
589 
590 int
591 libusb20_dev_get_fd(struct libusb20_device *pdev)
592 {
593 	return (pdev->file);
594 }
595 
596 int
597 libusb20_dev_kernel_driver_active(struct libusb20_device *pdev, uint8_t ifaceIndex)
598 {
599 	int error;
600 
601 	error = pdev->methods->kernel_driver_active(pdev, ifaceIndex);
602 	return (error);
603 }
604 
605 int
606 libusb20_dev_open(struct libusb20_device *pdev, uint16_t nTransferMax)
607 {
608 	struct libusb20_transfer *xfer;
609 	uint32_t size;
610 	uint16_t x;
611 	int error;
612 
613 	if (pdev->is_opened) {
614 		return (LIBUSB20_ERROR_BUSY);
615 	}
616 	if (nTransferMax >= 256) {
617 		return (LIBUSB20_ERROR_INVALID_PARAM);
618 	} else if (nTransferMax != 0) {
619 		size = sizeof(pdev->pTransfer[0]) * nTransferMax;
620 		pdev->pTransfer = malloc(size);
621 		if (pdev->pTransfer == NULL) {
622 			return (LIBUSB20_ERROR_NO_MEM);
623 		}
624 		memset(pdev->pTransfer, 0, size);
625 	}
626 	/* initialise all transfers */
627 	for (x = 0; x != nTransferMax; x++) {
628 
629 		xfer = pdev->pTransfer + x;
630 
631 		xfer->pdev = pdev;
632 		xfer->trIndex = x;
633 		xfer->callback = &dummy_callback;
634 	}
635 
636 	/* set "nTransfer" early */
637 	pdev->nTransfer = nTransferMax;
638 
639 	error = pdev->beMethods->open_device(pdev, nTransferMax);
640 
641 	if (error) {
642 		if (pdev->pTransfer != NULL) {
643 			free(pdev->pTransfer);
644 			pdev->pTransfer = NULL;
645 		}
646 		pdev->file = -1;
647 		pdev->file_ctrl = -1;
648 		pdev->nTransfer = 0;
649 	} else {
650 		pdev->is_opened = 1;
651 	}
652 	return (error);
653 }
654 
655 int
656 libusb20_dev_reset(struct libusb20_device *pdev)
657 {
658 	int error;
659 
660 	error = pdev->methods->reset_device(pdev);
661 	return (error);
662 }
663 
664 int
665 libusb20_dev_set_power_mode(struct libusb20_device *pdev, uint8_t power_mode)
666 {
667 	int error;
668 
669 	error = pdev->methods->set_power_mode(pdev, power_mode);
670 	return (error);
671 }
672 
673 uint8_t
674 libusb20_dev_get_power_mode(struct libusb20_device *pdev)
675 {
676 	int error;
677 	uint8_t power_mode;
678 
679 	error = pdev->methods->get_power_mode(pdev, &power_mode);
680 	if (error)
681 		power_mode = LIBUSB20_POWER_ON;	/* fake power mode */
682 	return (power_mode);
683 }
684 
685 int
686 libusb20_dev_set_alt_index(struct libusb20_device *pdev, uint8_t ifaceIndex, uint8_t altIndex)
687 {
688 	int error;
689 
690 	error = pdev->methods->set_alt_index(pdev, ifaceIndex, altIndex);
691 	return (error);
692 }
693 
694 int
695 libusb20_dev_set_config_index(struct libusb20_device *pdev, uint8_t configIndex)
696 {
697 	int error;
698 
699 	error = pdev->methods->set_config_index(pdev, configIndex);
700 	return (error);
701 }
702 
703 int
704 libusb20_dev_request_sync(struct libusb20_device *pdev,
705     struct LIBUSB20_CONTROL_SETUP_DECODED *setup, void *data,
706     uint16_t *pactlen, uint32_t timeout, uint8_t flags)
707 {
708 	int error;
709 
710 	error = pdev->methods->do_request_sync(pdev,
711 	    setup, data, pactlen, timeout, flags);
712 	return (error);
713 }
714 
715 int
716 libusb20_dev_req_string_sync(struct libusb20_device *pdev,
717     uint8_t str_index, uint16_t langid, void *ptr, uint16_t len)
718 {
719 	struct LIBUSB20_CONTROL_SETUP_DECODED req;
720 	int error;
721 
722 	/* make sure memory is initialised */
723 	memset(ptr, 0, len);
724 
725 	if (len < 4) {
726 		/* invalid length */
727 		return (LIBUSB20_ERROR_INVALID_PARAM);
728 	}
729 	LIBUSB20_INIT(LIBUSB20_CONTROL_SETUP, &req);
730 
731 	/*
732 	 * We need to read the USB string in two steps else some USB
733 	 * devices will complain.
734 	 */
735 	req.bmRequestType =
736 	    LIBUSB20_REQUEST_TYPE_STANDARD |
737 	    LIBUSB20_RECIPIENT_DEVICE |
738 	    LIBUSB20_ENDPOINT_IN;
739 	req.bRequest = LIBUSB20_REQUEST_GET_DESCRIPTOR;
740 	req.wValue = (LIBUSB20_DT_STRING << 8) | str_index;
741 	req.wIndex = langid;
742 	req.wLength = 4;		/* bytes */
743 
744 	error = libusb20_dev_request_sync(pdev, &req,
745 	    ptr, NULL, 1000, LIBUSB20_TRANSFER_SINGLE_SHORT_NOT_OK);
746 	if (error) {
747 		return (error);
748 	}
749 	req.wLength = *(uint8_t *)ptr;	/* bytes */
750 	if (req.wLength > len) {
751 		/* partial string read */
752 		req.wLength = len;
753 	}
754 	error = libusb20_dev_request_sync(pdev, &req,
755 	    ptr, NULL, 1000, LIBUSB20_TRANSFER_SINGLE_SHORT_NOT_OK);
756 
757 	if (error) {
758 		return (error);
759 	}
760 	if (((uint8_t *)ptr)[1] != LIBUSB20_DT_STRING) {
761 		return (LIBUSB20_ERROR_OTHER);
762 	}
763 	return (0);			/* success */
764 }
765 
766 int
767 libusb20_dev_req_string_simple_sync(struct libusb20_device *pdev,
768     uint8_t str_index, void *ptr, uint16_t len)
769 {
770 	char *buf;
771 	int error;
772 	uint16_t langid;
773 	uint16_t n;
774 	uint16_t i;
775 	uint16_t c;
776 	uint8_t temp[255];
777 	uint8_t swap;
778 
779 	/* the following code derives from the FreeBSD USB kernel */
780 
781 	if ((len < 1) || (ptr == NULL)) {
782 		/* too short buffer */
783 		return (LIBUSB20_ERROR_INVALID_PARAM);
784 	}
785 	error = libusb20_dev_req_string_sync(pdev,
786 	    0, 0, temp, sizeof(temp));
787 	if (error < 0) {
788 		*(uint8_t *)ptr = 0;	/* zero terminate */
789 		return (error);
790 	}
791 	langid = temp[2] | (temp[3] << 8);
792 
793 	error = libusb20_dev_req_string_sync(pdev, str_index,
794 	    langid, temp, sizeof(temp));
795 	if (error < 0) {
796 		*(uint8_t *)ptr = 0;	/* zero terminate */
797 		return (error);
798 	}
799 	if (temp[0] < 2) {
800 		/* string length is too short */
801 		*(uint8_t *)ptr = 0;	/* zero terminate */
802 		return (LIBUSB20_ERROR_OTHER);
803 	}
804 	/* reserve one byte for terminating zero */
805 	len--;
806 
807 	/* find maximum length */
808 	n = (temp[0] / 2) - 1;
809 	if (n > len) {
810 		n = len;
811 	}
812 	/* reset swap state */
813 	swap = 3;
814 
815 	/* setup output buffer pointer */
816 	buf = ptr;
817 
818 	/* convert and filter */
819 	for (i = 0; (i != n); i++) {
820 		c = temp[(2 * i) + 2] | (temp[(2 * i) + 3] << 8);
821 
822 		/* convert from Unicode, handle buggy strings */
823 		if (((c & 0xff00) == 0) && (swap & 1)) {
824 			/* Little Endian, default */
825 			*buf = c;
826 			swap = 1;
827 		} else if (((c & 0x00ff) == 0) && (swap & 2)) {
828 			/* Big Endian */
829 			*buf = c >> 8;
830 			swap = 2;
831 		} else {
832 			/* skip invalid character */
833 			continue;
834 		}
835 		/*
836 		 * Filter by default - we don't allow greater and less than
837 		 * signs because they might confuse the dmesg printouts!
838 		 */
839 		if ((*buf == '<') || (*buf == '>') || (!isprint(*buf))) {
840 			/* skip invalid character */
841 			continue;
842 		}
843 		buf++;
844 	}
845 	*buf = 0;			/* zero terminate string */
846 
847 	return (0);
848 }
849 
850 struct libusb20_config *
851 libusb20_dev_alloc_config(struct libusb20_device *pdev, uint8_t configIndex)
852 {
853 	struct libusb20_config *retval = NULL;
854 	uint8_t *ptr;
855 	uint16_t len;
856 	uint8_t do_close;
857 	int error;
858 
859 	if (!pdev->is_opened) {
860 		error = libusb20_dev_open(pdev, 0);
861 		if (error) {
862 			return (NULL);
863 		}
864 		do_close = 1;
865 	} else {
866 		do_close = 0;
867 	}
868 	error = pdev->methods->get_config_desc_full(pdev,
869 	    &ptr, &len, configIndex);
870 
871 	if (error) {
872 		goto done;
873 	}
874 	/* parse new config descriptor */
875 	retval = libusb20_parse_config_desc(ptr);
876 
877 	/* free config descriptor */
878 	free(ptr);
879 
880 done:
881 	if (do_close) {
882 		error = libusb20_dev_close(pdev);
883 	}
884 	return (retval);
885 }
886 
887 struct libusb20_device *
888 libusb20_dev_alloc(void)
889 {
890 	struct libusb20_device *pdev;
891 
892 	pdev = malloc(sizeof(*pdev));
893 	if (pdev == NULL) {
894 		return (NULL);
895 	}
896 	memset(pdev, 0, sizeof(*pdev));
897 
898 	pdev->file = -1;
899 	pdev->file_ctrl = -1;
900 	pdev->methods = &libusb20_dummy_methods;
901 	return (pdev);
902 }
903 
904 uint8_t
905 libusb20_dev_get_config_index(struct libusb20_device *pdev)
906 {
907 	int error;
908 	uint8_t cfg_index;
909 	uint8_t do_close;
910 
911 	if (!pdev->is_opened) {
912 		error = libusb20_dev_open(pdev, 0);
913 		if (error == 0) {
914 			do_close = 1;
915 		} else {
916 			do_close = 0;
917 		}
918 	} else {
919 		do_close = 0;
920 	}
921 
922 	error = pdev->methods->get_config_index(pdev, &cfg_index);
923 	if (error) {
924 		cfg_index = 0 - 1;	/* current config index */
925 	}
926 	if (do_close) {
927 		if (libusb20_dev_close(pdev)) {
928 			/* ignore */
929 		}
930 	}
931 	return (cfg_index);
932 }
933 
934 uint8_t
935 libusb20_dev_get_mode(struct libusb20_device *pdev)
936 {
937 	return (pdev->usb_mode);
938 }
939 
940 uint8_t
941 libusb20_dev_get_speed(struct libusb20_device *pdev)
942 {
943 	return (pdev->usb_speed);
944 }
945 
946 /* if this function returns an error, the device is gone */
947 int
948 libusb20_dev_process(struct libusb20_device *pdev)
949 {
950 	int error;
951 
952 	error = pdev->methods->process(pdev);
953 	return (error);
954 }
955 
956 void
957 libusb20_dev_wait_process(struct libusb20_device *pdev, int timeout)
958 {
959 	struct pollfd pfd[1];
960 
961 	if (!pdev->is_opened) {
962 		return;
963 	}
964 	pfd[0].fd = pdev->file;
965 	pfd[0].events = (POLLIN | POLLOUT | POLLRDNORM | POLLWRNORM);
966 	pfd[0].revents = 0;
967 
968 	if (poll(pfd, 1, timeout)) {
969 		/* ignore any error */
970 	}
971 	return;
972 }
973 
974 void
975 libusb20_dev_free(struct libusb20_device *pdev)
976 {
977 	if (pdev == NULL) {
978 		/* be NULL safe */
979 		return;
980 	}
981 	if (pdev->is_opened) {
982 		if (libusb20_dev_close(pdev)) {
983 			/* ignore any errors */
984 		}
985 	}
986 	free(pdev);
987 	return;
988 }
989 
990 int
991 libusb20_dev_get_info(struct libusb20_device *pdev,
992     struct usb_device_info *pinfo)
993 {
994 	if (pinfo == NULL)
995 		return (LIBUSB20_ERROR_INVALID_PARAM);
996 
997 	return (pdev->beMethods->dev_get_info(pdev, pinfo));
998 }
999 
1000 const char *
1001 libusb20_dev_get_backend_name(struct libusb20_device *pdev)
1002 {
1003 	return (pdev->beMethods->get_backend_name());
1004 }
1005 
1006 const char *
1007 libusb20_dev_get_desc(struct libusb20_device *pdev)
1008 {
1009 	return (pdev->usb_desc);
1010 }
1011 
1012 void
1013 libusb20_dev_set_debug(struct libusb20_device *pdev, int debug)
1014 {
1015 	pdev->debug = debug;
1016 	return;
1017 }
1018 
1019 int
1020 libusb20_dev_get_debug(struct libusb20_device *pdev)
1021 {
1022 	return (pdev->debug);
1023 }
1024 
1025 uint8_t
1026 libusb20_dev_get_address(struct libusb20_device *pdev)
1027 {
1028 	return (pdev->device_address);
1029 }
1030 
1031 uint8_t
1032 libusb20_dev_get_bus_number(struct libusb20_device *pdev)
1033 {
1034 	return (pdev->bus_number);
1035 }
1036 
1037 int
1038 libusb20_dev_get_iface_desc(struct libusb20_device *pdev,
1039     uint8_t iface_index, char *buf, uint8_t len)
1040 {
1041 	if ((buf == NULL) || (len == 0))
1042 		return (LIBUSB20_ERROR_INVALID_PARAM);
1043 
1044 	return (pdev->beMethods->dev_get_iface_desc(
1045 	    pdev, iface_index, buf, len));
1046 }
1047 
1048 /* USB backend operations */
1049 
1050 int
1051 libusb20_be_get_dev_quirk(struct libusb20_backend *pbe,
1052     uint16_t quirk_index, struct libusb20_quirk *pq)
1053 {
1054 	return (pbe->methods->root_get_dev_quirk(pbe, quirk_index, pq));
1055 }
1056 
1057 int
1058 libusb20_be_get_quirk_name(struct libusb20_backend *pbe,
1059     uint16_t quirk_index, struct libusb20_quirk *pq)
1060 {
1061 	return (pbe->methods->root_get_quirk_name(pbe, quirk_index, pq));
1062 }
1063 
1064 int
1065 libusb20_be_add_dev_quirk(struct libusb20_backend *pbe,
1066     struct libusb20_quirk *pq)
1067 {
1068 	return (pbe->methods->root_add_dev_quirk(pbe, pq));
1069 }
1070 
1071 int
1072 libusb20_be_remove_dev_quirk(struct libusb20_backend *pbe,
1073     struct libusb20_quirk *pq)
1074 {
1075 	return (pbe->methods->root_remove_dev_quirk(pbe, pq));
1076 }
1077 
1078 int
1079 libusb20_be_set_template(struct libusb20_backend *pbe, int temp)
1080 {
1081 	return (pbe->methods->root_set_template(pbe, temp));
1082 }
1083 
1084 int
1085 libusb20_be_get_template(struct libusb20_backend *pbe, int *ptemp)
1086 {
1087 	int temp;
1088 
1089 	if (ptemp == NULL)
1090 		ptemp = &temp;
1091 
1092 	return (pbe->methods->root_get_template(pbe, ptemp));
1093 }
1094 
1095 struct libusb20_device *
1096 libusb20_be_device_foreach(struct libusb20_backend *pbe, struct libusb20_device *pdev)
1097 {
1098 	if (pbe == NULL) {
1099 		pdev = NULL;
1100 	} else if (pdev == NULL) {
1101 		pdev = TAILQ_FIRST(&(pbe->usb_devs));
1102 	} else {
1103 		pdev = TAILQ_NEXT(pdev, dev_entry);
1104 	}
1105 	return (pdev);
1106 }
1107 
1108 struct libusb20_backend *
1109 libusb20_be_alloc(const struct libusb20_backend_methods *methods)
1110 {
1111 	struct libusb20_backend *pbe;
1112 
1113 	pbe = malloc(sizeof(*pbe));
1114 	if (pbe == NULL) {
1115 		return (NULL);
1116 	}
1117 	memset(pbe, 0, sizeof(*pbe));
1118 
1119 	TAILQ_INIT(&(pbe->usb_devs));
1120 
1121 	pbe->methods = methods;		/* set backend methods */
1122 
1123 	/* do the initial device scan */
1124 	if (pbe->methods->init_backend) {
1125 		pbe->methods->init_backend(pbe);
1126 	}
1127 	return (pbe);
1128 }
1129 
1130 struct libusb20_backend *
1131 libusb20_be_alloc_linux(void)
1132 {
1133 	struct libusb20_backend *pbe;
1134 
1135 #ifdef __linux__
1136 	pbe = libusb20_be_alloc(&libusb20_linux_backend);
1137 #else
1138 	pbe = NULL;
1139 #endif
1140 	return (pbe);
1141 }
1142 
1143 struct libusb20_backend *
1144 libusb20_be_alloc_ugen20(void)
1145 {
1146 	struct libusb20_backend *pbe;
1147 
1148 #ifdef __FreeBSD__
1149 	pbe = libusb20_be_alloc(&libusb20_ugen20_backend);
1150 #else
1151 	pbe = NULL;
1152 #endif
1153 	return (pbe);
1154 }
1155 
1156 struct libusb20_backend *
1157 libusb20_be_alloc_default(void)
1158 {
1159 	struct libusb20_backend *pbe;
1160 
1161 	pbe = libusb20_be_alloc_linux();
1162 	if (pbe) {
1163 		return (pbe);
1164 	}
1165 	pbe = libusb20_be_alloc_ugen20();
1166 	if (pbe) {
1167 		return (pbe);
1168 	}
1169 	return (NULL);			/* no backend found */
1170 }
1171 
1172 void
1173 libusb20_be_free(struct libusb20_backend *pbe)
1174 {
1175 	struct libusb20_device *pdev;
1176 
1177 	if (pbe == NULL) {
1178 		/* be NULL safe */
1179 		return;
1180 	}
1181 	while ((pdev = libusb20_be_device_foreach(pbe, NULL))) {
1182 		libusb20_be_dequeue_device(pbe, pdev);
1183 		libusb20_dev_free(pdev);
1184 	}
1185 	if (pbe->methods->exit_backend) {
1186 		pbe->methods->exit_backend(pbe);
1187 	}
1188 	/* free backend */
1189 	free(pbe);
1190 }
1191 
1192 void
1193 libusb20_be_enqueue_device(struct libusb20_backend *pbe, struct libusb20_device *pdev)
1194 {
1195 	pdev->beMethods = pbe->methods;	/* copy backend methods */
1196 	TAILQ_INSERT_TAIL(&(pbe->usb_devs), pdev, dev_entry);
1197 }
1198 
1199 void
1200 libusb20_be_dequeue_device(struct libusb20_backend *pbe,
1201     struct libusb20_device *pdev)
1202 {
1203 	TAILQ_REMOVE(&(pbe->usb_devs), pdev, dev_entry);
1204 }
1205