VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/freebsd/USBProxyDevice-freebsd.cpp@ 41997

Last change on this file since 41997 was 37596, checked in by vboxsync, 14 years ago

*: RTFILE becomes a pointer, RTFileOpen++ expands it's flags paramter from uint32_t to uint64_t.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 29.8 KB
Line 
1/* $Id: USBProxyDevice-freebsd.cpp 37596 2011-06-22 19:30:06Z vboxsync $ */
2/** @file
3 * USB device proxy - the FreeBSD backend.
4 */
5
6/*
7 * Copyright (C) 2006-2007 Oracle Corporation
8 * Copyright (C) 2010 Hans Petter Selasky
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 */
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_DRV_USBPROXY
23#ifdef VBOX
24# include <iprt/stdint.h>
25#endif
26#include <sys/types.h>
27#include <sys/stat.h>
28#include <sys/ioctl.h>
29#include <sys/poll.h>
30#include <stdint.h>
31#include <stdio.h>
32#include <string.h>
33#include <stdlib.h>
34#include <limits.h>
35#include <unistd.h>
36#include <fcntl.h>
37#include <errno.h>
38#include <dev/usb/usb.h>
39#include <dev/usb/usbdi.h>
40#include <dev/usb/usb_ioctl.h>
41
42#include <VBox/vmm/pdm.h>
43#include <VBox/err.h>
44#include <VBox/log.h>
45#include <VBox/vusb.h>
46#include <iprt/assert.h>
47#include <iprt/stream.h>
48#include <iprt/alloc.h>
49#include <iprt/thread.h>
50#include <iprt/time.h>
51#include <iprt/asm.h>
52#include <iprt/string.h>
53#include <iprt/file.h>
54#include "../USBProxyDevice.h"
55
56/** Maximum endpoints supported. */
57#define USBFBSD_MAXENDPOINTS 127
58#define USBFBSD_MAXFRAMES 56
59
60/** This really needs to be defined in vusb.h! */
61#ifndef VUSB_DIR_TO_DEV
62# define VUSB_DIR_TO_DEV 0x00
63#endif
64
65/*******************************************************************************
66* Structures and Typedefs *
67*******************************************************************************/
68typedef struct USBENDPOINTFBSD
69{
70 /** Flag whether it is opened. */
71 bool fOpen;
72 /** Flag whether it is cancelling. */
73 bool fCancelling;
74 /** Buffer pointers. */
75 void *apvData[USBFBSD_MAXFRAMES];
76 /** Buffer lengths. */
77 uint32_t acbData[USBFBSD_MAXFRAMES];
78 /** Initial buffer length. */
79 uint32_t cbData0;
80 /** Pointer to the URB. */
81 PVUSBURB pUrb;
82 /** Copy of endpoint number. */
83 unsigned iEpNum;
84 /** Maximum transfer length. */
85 unsigned cMaxIo;
86 /** Maximum frame count. */
87 unsigned cMaxFrames;
88} USBENDPOINTFBSD, *PUSBENDPOINTFBSD;
89
90/**
91 * Data for the FreeBSD usb proxy backend.
92 */
93typedef struct USBPROXYDEVFBSD
94{
95 /** The open file. */
96 RTFILE hFile;
97 /** Software endpoint structures */
98 USBENDPOINTFBSD aSwEndpoint[USBFBSD_MAXENDPOINTS];
99 /** Flag whether an URB is cancelling. */
100 bool fCancelling;
101 /** Flag whether initialised or not */
102 bool fInit;
103 /** Kernel endpoint structures */
104 struct usb_fs_endpoint aHwEndpoint[USBFBSD_MAXENDPOINTS];
105} USBPROXYDEVFBSD, *PUSBPROXYDEVFBSD;
106
107/*******************************************************************************
108* Internal Functions *
109*******************************************************************************/
110static int usbProxyFreeBSDEndpointClose(PUSBPROXYDEV pProxyDev, int Endpoint);
111
112/**
113 * Wrapper for the ioctl call.
114 *
115 * This wrapper will repeat the call if we get an EINTR or EAGAIN. It can also
116 * handle ENODEV (detached device) errors.
117 *
118 * @returns whatever ioctl returns.
119 * @param pProxyDev The proxy device.
120 * @param iCmd The ioctl command / function.
121 * @param pvArg The ioctl argument / data.
122 * @param fHandleNoDev Whether to handle ENXIO.
123 * @internal
124 */
125static int usbProxyFreeBSDDoIoCtl(PUSBPROXYDEV pProxyDev, unsigned long iCmd,
126 void *pvArg, bool fHandleNoDev)
127{
128 int rc = VINF_SUCCESS;
129 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
130
131 LogFlow(("usbProxyFreeBSDDoIoCtl: iCmd=%#x\n", iCmd));
132
133 do
134 {
135 rc = ioctl(RTFileToNative(pDevFBSD->hFile), iCmd, pvArg);
136 if (rc >= 0)
137 return VINF_SUCCESS;
138 } while (errno == EINTR);
139
140 if (errno == ENXIO && fHandleNoDev)
141 {
142 Log(("usbProxyFreeBSDDoIoCtl: ENXIO -> unplugged. pProxyDev=%s\n",
143 pProxyDev->pUsbIns->pszName));
144 errno = ENODEV;
145 }
146 else if (errno != EAGAIN)
147 {
148 LogFlow(("usbProxyFreeBSDDoIoCtl: Returned %d. pProxyDev=%s\n",
149 errno, pProxyDev->pUsbIns->pszName));
150 }
151 return RTErrConvertFromErrno(errno);
152}
153
154/**
155 * Init USB subsystem.
156 */
157static int usbProxyFreeBSDFsInit(PUSBPROXYDEV pProxyDev)
158{
159 struct usb_fs_init UsbFsInit;
160 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
161 int rc;
162
163 LogFlow(("usbProxyFreeBSDFsInit: pProxyDev=%p\n", (void *)pProxyDev));
164
165 /* Sanity check */
166 AssertPtrReturn(pDevFBSD, VERR_INVALID_PARAMETER);
167
168 if (pDevFBSD->fInit == true)
169 return VINF_SUCCESS;
170
171 /* Zero default */
172 memset(&UsbFsInit, 0, sizeof(UsbFsInit));
173
174 UsbFsInit.pEndpoints = pDevFBSD->aHwEndpoint;
175 UsbFsInit.ep_index_max = USBFBSD_MAXENDPOINTS;
176
177 /* Init USB subsystem */
178 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_FS_INIT, &UsbFsInit, false);
179 if (RT_SUCCESS(rc))
180 pDevFBSD->fInit = true;
181
182 return rc;
183}
184
185/**
186 * Uninit USB subsystem.
187 */
188static int usbProxyFreeBSDFsUnInit(PUSBPROXYDEV pProxyDev)
189{
190 struct usb_fs_uninit UsbFsUninit;
191 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
192 int rc;
193
194 LogFlow(("usbProxyFreeBSDFsUnInit: ProxyDev=%p\n", (void *)pProxyDev));
195
196 /* Sanity check */
197 AssertPtrReturn(pDevFBSD, VERR_INVALID_PARAMETER);
198
199 if (pDevFBSD->fInit != true)
200 return VINF_SUCCESS;
201
202 /* Close any open endpoints. */
203 for (unsigned n = 0; n != USBFBSD_MAXENDPOINTS; n++)
204 usbProxyFreeBSDEndpointClose(pProxyDev, n);
205
206 /* Zero default */
207 memset(&UsbFsUninit, 0, sizeof(UsbFsUninit));
208
209 /* Uninit USB subsystem */
210 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_FS_UNINIT, &UsbFsUninit, false);
211 if (RT_SUCCESS(rc))
212 pDevFBSD->fInit = false;
213
214 return rc;
215}
216
217/**
218 * Setup a USB request packet.
219 */
220static void usbProxyFreeBSDSetupReq(struct usb_device_request *pSetupData,
221 uint8_t bmRequestType, uint8_t bRequest, uint16_t wValue,
222 uint16_t wIndex, uint16_t wLength)
223{
224 LogFlow(("usbProxyFreeBSDSetupReq: pSetupData=%p bmRequestType=%x "
225 "bRequest=%x wValue=%x wIndex=%x wLength=%x\n", (void *)pSetupData,
226 bmRequestType, bRequest, wValue, wIndex, wLength));
227
228 pSetupData->bmRequestType = bmRequestType;
229 pSetupData->bRequest = bRequest;
230
231 /* Handle endianess here. Currently no swapping is needed. */
232 pSetupData->wValue[0] = wValue & 0xff;
233 pSetupData->wValue[1] = (wValue >> 8) & 0xff;
234 pSetupData->wIndex[0] = wIndex & 0xff;
235 pSetupData->wIndex[1] = (wIndex >> 8) & 0xff;
236 pSetupData->wLength[0] = wLength & 0xff;
237 pSetupData->wLength[1] = (wLength >> 8) & 0xff;
238}
239
240static int usbProxyFreeBSDEndpointOpen(PUSBPROXYDEV pProxyDev, int Endpoint, bool fIsoc, int index)
241{
242 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
243 PUSBENDPOINTFBSD pEndpointFBSD = NULL; /* shut up gcc */
244 struct usb_fs_endpoint *pXferEndpoint;
245 struct usb_fs_open UsbFsOpen;
246 int rc;
247
248 LogFlow(("usbProxyFreeBSDEndpointOpen: pProxyDev=%p Endpoint=%d\n",
249 (void *)pProxyDev, Endpoint));
250
251 for (; index < USBFBSD_MAXENDPOINTS; index++)
252 {
253 pEndpointFBSD = &pDevFBSD->aSwEndpoint[index];
254 if (pEndpointFBSD->fCancelling)
255 continue;
256 if ( pEndpointFBSD->fOpen
257 && !pEndpointFBSD->pUrb
258 && (int)pEndpointFBSD->iEpNum == Endpoint)
259 return index;
260 }
261
262 if (index == USBFBSD_MAXENDPOINTS)
263 {
264 for (index = 0; index != USBFBSD_MAXENDPOINTS; index++)
265 {
266 pEndpointFBSD = &pDevFBSD->aSwEndpoint[index];
267 if (pEndpointFBSD->fCancelling)
268 continue;
269 if (!pEndpointFBSD->fOpen)
270 break;
271 }
272 if (index == USBFBSD_MAXENDPOINTS)
273 return -1;
274 }
275 /* set ppBuffer and pLength */
276
277 pXferEndpoint = &pDevFBSD->aHwEndpoint[index];
278 pXferEndpoint->ppBuffer = &pEndpointFBSD->apvData[0];
279 pXferEndpoint->pLength = &pEndpointFBSD->acbData[0];
280
281 LogFlow(("usbProxyFreeBSDEndpointOpen: ep_index=%d ep_num=%d\n",
282 index, Endpoint));
283
284 memset(&UsbFsOpen, 0, sizeof(UsbFsOpen));
285
286 UsbFsOpen.ep_index = index;
287 UsbFsOpen.ep_no = Endpoint;
288 UsbFsOpen.max_bufsize = 256 * 1024;
289 /* Hardcoded assumption about the URBs we get. */
290
291 UsbFsOpen.max_frames = fIsoc ? USBFBSD_MAXFRAMES : 2;
292
293 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_FS_OPEN, &UsbFsOpen, true);
294 if (RT_FAILURE(rc))
295 {
296 if (rc == VERR_RESOURCE_BUSY)
297 LogFlow(("usbProxyFreeBSDEndpointOpen: EBUSY\n"));
298
299 return -1;
300 }
301 pEndpointFBSD->fOpen = true;
302 pEndpointFBSD->pUrb = NULL;
303 pEndpointFBSD->iEpNum = Endpoint;
304 pEndpointFBSD->cMaxIo = UsbFsOpen.max_bufsize;
305 pEndpointFBSD->cMaxFrames = UsbFsOpen.max_frames;
306
307 return index;
308}
309
310/**
311 * Close an endpoint.
312 *
313 * @returns VBox status code.
314 */
315static int usbProxyFreeBSDEndpointClose(PUSBPROXYDEV pProxyDev, int Endpoint)
316{
317 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
318 PUSBENDPOINTFBSD pEndpointFBSD = &pDevFBSD->aSwEndpoint[Endpoint];
319 struct usb_fs_close UsbFsClose;
320 int rc = VINF_SUCCESS;
321
322 LogFlow(("usbProxyFreeBSDEndpointClose: pProxyDev=%p Endpoint=%d\n",
323 (void *)pProxyDev, Endpoint));
324
325 /* check for cancelling */
326 if (pEndpointFBSD->pUrb != NULL)
327 {
328 pEndpointFBSD->fCancelling = true;
329 pDevFBSD->fCancelling = true;
330 }
331
332 /* check for opened */
333 if (pEndpointFBSD->fOpen)
334 {
335 pEndpointFBSD->fOpen = false;
336
337 /* Zero default */
338 memset(&UsbFsClose, 0, sizeof(UsbFsClose));
339
340 /* Set endpoint index */
341 UsbFsClose.ep_index = Endpoint;
342
343 /* Close endpoint */
344 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_FS_CLOSE, &UsbFsClose, true);
345 }
346 return rc;
347}
348
349/**
350 * Opens the device file.
351 *
352 * @returns VBox status code.
353 * @param pProxyDev The device instance.
354 * @param pszAddress If we are using usbfs, this is the path to the
355 * device. If we are using sysfs, this is a string of
356 * the form "sysfs:<sysfs path>//device:<device node>".
357 * In the second case, the two paths are guaranteed
358 * not to contain the substring "//".
359 * @param pvBackend Backend specific pointer, unused for the linux backend.
360 */
361static int usbProxyFreeBSDOpen(PUSBPROXYDEV pProxyDev, const char *pszAddress,
362 void *pvBackend)
363{
364 int rc;
365
366 LogFlow(("usbProxyFreeBSDOpen: pProxyDev=%p pszAddress=%s\n", pProxyDev, pszAddress));
367
368 /*
369 * Try open the device node.
370 */
371 RTFILE hFile;
372 rc = RTFileOpen(&hFile, pszAddress, RTFILE_O_READWRITE | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
373 if (RT_SUCCESS(rc))
374 {
375 /*
376 * Allocate and initialize the linux backend data.
377 */
378 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD)RTMemAllocZ(sizeof(USBPROXYDEVFBSD));
379 if (pDevFBSD)
380 {
381 pDevFBSD->hFile = hFile;
382 pProxyDev->Backend.pv = pDevFBSD;
383
384 rc = usbProxyFreeBSDFsInit(pProxyDev);
385 if (RT_SUCCESS(rc))
386 {
387 LogFlow(("usbProxyFreeBSDOpen(%p, %s): returns successfully hFile=%RTfile iActiveCfg=%d\n",
388 pProxyDev, pszAddress, pDevFBSD->hFile, pProxyDev->iActiveCfg));
389
390 return VINF_SUCCESS;
391 }
392
393 RTMemFree(pDevFBSD);
394 }
395 else
396 rc = VERR_NO_MEMORY;
397
398 RTFileClose(hFile);
399 }
400 else if (rc == VERR_ACCESS_DENIED)
401 rc = VERR_VUSB_USBFS_PERMISSION;
402
403 Log(("usbProxyFreeBSDOpen(%p, %s) failed, rc=%d!\n",
404 pProxyDev, pszAddress, rc));
405
406 pProxyDev->Backend.pv = NULL;
407
408 NOREF(pvBackend);
409 return rc;
410}
411
412
413/**
414 * Claims all the interfaces and figures out the
415 * current configuration.
416 *
417 * @returns VINF_SUCCESS.
418 * @param pProxyDev The proxy device.
419 */
420static int usbProxyFreeBSDInit(PUSBPROXYDEV pProxyDev)
421{
422 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
423 int rc;
424
425 LogFlow(("usbProxyFreeBSDInit: pProxyDev=%s\n",
426 pProxyDev->pUsbIns->pszName));
427
428 /* Retrieve current active configuration. */
429 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_GET_CONFIG,
430 &pProxyDev->iActiveCfg, true);
431 if (RT_FAILURE(rc) || pProxyDev->iActiveCfg == 255)
432 {
433 pProxyDev->cIgnoreSetConfigs = 0;
434 pProxyDev->iActiveCfg = -1;
435 }
436 else
437 {
438 pProxyDev->cIgnoreSetConfigs = 1;
439 pProxyDev->iActiveCfg++;
440 }
441
442 Log(("usbProxyFreeBSDInit: iActiveCfg=%d\n", pProxyDev->iActiveCfg));
443
444 return rc;
445}
446
447/**
448 * Closes the proxy device.
449 */
450static void usbProxyFreeBSDClose(PUSBPROXYDEV pProxyDev)
451{
452 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD)pProxyDev->Backend.pv;
453
454 LogFlow(("usbProxyFreeBSDClose: pProxyDev=%s\n", pProxyDev->pUsbIns->pszName));
455
456 /* sanity check */
457 AssertPtrReturnVoid(pDevFBSD);
458
459 usbProxyFreeBSDFsUnInit(pProxyDev);
460
461 RTFileClose(pDevFBSD->hFile);
462 pDevFBSD->hFile = NIL_RTFILE;
463
464 RTMemFree(pDevFBSD);
465 pProxyDev->Backend.pv = NULL;
466
467 LogFlow(("usbProxyFreeBSDClose: returns\n"));
468}
469
470/**
471 * Reset a device.
472 *
473 * @returns VBox status code.
474 * @param pDev The device to reset.
475 */
476static int usbProxyFreeBSDReset(PUSBPROXYDEV pProxyDev, bool fResetOnFreeBSD)
477{
478 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
479 int iParm;
480 int rc = VINF_SUCCESS;
481
482 LogFlow(("usbProxyFreeBSDReset: pProxyDev=%s\n",
483 pProxyDev->pUsbIns->pszName));
484
485 if (!fResetOnFreeBSD)
486 goto done;
487
488 /* We need to release kernel ressources first. */
489 rc = usbProxyFreeBSDFsUnInit(pProxyDev);
490 if (RT_FAILURE(rc))
491 goto done;
492
493 /* Resetting is only possible as super-user, ignore any failures: */
494 iParm = 0;
495 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_DEVICEENUMERATE, &iParm, true);
496 if (RT_FAILURE(rc))
497 {
498 /* Set the config instead of bus reset */
499 iParm = 255;
500 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_SET_CONFIG, &iParm, true);
501 if (RT_SUCCESS(rc))
502 {
503 iParm = 0;
504 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_SET_CONFIG, &iParm, true);
505 }
506 }
507 usleep(10000); /* nice it! */
508
509 /* Allocate kernel ressources again. */
510 rc = usbProxyFreeBSDFsInit(pProxyDev);
511 if (RT_FAILURE(rc))
512 goto done;
513
514 /* Retrieve current active configuration. */
515 rc = usbProxyFreeBSDInit(pProxyDev);
516
517done:
518 pProxyDev->cIgnoreSetConfigs = 2;
519
520 return rc;
521}
522
523/**
524 * SET_CONFIGURATION.
525 *
526 * The caller makes sure that it's not called first time after open or reset
527 * with the active interface.
528 *
529 * @returns success indicator.
530 * @param pProxyDev The device instance data.
531 * @param iCfg The configuration to set.
532 */
533static int usbProxyFreeBSDSetConfig(PUSBPROXYDEV pProxyDev, int iCfg)
534{
535 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
536 int iCfgIndex;
537 int rc;
538
539 LogFlow(("usbProxyFreeBSDSetConfig: pProxyDev=%s cfg=%x\n",
540 pProxyDev->pUsbIns->pszName, iCfg));
541
542 /* We need to release kernel ressources first. */
543 rc = usbProxyFreeBSDFsUnInit(pProxyDev);
544 if (RT_FAILURE(rc))
545 {
546 LogFlow(("usbProxyFreeBSDSetInterface: Freeing kernel resources "
547 "failed failed rc=%d\n", rc));
548 return false;
549 }
550
551 if (iCfg == 0)
552 {
553 /* Unconfigure */
554 iCfgIndex = 255;
555 }
556 else
557 {
558 /* Get the configuration index matching the value. */
559 for (iCfgIndex = 0; iCfgIndex < pProxyDev->DevDesc.bNumConfigurations; iCfgIndex++)
560 {
561 if (pProxyDev->paCfgDescs[iCfgIndex].Core.bConfigurationValue == iCfg)
562 break;
563 }
564
565 if (iCfgIndex == pProxyDev->DevDesc.bNumConfigurations)
566 {
567 LogFlow(("usbProxyFreeBSDSetConfig: configuration "
568 "%d not found\n", iCfg));
569 return false;
570 }
571 }
572
573 /* Set the config */
574 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_SET_CONFIG, &iCfgIndex, true);
575 if (RT_FAILURE(rc))
576 return false;
577
578 /* Allocate kernel ressources again. */
579 rc = usbProxyFreeBSDFsInit(pProxyDev);
580 if (RT_FAILURE(rc))
581 return false;
582
583 return true;
584}
585
586/**
587 * Claims an interface.
588 * @returns success indicator.
589 */
590static int usbProxyFreeBSDClaimInterface(PUSBPROXYDEV pProxyDev, int iIf)
591{
592 int rc;
593
594 LogFlow(("usbProxyFreeBSDClaimInterface: pProxyDev=%s "
595 "ifnum=%x\n", pProxyDev->pUsbIns->pszName, iIf));
596
597 /*
598 * Try to detach kernel driver on this interface, ignore any
599 * failures
600 */
601 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_IFACE_DRIVER_DETACH, &iIf, true);
602
603 /* Try to claim interface */
604 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_CLAIM_INTERFACE, &iIf, true);
605 if (RT_FAILURE(rc))
606 return false;
607
608 return true;
609}
610
611/**
612 * Releases an interface.
613 * @returns success indicator.
614 */
615static int usbProxyFreeBSDReleaseInterface(PUSBPROXYDEV pProxyDev, int iIf)
616{
617 int rc;
618
619 LogFlow(("usbProxyFreeBSDReleaseInterface: pProxyDev=%s "
620 "ifnum=%x\n", pProxyDev->pUsbIns->pszName, iIf));
621
622 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_RELEASE_INTERFACE, &iIf, true);
623 if (RT_FAILURE(rc))
624 return false;
625
626 return true;
627}
628
629/**
630 * SET_INTERFACE.
631 *
632 * @returns success indicator.
633 */
634static int
635usbProxyFreeBSDSetInterface(PUSBPROXYDEV pProxyDev, int iIf, int iAlt)
636{
637 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
638 struct usb_alt_interface UsbIntAlt;
639 int rc;
640
641 LogFlow(("usbProxyFreeBSDSetInterface: pProxyDev=%p iIf=%x iAlt=%x\n",
642 pProxyDev, iIf, iAlt));
643
644 /* We need to release kernel ressources first. */
645 rc = usbProxyFreeBSDFsUnInit(pProxyDev);
646 if (RT_FAILURE(rc))
647 {
648 LogFlow(("usbProxyFreeBSDSetInterface: Freeing kernel resources "
649 "failed failed rc=%d\n", rc));
650 return false;
651 }
652 memset(&UsbIntAlt, 0, sizeof(UsbIntAlt));
653 UsbIntAlt.uai_interface_index = iIf;
654 UsbIntAlt.uai_alt_index = iAlt;
655
656 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_SET_ALTINTERFACE, &UsbIntAlt, true);
657 if (RT_FAILURE(rc))
658 {
659 LogFlow(("usbProxyFreeBSDSetInterface: Setting interface %d %d "
660 "failed rc=%d\n", iIf, iAlt, rc));
661 return false;
662 }
663
664 rc = usbProxyFreeBSDFsInit(pProxyDev);
665 if (RT_FAILURE(rc))
666 return false;
667
668 return true;
669}
670
671/**
672 * Clears the halted endpoint 'ep_num'.
673 */
674static bool usbProxyFreeBSDClearHaltedEp(PUSBPROXYDEV pProxyDev, unsigned int ep_num)
675{
676 struct usb_ctl_request Req;
677 int rc;
678
679 LogFlow(("usbProxyFreeBSDClearHaltedEp: pProxyDev=%s ep_num=%u\n",
680 pProxyDev->pUsbIns->pszName, ep_num));
681
682 /*
683 * Clearing the zero control pipe doesn't make sense.
684 * Just ignore it.
685 */
686 if ((ep_num & 0xF) == 0)
687 return true;
688
689 memset(&Req, 0, sizeof(Req));
690
691 usbProxyFreeBSDSetupReq(&Req.ucr_request,
692 VUSB_DIR_TO_DEV | VUSB_TO_ENDPOINT,
693 VUSB_REQ_CLEAR_FEATURE, 0, ep_num, 0);
694
695 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_DO_REQUEST, &Req, true);
696
697 LogFlow(("usbProxyFreeBSDClearHaltedEp: rc=%Rrc\n", rc));
698
699 if (RT_FAILURE(rc))
700 return false;
701
702 return true;
703}
704
705/**
706 * @copydoc USBPROXYBACK::pfnUrbQueue
707 */
708static int usbProxyFreeBSDUrbQueue(PVUSBURB pUrb)
709{
710 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUrb->pUsbIns, PUSBPROXYDEV);
711 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
712 PUSBENDPOINTFBSD pEndpointFBSD;
713 struct usb_fs_endpoint *pXferEndpoint;
714 struct usb_fs_start UsbFsStart;
715 unsigned cFrames;
716 uint8_t *pbData;
717 int index;
718 int ep_num;
719 int rc;
720
721 LogFlow(("usbProxyFreeBSDUrbQueue: pUrb=%p EndPt=%u Dir=%u\n",
722 pUrb, (unsigned)pUrb->EndPt, (unsigned)pUrb->enmDir));
723
724 ep_num = pUrb->EndPt;
725
726 if ((pUrb->enmType != VUSBXFERTYPE_MSG) && (pUrb->enmDir == VUSBDIRECTION_IN))
727 ep_num |= 0x80;
728
729 index = 0;
730
731retry:
732
733 index = usbProxyFreeBSDEndpointOpen(pProxyDev, ep_num,
734 (pUrb->enmType == VUSBXFERTYPE_ISOC),
735 index);
736
737 if (index < 0)
738 return false;
739
740 pEndpointFBSD = &pDevFBSD->aSwEndpoint[index];
741 pXferEndpoint = &pDevFBSD->aHwEndpoint[index];
742
743 pbData = pUrb->abData;
744
745 switch (pUrb->enmType)
746 {
747 case VUSBXFERTYPE_MSG:
748 {
749 pEndpointFBSD->apvData[0] = pbData;
750 pEndpointFBSD->acbData[0] = 8;
751
752 /* check wLength */
753 if (pbData[6] || pbData[7])
754 {
755 pEndpointFBSD->apvData[1] = pbData + 8;
756 pEndpointFBSD->acbData[1] = pbData[6] | (pbData[7] << 8);
757 cFrames = 2;
758 }
759 else
760 {
761 pEndpointFBSD->apvData[1] = NULL;
762 pEndpointFBSD->acbData[1] = 0;
763 cFrames = 1;
764 }
765
766 LogFlow(("usbProxyFreeBSDUrbQueue: pUrb->cbData=%u, 0x%02x, "
767 "0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x, 0x%02x\n",
768 pUrb->cbData, pbData[0], pbData[1], pbData[2], pbData[3],
769 pbData[4], pbData[5], pbData[6], pbData[7]));
770
771 pXferEndpoint->timeout = USB_FS_TIMEOUT_NONE;
772 pXferEndpoint->flags = USB_FS_FLAG_MULTI_SHORT_OK;
773 break;
774 }
775 case VUSBXFERTYPE_ISOC:
776 {
777 unsigned i;
778
779 for (i = 0; i < pUrb->cIsocPkts; i++)
780 {
781 if (i >= pEndpointFBSD->cMaxFrames)
782 break;
783 pEndpointFBSD->apvData[i] = pbData + pUrb->aIsocPkts[i].off;
784 pEndpointFBSD->acbData[i] = pUrb->aIsocPkts[i].cb;
785 }
786 /* Timeout handling will be done during reap. */
787 pXferEndpoint->timeout = USB_FS_TIMEOUT_NONE;
788 pXferEndpoint->flags = USB_FS_FLAG_MULTI_SHORT_OK;
789 cFrames = i;
790 break;
791 }
792 default:
793 {
794 pEndpointFBSD->apvData[0] = pbData;
795 pEndpointFBSD->cbData0 = pUrb->cbData;
796
797 /* XXX maybe we have to loop */
798 if (pUrb->cbData > pEndpointFBSD->cMaxIo)
799 pEndpointFBSD->acbData[0] = pEndpointFBSD->cMaxIo;
800 else
801 pEndpointFBSD->acbData[0] = pUrb->cbData;
802
803 /* Timeout handling will be done during reap. */
804 pXferEndpoint->timeout = USB_FS_TIMEOUT_NONE;
805 pXferEndpoint->flags = pUrb->fShortNotOk ? 0 : USB_FS_FLAG_MULTI_SHORT_OK;
806 cFrames = 1;
807 break;
808 }
809 }
810
811 /* store number of frames */
812 pXferEndpoint->nFrames = cFrames;
813
814 /* zero-default */
815 memset(&UsbFsStart, 0, sizeof(UsbFsStart));
816
817 /* Start the transfer */
818 UsbFsStart.ep_index = index;
819
820 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_FS_START, &UsbFsStart, true);
821
822 LogFlow(("usbProxyFreeBSDUrbQueue: USB_FS_START returned rc=%d "
823 "len[0]=%u len[1]=%u cbData=%u index=%u ep_num=%u\n", rc,
824 (unsigned)pEndpointFBSD->acbData[0],
825 (unsigned)pEndpointFBSD->acbData[1],
826 (unsigned)pUrb->cbData,
827 (unsigned)index, (unsigned)ep_num));
828
829 if (RT_FAILURE(rc))
830 {
831 if (rc == VERR_RESOURCE_BUSY)
832 {
833 index++;
834 goto retry;
835 }
836 return false;
837 }
838 pUrb->Dev.pvPrivate = (void *)(long)(index + 1);
839 pEndpointFBSD->pUrb = pUrb;
840
841 return true;
842}
843
844/**
845 * Reap URBs in-flight on a device.
846 *
847 * @returns Pointer to a completed URB.
848 * @returns NULL if no URB was completed.
849 * @param pProxyDev The device.
850 * @param cMillies Number of milliseconds to wait. Use 0 to not wait at all.
851 */
852static PVUSBURB usbProxyFreeBSDUrbReap(PUSBPROXYDEV pProxyDev, RTMSINTERVAL cMillies)
853{
854 struct usb_fs_endpoint *pXferEndpoint;
855 PUSBPROXYDEVFBSD pDevFBSD = (PUSBPROXYDEVFBSD) pProxyDev->Backend.pv;
856 PUSBENDPOINTFBSD pEndpointFBSD;
857 PVUSBURB pUrb;
858 struct usb_fs_complete UsbFsComplete;
859 struct pollfd PollFd;
860 int rc;
861
862 LogFlow(("usbProxyFreeBSDUrbReap: pProxyDev=%p, cMillies=%u\n",
863 pProxyDev, cMillies));
864
865repeat:
866
867 pUrb = NULL;
868
869 /* check for cancelled transfers */
870 if (pDevFBSD->fCancelling)
871 {
872 for (unsigned n = 0; n < USBFBSD_MAXENDPOINTS; n++)
873 {
874 pEndpointFBSD = &pDevFBSD->aSwEndpoint[n];
875 if (pEndpointFBSD->fCancelling)
876 {
877 pEndpointFBSD->fCancelling = false;
878 pUrb = pEndpointFBSD->pUrb;
879 pEndpointFBSD->pUrb = NULL;
880
881 if (pUrb != NULL)
882 break;
883 }
884 }
885
886 if (pUrb != NULL)
887 {
888 pUrb->enmStatus = VUSBSTATUS_INVALID;
889 pUrb->Dev.pvPrivate = NULL;
890
891 switch (pUrb->enmType)
892 {
893 case VUSBXFERTYPE_MSG:
894 pUrb->cbData = 0;
895 break;
896 case VUSBXFERTYPE_ISOC:
897 pUrb->cbData = 0;
898 for (int n = 0; n < (int)pUrb->cIsocPkts; n++)
899 pUrb->aIsocPkts[n].cb = 0;
900 break;
901 default:
902 pUrb->cbData = 0;
903 break;
904 }
905 return pUrb;
906 }
907 pDevFBSD->fCancelling = false;
908 }
909 /* Zero default */
910
911 memset(&UsbFsComplete, 0, sizeof(UsbFsComplete));
912
913 /* Check if any endpoints are complete */
914 rc = usbProxyFreeBSDDoIoCtl(pProxyDev, USB_FS_COMPLETE, &UsbFsComplete, true);
915 if (RT_SUCCESS(rc))
916 {
917 pXferEndpoint = &pDevFBSD->aHwEndpoint[UsbFsComplete.ep_index];
918 pEndpointFBSD = &pDevFBSD->aSwEndpoint[UsbFsComplete.ep_index];
919
920 LogFlow(("usbProxyFreeBSDUrbReap: Reaped "
921 "URB %#p\n", pEndpointFBSD->pUrb));
922
923 if (pXferEndpoint->status == USB_ERR_CANCELLED)
924 goto repeat;
925
926 pUrb = pEndpointFBSD->pUrb;
927 pEndpointFBSD->pUrb = NULL;
928 if (pUrb == NULL)
929 goto repeat;
930
931 switch (pXferEndpoint->status)
932 {
933 case USB_ERR_NORMAL_COMPLETION:
934 pUrb->enmStatus = VUSBSTATUS_OK;
935 break;
936 case USB_ERR_STALLED:
937 pUrb->enmStatus = VUSBSTATUS_STALL;
938 break;
939 default:
940 pUrb->enmStatus = VUSBSTATUS_INVALID;
941 break;
942 }
943
944 pUrb->Dev.pvPrivate = NULL;
945
946 switch (pUrb->enmType)
947 {
948 case VUSBXFERTYPE_MSG:
949 pUrb->cbData = pEndpointFBSD->acbData[0] + pEndpointFBSD->acbData[1];
950 break;
951 case VUSBXFERTYPE_ISOC:
952 {
953 int n;
954
955 if (pUrb->enmDir == VUSBDIRECTION_OUT)
956 break;
957 pUrb->cbData = 0;
958 for (n = 0; n < (int)pUrb->cIsocPkts; n++)
959 {
960 if (n >= (int)pEndpointFBSD->cMaxFrames)
961 break;
962 pUrb->cbData += pEndpointFBSD->acbData[n];
963 pUrb->aIsocPkts[n].cb = pEndpointFBSD->acbData[n];
964 }
965 for (; n < (int)pUrb->cIsocPkts; n++)
966 pUrb->aIsocPkts[n].cb = 0;
967
968 break;
969 }
970 default:
971 pUrb->cbData = pEndpointFBSD->acbData[0];
972 break;
973 }
974
975 LogFlow(("usbProxyFreeBSDUrbReap: Status=%d epindex=%u "
976 "len[0]=%d len[1]=%d\n",
977 (int)pXferEndpoint->status,
978 (unsigned)UsbFsComplete.ep_index,
979 (unsigned)pEndpointFBSD->acbData[0],
980 (unsigned)pEndpointFBSD->acbData[1]));
981
982 }
983 else if (cMillies && rc == VERR_RESOURCE_BUSY)
984 {
985 /* Poll for finished transfers */
986 PollFd.fd = RTFileToNative(pDevFBSD->hFile);
987 PollFd.events = POLLIN | POLLRDNORM;
988 PollFd.revents = 0;
989
990 rc = poll(&PollFd, 1, (cMillies == RT_INDEFINITE_WAIT) ? INFTIM : cMillies);
991 if (rc >= 1)
992 {
993 goto repeat;
994 }
995 else
996 {
997 LogFlow(("usbProxyFreeBSDUrbReap: "
998 "poll returned rc=%d\n", rc));
999 }
1000 }
1001 return pUrb;
1002}
1003
1004/**
1005 * Cancels the URB.
1006 * The URB requires reaping, so we don't change its state.
1007 */
1008static void usbProxyFreeBSDUrbCancel(PVUSBURB pUrb)
1009{
1010 PUSBPROXYDEV pProxyDev = PDMINS_2_DATA(pUrb->pUsbIns, PUSBPROXYDEV);
1011 int index;
1012
1013 index = (int)(long)pUrb->Dev.pvPrivate - 1;
1014
1015 if (index < 0 || index >= USBFBSD_MAXENDPOINTS)
1016 return;
1017
1018 LogFlow(("usbProxyFreeBSDUrbCancel: epindex=%u\n", (unsigned)index));
1019
1020 usbProxyFreeBSDEndpointClose(pProxyDev, index);
1021}
1022
1023/**
1024 * The FreeBSD USB Proxy Backend.
1025 */
1026extern const USBPROXYBACK g_USBProxyDeviceHost =
1027{
1028 "host",
1029 usbProxyFreeBSDOpen,
1030 usbProxyFreeBSDInit,
1031 usbProxyFreeBSDClose,
1032 usbProxyFreeBSDReset,
1033 usbProxyFreeBSDSetConfig,
1034 usbProxyFreeBSDClaimInterface,
1035 usbProxyFreeBSDReleaseInterface,
1036 usbProxyFreeBSDSetInterface,
1037 usbProxyFreeBSDClearHaltedEp,
1038 usbProxyFreeBSDUrbQueue,
1039 usbProxyFreeBSDUrbCancel,
1040 usbProxyFreeBSDUrbReap,
1041 0
1042};
1043
1044/*
1045 * Local Variables:
1046 * mode: c
1047 * c-file-style: "bsd"
1048 * c-basic-offset: 4
1049 * tab-width: 4
1050 * indent-tabs-mode: s
1051 * End:
1052 */
1053
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette