VirtualBox

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

Last change on this file since 56292 was 56292, checked in by vboxsync, 10 years ago

Devices: Updated (C) year.

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