VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/usbip/USBProxyDevice-usbip.cpp@ 57034

Last change on this file since 57034 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: 35.0 KB
Line 
1/* $Id: USBProxyDevice-usbip.cpp 56292 2015-06-09 14:20:46Z vboxsync $ */
2/** @file
3 * USB device proxy - USB/IP backend.
4 */
5
6/*
7 * Copyright (C) 2014-2015 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18
19/*******************************************************************************
20* Header Files *
21*******************************************************************************/
22#define LOG_GROUP LOG_GROUP_DRV_USBPROXY
23
24#include <VBox/log.h>
25#include <VBox/err.h>
26#include <VBox/vmm/pdm.h>
27
28#include <iprt/asm.h>
29#include <iprt/assert.h>
30#include <iprt/alloc.h>
31#include <iprt/string.h>
32#include <iprt/socket.h>
33#include <iprt/poll.h>
34#include <iprt/tcp.h>
35#include <iprt/pipe.h>
36#include <iprt/list.h>
37#include <iprt/semaphore.h>
38
39#include "../USBProxyDevice.h"
40
41
42/*******************************************************************************
43* Constants And Macros, Structures and Typedefs *
44*******************************************************************************/
45
46/** The USB version number used for the protocol. */
47#define USBIP_VERSION UINT16_C(0x0100)
48/** Request indicator in the command code. */
49#define USBIP_INDICATOR_REQ RT_BIT(15)
50
51/** Command/Reply code for OP_REQ/RET_DEVLIST. */
52#define USBIP_REQ_RET_DEVLIST UINT16_C(5)
53/** Command/Reply code for OP_REQ/REP_IMPORT. */
54#define USBIP_REQ_RET_IMPORT UINT16_C(3)
55/** USB submit command identifier. */
56#define USBIP_CMD_SUBMIT UINT32_C(1)
57/** USB submit status identifier. */
58#define USBIP_RET_SUBMIT UINT32_C(2)
59/** URB unlink (cancel) command identifier. */
60#define USBIP_CMD_UNLINK UINT32_C(3)
61/** URB unlink (cancel) reply identifier. */
62#define USBIP_RET_UNLINK UINT32_C(4)
63
64/** Short read is not okay for the specified URB. */
65#define USBIP_XFER_FLAGS_SHORT_NOT_OK RT_BIT_32(0)
66/** Queue the isochronous URB as soon as possible. */
67#define USBIP_XFER_FLAGS_ISO_ASAP RT_BIT_32(1)
68/** Don't use DMA mappings for this URB. */
69#define USBIP_XFER_FLAGS_NO_TRANSFER_DMA_MAP RT_BIT_32(2)
70/** Explain - only applies to UHCI. */
71#define USBIP_XFER_FLAGS_FSBR RT_BIT_32(4)
72
73/** URB direction - input. */
74#define USBIP_DIR_IN UINT32_C(0)
75/** URB direction - output. */
76#define USBIP_DIR_OUT UINT32_C(1)
77
78/**
79 * Exported device entry in the OP_RET_DEVLIST reply.
80 */
81#pragma pack(1)
82typedef struct UsbIpExportedDevice
83{
84 /** Path of the device, zero terminated string. */
85 char szPath[256];
86 /** Bus ID of the exported device, zero terminated string. */
87 char szBusId[32];
88 /** Bus number. */
89 uint32_t u32BusNum;
90 /** Device number. */
91 uint32_t u32DevNum;
92 /** Speed indicator of the device. */
93 uint32_t u32Speed;
94 /** Vendor ID of the device. */
95 uint16_t u16VendorId;
96 /** Product ID of the device. */
97 uint16_t u16ProductId;
98 /** Device release number. */
99 uint16_t u16BcdDevice;
100 /** Device class. */
101 uint8_t bDeviceClass;
102 /** Device Subclass. */
103 uint8_t bDeviceSubClass;
104 /** Device protocol. */
105 uint8_t bDeviceProtocol;
106 /** Current configuration value of the device. */
107 uint8_t bNumConfigurations;
108 /** Number of interfaces for the device. */
109 uint8_t bNumInterfaces;
110} UsbIpExportedDevice;
111/** Pointer to a exported device entry. */
112typedef UsbIpExportedDevice *PUsbIpExportedDevice;
113#pragma pack()
114
115/**
116 * Interface descriptor entry for an exported device.
117 */
118#pragma pack(1)
119typedef struct UsbIpDeviceInterface
120{
121 /** Intefrace class. */
122 uint8_t bInterfaceClass;
123 /** Interface sub class. */
124 uint8_t bInterfaceSubClass;
125 /** Interface protocol identifier. */
126 uint8_t bInterfaceProtocol;
127 /** Padding byte for alignment. */
128 uint8_t bPadding;
129} UsbIpDeviceInterface;
130/** Pointer to an interface descriptor entry. */
131typedef UsbIpDeviceInterface *PUsbIpDeviceInterface;
132#pragma pack()
133
134/**
135 * USB/IP Import request.
136 */
137#pragma pack(1)
138typedef struct UsbIpReqImport
139{
140 /** Protocol version number. */
141 uint16_t u16Version;
142 /** Command code. */
143 uint16_t u16Cmd;
144 /** Status field, unused. */
145 uint32_t u32Status;
146 /** Bus Id of the device as zero terminated string. */
147 char aszBusId[32];
148} UsbIpReqImport;
149/** Pointer to a import request. */
150typedef UsbIpReqImport *PUsbIpReqImport;
151#pragma pack()
152
153/**
154 * USB/IP Import reply.
155 *
156 * This is only the header, for successful
157 * imports the device details are sent to as
158 * defined in UsbIpExportedDevice.
159 */
160#pragma pack(1)
161typedef struct UsbIpRetImport
162{
163 /** Protocol version number. */
164 uint16_t u16Version;
165 /** Command code. */
166 uint16_t u16Cmd;
167 /** Status field, unused. */
168 uint32_t u32Status;
169} UsbIpRetImport;
170/** Pointer to a import reply. */
171typedef UsbIpRetImport *PUsbIpRetImport;
172#pragma pack()
173
174/**
175 * Command/Reply header common to the submit and unlink commands
176 * replies.
177 */
178#pragma pack(1)
179typedef struct UsbIpReqRetHdr
180{
181 /** Request/Return code. */
182 uint32_t u32ReqRet;
183 /** Sequence number to identify the URB. */
184 uint32_t u32SeqNum;
185 /** Device id. */
186 uint32_t u32DevId;
187 /** Direction of the endpoint (host->device, device->host). */
188 uint32_t u32Direction;
189 /** Endpoint number. */
190 uint32_t u32Endpoint;
191} UsbIpReqRetHdr;
192/** Pointer to a request/reply header. */
193typedef UsbIpReqRetHdr *PUsbIpReqRetHdr;
194#pragma pack()
195
196/**
197 * USB/IP Submit request.
198 */
199#pragma pack(1)
200typedef struct UsbIpReqSubmit
201{
202 /** The request header. */
203 UsbIpReqRetHdr Hdr;
204 /** Transfer flags for the URB. */
205 uint32_t u32XferFlags;
206 /** Transfer buffer length. */
207 uint32_t u32TransferBufferLength;
208 /** Frame to transmit an ISO frame. */
209 uint32_t u32StartFrame;
210 /** Number of isochronous packets. */
211 uint32_t u32NumIsocPkts;
212 /** Maximum time for the request on the server side host controller. */
213 uint32_t u32Interval;
214 /** Setup data for a control URB. */
215 VUSBSETUP Setup;
216} UsbIpReqSubmit;
217/** Pointer to a submit request. */
218typedef UsbIpReqSubmit *PUsbIpReqSubmit;
219#pragma pack()
220
221/**
222 * USB/IP Submit reply.
223 */
224#pragma pack(1)
225typedef struct UsbIpRetSubmit
226{
227 /** The reply header. */
228 UsbIpReqRetHdr Hdr;
229 /** Status code. */
230 uint32_t u32Status;
231 /** Actual length of the reply buffer. */
232 uint32_t u32ActualLength;
233 /** The actual selected frame for a isochronous transmit. */
234 uint32_t u32StartFrame;
235 /** Number of isochronous packets. */
236 uint32_t u32NumIsocPkts;
237 /** Number of failed isochronous packets. */
238 uint32_t u32ErrorCount;
239 /** Setup data for a control URB. */
240 VUSBSETUP Setup;
241} UsbIpRetSubmit;
242/** Pointer to a submit reply. */
243typedef UsbIpRetSubmit *PUsbIpRetSubmit;
244#pragma pack()
245
246/**
247 * Unlink URB request.
248 */
249#pragma pack(1)
250typedef struct UsbIpReqUnlink
251{
252 /** The request header. */
253 UsbIpReqRetHdr Hdr;
254 /** The sequence number to unlink. */
255 uint32_t u32SeqNum;
256} UsbIpReqUnlink;
257/** Pointer to a URB unlink request. */
258typedef UsbIpReqUnlink *PUsbIpReqUnlink;
259#pragma pack()
260
261/**
262 * Unlink URB reply.
263 */
264#pragma pack(1)
265typedef struct UsbIpRetUnlink
266{
267 /** The reply header. */
268 UsbIpReqRetHdr Hdr;
269 /** Status of the request. */
270 uint32_t u32Status;
271} UsbIpRetUnlink;
272/** Pointer to a URB unlink request. */
273typedef UsbIpRetUnlink *PUsbIpRetUnlink;
274#pragma pack()
275
276/**
277 * Union of possible replies from the server during normal operation.
278 */
279#pragma pack(1)
280typedef union UsbIpRet
281{
282 /** The header. */
283 UsbIpReqRetHdr Hdr;
284 /** Submit reply. */
285 UsbIpRetSubmit RetSubmit;
286 /** Unlink reply. */
287 UsbIpRetUnlink RetUnlink;
288 /** Byte view. */
289 uint8_t abReply[1];
290} UsbIpRet;
291/** Pointer to a reply union. */
292typedef UsbIpRet *PUsbIpRet;
293#pragma pack()
294
295/**
296 * USB/IP backend specific data for one URB.
297 * Required for tracking in flight and landed URBs.
298 */
299typedef struct USBPROXYURBUSBIP
300{
301 /** List node for the in flight or landed URB list. */
302 RTLISTNODE NodeList;
303 /** Sequence number the assigned URB is identified by. */
304 uint32_t u32SeqNumUrb;
305 /** Pointer to the VUSB URB. */
306 PVUSBURB pVUsbUrb;
307} USBPROXYURBUSBIP;
308/** Pointer to a USB/IP URB. */
309typedef USBPROXYURBUSBIP *PUSBPROXYURBUSBIP;
310
311/**
312 * Backend data for the USB/IP USB Proxy device backend.
313 */
314typedef struct USBPROXYDEVUSBIP
315{
316 /** IPRT socket handle. */
317 RTSOCKET hSocket;
318 /** Pollset with the wakeup pipe and socket. */
319 RTPOLLSET hPollSet;
320 /** Pipe endpoint - read (in the pollset). */
321 RTPIPE hPipeR;
322 /** Pipe endpoint - write. */
323 RTPIPE hPipeW;
324 /** Flag whether the reaper thread was woken up. */
325 volatile bool fWokenUp;
326 /** Next sequence number to use for identifying submitted URBs. */
327 volatile uint32_t u32SeqNumNext;
328 /** Fast mutex protecting the lists below against concurrent access. */
329 RTSEMFASTMUTEX hMtxLists;
330 /** List of in flight URBs. */
331 RTLISTANCHOR ListUrbsInFlight;
332 /** List of landed URBs. */
333 RTLISTANCHOR ListUrbsLanded;
334 /** Port of the USB/IP host to connect to. */
335 uint32_t uPort;
336 /** USB/IP host address. */
337 char *pszHost;
338 /** USB Bus ID of the device to capture. */
339 char *pszBusId;
340 /** The device ID to use to identify the device. */
341 uint32_t u32DevId;
342} USBPROXYDEVUSBIP, *PUSBPROXYDEVUSBIP;
343
344/** Pollset id of the socket. */
345#define USBIP_POLL_ID_SOCKET 0
346/** Pollset id of the pipe. */
347#define USBIP_POLL_ID_PIPE 1
348
349/**
350 * Converts a request/reply header from network to host endianness.
351 *
352 * @returns nothing.
353 * @param pHdr The header to convert.
354 */
355DECLINLINE(void) usbProxyUsbIpReqRetHdrN2H(PUsbIpReqRetHdr pHdr)
356{
357 pHdr->u32ReqRet = RT_H2N_U32(pHdr->u32ReqRet);
358 pHdr->u32SeqNum = RT_H2N_U32(pHdr->u32SeqNum);
359 pHdr->u32DevId = RT_H2N_U32(pHdr->u32DevId);
360 pHdr->u32Direction = RT_H2N_U32(pHdr->u32Direction);
361 pHdr->u32Endpoint = RT_H2N_U32(pHdr->u32Endpoint);
362}
363
364/**
365 * Converts a request/reply header from host to network endianness.
366 *
367 * @returns nothing.
368 * @param pHdr The header to convert.
369 */
370DECLINLINE(void) usbProxyUsbIpReqRetHdrH2N(PUsbIpReqRetHdr pHdr)
371{
372 pHdr->u32ReqRet = RT_N2H_U32(pHdr->u32ReqRet);
373 pHdr->u32SeqNum = RT_N2H_U32(pHdr->u32SeqNum);
374 pHdr->u32DevId = RT_N2H_U32(pHdr->u32DevId);
375 pHdr->u32Direction = RT_N2H_U32(pHdr->u32Direction);
376 pHdr->u32Endpoint = RT_N2H_U32(pHdr->u32Endpoint);
377}
378
379/**
380 * Converts a submit request from host to network endianness.
381 *
382 * @returns nothing.
383 * @param pReqSubmit The submit request to convert.
384 */
385DECLINLINE(void) usbProxyUsbIpReqSubmitH2N(PUsbIpReqSubmit pReqSubmit)
386{
387 usbProxyUsbIpReqRetHdrH2N(&pReqSubmit->Hdr);
388 pReqSubmit->u32XferFlags = RT_H2N_U32(pReqSubmit->u32XferFlags);
389 pReqSubmit->u32TransferBufferLength = RT_H2N_U32(pReqSubmit->u32TransferBufferLength);
390 pReqSubmit->u32StartFrame = RT_H2N_U32(pReqSubmit->u32StartFrame);
391 pReqSubmit->u32NumIsocPkts = RT_H2N_U32(pReqSubmit->u32NumIsocPkts);
392 pReqSubmit->u32Interval = RT_H2N_U32(pReqSubmit->u32Interval);
393}
394
395/**
396 * Converts a submit reply from network to host endianness.
397 *
398 * @returns nothing.
399 * @param pReqSubmit The submit reply to convert.
400 */
401DECLINLINE(void) usbProxyUsbIpRetSubmitN2H(PUsbIpRetSubmit pRetSubmit)
402{
403 usbProxyUsbIpReqRetHdrN2H(&pRetSubmit->Hdr);
404 pRetSubmit->u32Status = RT_N2H_U32(pRetSubmit->u32Status);
405 pRetSubmit->u32ActualLength = RT_N2H_U32(pRetSubmit->u32ActualLength);
406 pRetSubmit->u32StartFrame = RT_N2H_U32(pRetSubmit->u32StartFrame);
407 pRetSubmit->u32NumIsocPkts = RT_N2H_U32(pRetSubmit->u32NumIsocPkts);
408 pRetSubmit->u32ErrorCount = RT_N2H_U32(pRetSubmit->u32ErrorCount);
409}
410
411/**
412 * Converts a unlink request from host to network endianness.
413 *
414 * @returns nothing.
415 * @param pReqUnlink The unlink request to convert.
416 */
417DECLINLINE(void) usbProxyUsbIpReqUnlinkH2N(PUsbIpReqUnlink pReqUnlink)
418{
419 usbProxyUsbIpReqRetHdrH2N(&pReqUnlink->Hdr);
420 pReqUnlink->u32SeqNum = RT_H2N_U32(pReqUnlink->u32SeqNum);
421}
422
423/**
424 * Converts a unlink reply from network to host endianness.
425 *
426 * @returns nothing.
427 * @param pRetUnlink The unlink reply to convert.
428 */
429DECLINLINE(void) usbProxyUsbIpRetUnlinkN2H(PUsbIpRetUnlink pRetUnlink)
430{
431 usbProxyUsbIpReqRetHdrN2H(&pRetUnlink->Hdr);
432 pRetUnlink->u32Status = RT_N2H_U32(pRetUnlink->u32Status);
433}
434
435/**
436 * Convert the given exported device structure from host to network byte order.
437 *
438 * @returns nothing.
439 * @param pDevice The device structure to convert.
440 */
441DECLINLINE(void) usbProxyUsbIpExportedDeviceN2H(PUsbIpExportedDevice pDevice)
442{
443 pDevice->u32BusNum = RT_N2H_U32(pDevice->u32BusNum);
444 pDevice->u32DevNum = RT_N2H_U32(pDevice->u32DevNum);
445 pDevice->u32Speed = RT_N2H_U16(pDevice->u32Speed);
446 pDevice->u16VendorId = RT_N2H_U16(pDevice->u16VendorId);
447 pDevice->u16ProductId = RT_N2H_U16(pDevice->u16ProductId);
448 pDevice->u16BcdDevice = RT_N2H_U16(pDevice->u16BcdDevice);
449}
450
451/**
452 * Converts a USB/IP status code to VBox status code.
453 *
454 * @returns VBox status code.
455 * @param u32Status The USB/IP status code from the reply.
456 */
457DECLINLINE(int) usbProxyUsbIpErrConvertFromStatus(uint32_t u32Status)
458{
459 if (RT_LIKELY(u32Status == 0))
460 return VINF_SUCCESS;
461
462 return VERR_NOT_IMPLEMENTED;
463}
464
465/**
466 * Gets the next free sequence number.
467 *
468 * @returns Next free sequence number.
469 * @param pProxyDevUsbIp The USB/IP proxy device data.
470 */
471DECLINLINE(uint32_t) usbProxyUsbIpSeqNumGet(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
472{
473 return ASMAtomicIncU32(&pProxyDevUsbIp->u32SeqNumNext);
474}
475
476/**
477 * Links a given URB into the given list.
478 *
479 * @returns nothing.
480 * @param pProxyDevUsbIp The USB/IP proxy device data.
481 * @param pList The list to link the URB into.
482 * @param pUrbUsbIp The URB to link.
483 */
484DECLINLINE(void) usbProxyUsbIpLinkUrb(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PRTLISTANCHOR pList, PUSBPROXYURBUSBIP pUrbUsbIp)
485{
486 int rc = RTSemFastMutexRequest(pProxyDevUsbIp->hMtxLists);
487 AssertRC(rc);
488 RTListAppend(pList, &pUrbUsbIp->NodeList);
489 RTSemFastMutexRelease(pProxyDevUsbIp->hMtxLists);
490}
491
492/**
493 * Unlinks a given URB from the current assigned list.
494 *
495 * @returns nothing.
496 * @param pProxyDevUsbIp The USB/IP proxy device data.
497 * @param pUrbUsbIp The URB to unlink.
498 */
499DECLINLINE(void) usbProxyUsbIpUnlinkUrb(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PUSBPROXYURBUSBIP pUrbUsbIp)
500{
501 int rc = RTSemFastMutexRequest(pProxyDevUsbIp->hMtxLists);
502 AssertRC(rc);
503 RTListNodeRemove(&pUrbUsbIp->NodeList);
504 RTSemFastMutexRelease(pProxyDevUsbIp->hMtxLists);
505}
506
507/**
508 * Allocates a USB/IP proxy specific URB state.
509 *
510 * @returns Pointer to the USB/IP specific URB data or NULL on failure.
511 * @param pProxyDevUsbIp The USB/IP proxy device data.
512 */
513static PUSBPROXYURBUSBIP usbProxyUsbIpUrbAlloc(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
514{
515 NOREF(pProxyDevUsbIp);
516 return (PUSBPROXYURBUSBIP)RTMemAllocZ(sizeof(USBPROXYURBUSBIP));
517}
518
519/**
520 * Frees the given USB/IP URB state.
521 *
522 * @returns nothing.
523 * @param pProxyDevUsbIp The USB/IP proxy device data.
524 * @param pUrbUsbIp The USB/IP speciic URB data.
525 */
526static void usbProxyUsbIpUrbFree(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PUSBPROXYURBUSBIP pUrbUsbIp)
527{
528 NOREF(pProxyDevUsbIp);
529 RTMemFree(pUrbUsbIp);
530}
531
532/**
533 * Parse the string representation of the host address.
534 *
535 * @returns VBox status code.
536 * @param pProxyDevUsbIp The USB/IP proxy device data to parse the address for.
537 * @param pszAddress The address string to parse.
538 */
539static int usbProxyUsbIpParseAddress(PUSBPROXYDEVUSBIP pProxyDevUsbIp, const char *pszAddress)
540{
541 int rc = VINF_SUCCESS;
542 uint32_t uPort;
543 char *pszPortStart = RTStrStr(pszAddress, ":");
544 char *pszHost = NULL;
545 if (pszPortStart)
546 {
547 size_t cbHost = pszPortStart - pszAddress;
548 pszPortStart++;
549 rc = RTStrToUInt32Ex(pszPortStart, NULL, 10 /* uBase */, &pProxyDevUsbIp->uPort);
550 if ( rc == VINF_SUCCESS
551 || cbHost == 0)
552 {
553 rc = RTStrAllocEx(&pProxyDevUsbIp->pszHost, cbHost + 1);
554 if (RT_SUCCESS(rc))
555 {
556 rc = RTStrCopyEx(pProxyDevUsbIp->pszHost, cbHost + 1, pszAddress, cbHost);
557 AssertRC(rc);
558 return VINF_SUCCESS;
559 }
560 }
561 else
562 rc = VERR_INVALID_PARAMETER;
563 }
564 else
565 rc = VERR_INVALID_PARAMETER;
566
567 return rc;
568}
569
570/**
571 * Connects to the USB/IP host and claims the device given in the proxy device data.
572 *
573 * @returns VBox status.
574 * @param pProxyDevUsbIp The USB/IP proxy device data.
575 */
576static int usbProxyUsbIpConnect(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
577{
578 int rc = VINF_SUCCESS;
579 rc = RTTcpClientConnect(pProxyDevUsbIp->pszHost, pProxyDevUsbIp->uPort, &pProxyDevUsbIp->hSocket);
580 if (RT_SUCCESS(rc))
581 {
582 /* Disable send coalescing. */
583 rc = RTTcpSetSendCoalescing(pProxyDevUsbIp->hSocket, false);
584 if (RT_FAILURE(rc))
585 LogRel(("UsbIp: Disabling send coalescing failed (rc=%Rrc), continuing nevertheless but expect reduced performance\n", rc));
586
587 /* Import the device, i.e. claim it for our use. */
588 UsbIpReqImport ReqImport;
589 ReqImport.u16Version = RT_H2N_U16(USBIP_VERSION);
590 ReqImport.u16Cmd = RT_H2N_U16(USBIP_INDICATOR_REQ | USBIP_REQ_RET_IMPORT);
591 ReqImport.u32Status = RT_H2N_U32(0);
592 rc = RTStrCopy(&ReqImport.aszBusId[0], sizeof(ReqImport.aszBusId[0]), pProxyDevUsbIp->pszBusId);
593 if (rc == VINF_SUCCESS)
594 {
595 rc = RTTcpWrite(pProxyDevUsbIp->hSocket, &ReqImport, sizeof(ReqImport));
596 if (RT_SUCCESS(rc))
597 {
598 /* Read the reply. */
599 UsbIpRetImport RetImport;
600 rc = RTTcpRead(pProxyDevUsbIp->hSocket, &RetImport, sizeof(RetImport), NULL);
601 if (RT_SUCCESS(rc))
602 {
603 RetImport.u16Version = RT_N2H_U16(RetImport.u16Version);
604 RetImport.u16Cmd = RT_N2H_U16(RetImport.u16Cmd);
605 RetImport.u32Status = RT_N2H_U16(RetImport.u32Status);
606 if ( RetImport.u16Version == USBIP_VERSION
607 && RetImport.u16Cmd == USBIP_REQ_RET_IMPORT
608 && RetImport.u32Status == 0)
609 {
610 /* Read the device data. */
611 UsbIpExportedDevice Device;
612 rc = RTTcpRead(pProxyDevUsbIp->hSocket, &Device, sizeof(Device), NULL);
613 if (RT_SUCCESS(rc))
614 {
615 usbProxyUsbIpExportedDeviceN2H(&Device);
616 pProxyDevUsbIp->u32DevId = (Device.u32BusNum << 16) | Device.u32DevNum;
617
618 rc = RTPollSetAddSocket(pProxyDevUsbIp->hPollSet, pProxyDevUsbIp->hSocket,
619 RTPOLL_EVT_READ | RTPOLL_EVT_ERROR, USBIP_POLL_ID_SOCKET);
620 }
621 }
622 else
623 {
624 /* Check what went wrong and leave a meaningful error message in the log. */
625 if (RetImport.u16Version != USBIP_VERSION)
626 LogRel(("UsbIp: Unexpected protocol version received from host (%#x vs. %#x)\n",
627 RetImport.u16Version, USBIP_VERSION));
628 else if (RetImport.u16Cmd != USBIP_REQ_RET_IMPORT)
629 LogRel(("UsbIp: Unexpected reply code received from host (%#x vs. %#x)\n",
630 RetImport.u16Cmd, USBIP_REQ_RET_IMPORT));
631 else if (RetImport.u32Status != 0)
632 LogRel(("UsbIp: Claiming the device has failed on the host with an unspecified error\n"));
633 else
634 AssertMsgFailed(("Something went wrong with if condition\n"));
635 }
636 }
637 }
638 }
639 else
640 {
641 LogRel(("UsbIp: Given bus ID is exceeds permitted protocol length: %u vs %u\n",
642 strlen(pProxyDevUsbIp->pszBusId) + 1, sizeof(ReqImport.aszBusId[0])));
643 rc = VERR_INVALID_PARAMETER;
644 }
645
646 if (RT_FAILURE(rc))
647 RTTcpClientCloseEx(pProxyDevUsbIp->hSocket, false /*fGracefulShutdown*/);
648 }
649 if (RT_FAILURE(rc))
650 LogRel(("UsbIp: Connecting to the host %s failed with %Rrc\n", pProxyDevUsbIp->pszHost, rc));
651 return rc;
652}
653
654/**
655 * Disconnects from the USB/IP host releasing the device given in the proxy device data.
656 *
657 * @returns VBox status code.
658 * @param pProxyDevUsbIp The USB/IP proxy device data.
659 */
660static int usbProxyUsbIpDisconnect(PUSBPROXYDEVUSBIP pProxyDevUsbIp)
661{
662 int rc = VINF_SUCCESS;
663
664 rc = RTTcpClientCloseEx(pProxyDevUsbIp->hSocket, false /*fGracefulShutdown*/);
665 if (RT_SUCCESS(rc))
666 pProxyDevUsbIp->hSocket = NIL_RTSOCKET;
667 return rc;
668}
669
670/**
671 * Synchronously exchange a given control message with the remote device.
672 *
673 * @eturns VBox status code.
674 * @param pProxyDevUsbIp The USB/IP proxy device data.
675 * @param pSetup The setup message.
676 *
677 * @note This method is only used to implement the *SetConfig, *SetInterface and *ClearHaltedEp
678 * callbacks because the USB/IP protocol lacks dedicated requests for these.
679 * @remark It is assumed that this method is never called while usbProxyUsbIpUrbReap is called
680 * on another thread.
681 */
682static int usbProxyUsbIpCtrlUrbExchangeSync(PUSBPROXYDEVUSBIP pProxyDevUsbIp, PVUSBSETUP pSetup)
683{
684 int rc = VINF_SUCCESS;
685
686 UsbIpReqSubmit ReqSubmit;
687 uint32_t u32SeqNum = usbProxyUsbIpSeqNumGet(pProxyDevUsbIp);
688 ReqSubmit.Hdr.u32ReqRet = USBIP_CMD_SUBMIT;
689 ReqSubmit.Hdr.u32SeqNum = u32SeqNum;
690 ReqSubmit.Hdr.u32DevId = pProxyDevUsbIp->u32DevId;
691 ReqSubmit.Hdr.u32Direction = USBIP_DIR_OUT;
692 ReqSubmit.Hdr.u32Endpoint = 0; /* Only default control endpoint is allowed for these kind of messages. */
693 ReqSubmit.u32XferFlags = 0;
694 ReqSubmit.u32TransferBufferLength = 0;
695 ReqSubmit.u32StartFrame = 0;
696 ReqSubmit.u32NumIsocPkts = 0;
697 ReqSubmit.u32Interval = 0;
698 memcpy(&ReqSubmit.Setup, pSetup, sizeof(ReqSubmit.Setup));
699 usbProxyUsbIpReqSubmitH2N(&ReqSubmit);
700
701 /* Send the command. */
702 rc = RTTcpWrite(pProxyDevUsbIp->hSocket, &ReqSubmit, sizeof(ReqSubmit));
703 if (RT_SUCCESS(rc))
704 {
705 /* Wait for the response. */
706 /** @todo: Don't wait indefinitely long. */
707 UsbIpRetSubmit RetSubmit;
708 rc = RTTcpRead(pProxyDevUsbIp->hSocket, &RetSubmit, sizeof(RetSubmit), NULL);
709 if (RT_SUCCESS(rc))
710 {
711 usbProxyUsbIpRetSubmitN2H(&RetSubmit);
712 rc = usbProxyUsbIpErrConvertFromStatus(RetSubmit.u32Status);
713 }
714 }
715 return rc;
716}
717
718/*
719 * The USB proxy device functions.
720 */
721
722static DECLCALLBACK(int) usbProxyUsbIpOpen(PUSBPROXYDEV pProxyDev, const char *pszAddress, void *pvBackend)
723{
724 LogFlowFunc(("pProxyDev=%p pszAddress=%s, pvBackend=%p\n", pProxyDev, pszAddress, pvBackend));
725
726 PUSBPROXYDEVUSBIP pDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
727 int rc = VINF_SUCCESS;
728
729 RTListInit(&pDevUsbIp->ListUrbsInFlight);
730 RTListInit(&pDevUsbIp->ListUrbsLanded);
731 pDevUsbIp->hSocket = NIL_RTSOCKET;
732 pDevUsbIp->hPollSet = NIL_RTPOLLSET;
733 pDevUsbIp->hPipeW = NIL_RTPIPE;
734 pDevUsbIp->hPipeR = NIL_RTPIPE;
735 pDevUsbIp->fWokenUp = false;
736 pDevUsbIp->u32SeqNumNext = 0;
737 pDevUsbIp->pszHost = NULL;
738 pDevUsbIp->pszBusId = NULL;
739
740 /* Setup wakeup pipe and poll set first. */
741 rc = RTPipeCreate(&pDevUsbIp->hPipeR, &pDevUsbIp->hPipeW, 0);
742 if (RT_SUCCESS(rc))
743 {
744 rc = RTPollSetCreate(&pDevUsbIp->hPollSet);
745 if (RT_SUCCESS(rc))
746 {
747 rc = RTPollSetAddPipe(pDevUsbIp->hPollSet, pDevUsbIp->hPipeR,
748 RTPOLL_EVT_READ, USBIP_POLL_ID_PIPE);
749 if (RT_SUCCESS(rc))
750 {
751 /* Connect to the USB/IP host. */
752 rc = usbProxyUsbIpParseAddress(pDevUsbIp, pszAddress);
753 if (RT_SUCCESS(rc))
754 rc = usbProxyUsbIpConnect(pDevUsbIp);
755 }
756
757 if (RT_FAILURE(rc))
758 {
759 RTPollSetRemove(pDevUsbIp->hPollSet, USBIP_POLL_ID_PIPE);
760 int rc2 = RTPollSetDestroy(pDevUsbIp->hPollSet);
761 AssertRC(rc2);
762 }
763 }
764
765 if (RT_FAILURE(rc))
766 {
767 int rc2 = RTPipeClose(pDevUsbIp->hPipeR);
768 AssertRC(rc2);
769 rc2 = RTPipeClose(pDevUsbIp->hPipeW);
770 AssertRC(rc2);
771 }
772 }
773
774 return rc;
775}
776
777static DECLCALLBACK(void) usbProxyUsbIpClose(PUSBPROXYDEV pProxyDev)
778{
779 int rc = VINF_SUCCESS;
780 LogFlowFunc(("pProxyDev = %p\n", pProxyDev));
781
782 PUSBPROXYDEVUSBIP pDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
783 /* Destroy the pipe and pollset if necessary. */
784 if (pDevUsbIp->hPollSet != NIL_RTPOLLSET)
785 {
786 if (pDevUsbIp->hSocket != NIL_RTSOCKET)
787 {
788 rc = RTPollSetRemove(pDevUsbIp->hPollSet, USBIP_POLL_ID_SOCKET);
789 Assert(RT_SUCCESS(rc) || rc == VERR_POLL_HANDLE_ID_NOT_FOUND);
790 }
791 rc = RTPollSetRemove(pDevUsbIp->hPollSet, USBIP_POLL_ID_PIPE);
792 AssertRC(rc);
793 rc = RTPollSetDestroy(pDevUsbIp->hPollSet);
794 AssertRC(rc);
795 rc = RTPipeClose(pDevUsbIp->hPipeR);
796 AssertRC(rc);
797 rc = RTPipeClose(pDevUsbIp->hPipeW);
798 AssertRC(rc);
799 }
800
801 if (pDevUsbIp->hSocket != NIL_RTSOCKET)
802 usbProxyUsbIpDisconnect(pDevUsbIp);
803 if (pDevUsbIp->pszHost)
804 RTStrFree(pDevUsbIp->pszHost);
805 if (pDevUsbIp->pszBusId)
806 RTStrFree(pDevUsbIp->pszBusId);
807
808 /* Clear the URB lists. */
809 rc = RTSemFastMutexRequest(pDevUsbIp->hMtxLists);
810 AssertRC(rc);
811 PUSBPROXYURBUSBIP pIter = NULL;
812 PUSBPROXYURBUSBIP pIterNext = NULL;
813 RTListForEachSafe(&pDevUsbIp->ListUrbsInFlight, pIter, pIterNext, USBPROXYURBUSBIP, NodeList)
814 {
815 RTListNodeRemove(&pIter->NodeList);
816 RTMemFree(pIter);
817 }
818
819 RTListForEachSafe(&pDevUsbIp->ListUrbsLanded, pIter, pIterNext, USBPROXYURBUSBIP, NodeList)
820 {
821 RTListNodeRemove(&pIter->NodeList);
822 RTMemFree(pIter);
823 }
824 RTSemFastMutexRelease(pDevUsbIp->hMtxLists);
825 RTSemFastMutexDestroy(pDevUsbIp->hMtxLists);
826}
827
828static DECLCALLBACK(int) usbProxyUsbIpReset(PUSBPROXYDEV pProxyDev, bool fResetOnLinux)
829{
830 LogFlowFunc(("pProxyDev = %p\n", pProxyDev));
831
832 PUSBPROXYDEVUSBIP pDev = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
833 return VINF_SUCCESS; /* No way to reset the device with the current protocol. */
834}
835
836static DECLCALLBACK(int) usbProxyUsbIpSetConfig(PUSBPROXYDEV pProxyDev, int iCfg)
837{
838 LogFlowFunc(("pProxyDev=%s cfg=%#x\n", pProxyDev->pUsbIns->pszName, iCfg));
839
840 PUSBPROXYDEVUSBIP pDev = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
841 VUSBSETUP Setup;
842
843 Setup.bmRequestType = 0;
844 Setup.bRequest = 0x09;
845 Setup.wValue = iCfg;
846 Setup.wIndex = 0;
847 Setup.wLength = 0;
848 return usbProxyUsbIpCtrlUrbExchangeSync(pDev, &Setup);
849}
850
851static DECLCALLBACK(int) usbProxyUsbIpClaimInterface(PUSBPROXYDEV pProxyDev, int ifnum)
852{
853 LogFlowFunc(("pProxyDev=%s ifnum=%#x\n", pProxyDev->pUsbIns->pszName, ifnum));
854 return VINF_SUCCESS;
855}
856
857static DECLCALLBACK(int) usbProxyUsbIpReleaseInterface(PUSBPROXYDEV pProxyDev, int ifnum)
858{
859 LogFlowFunc(("pProxyDev=%s ifnum=%#x\n", pProxyDev->pUsbIns->pszName, ifnum));
860 return VINF_SUCCESS;
861}
862
863static DECLCALLBACK(int) usbProxyUsbIpSetInterface(PUSBPROXYDEV pProxyDev, int ifnum, int setting)
864{
865 LogFlowFunc(("pProxyDev=%p ifnum=%#x setting=%#x\n", pProxyDev, ifnum, setting));
866
867 PUSBPROXYDEVUSBIP pDev = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
868 VUSBSETUP Setup;
869
870 Setup.bmRequestType = 0x1;
871 Setup.bRequest = 0x11; /* SET_INTERFACE */
872 Setup.wValue = setting;
873 Setup.wIndex = ifnum;
874 Setup.wLength = 0;
875 return usbProxyUsbIpCtrlUrbExchangeSync(pDev, &Setup);
876}
877
878static DECLCALLBACK(int) usbProxyUsbIpClearHaltedEp(PUSBPROXYDEV pProxyDev, unsigned int iEp)
879{
880 LogFlowFunc(("pProxyDev=%s ep=%u\n", pProxyDev->pUsbIns->pszName, iEp));
881
882 PUSBPROXYDEVUSBIP pDev = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
883 VUSBSETUP Setup;
884
885 Setup.bmRequestType = 0x2;
886 Setup.bRequest = 0x01; /* CLEAR_FEATURE */
887 Setup.wValue = 0x00; /* ENDPOINT_HALT */
888 Setup.wIndex = iEp;
889 Setup.wLength = 0;
890 return usbProxyUsbIpCtrlUrbExchangeSync(pDev, &Setup);
891}
892
893static DECLCALLBACK(int) usbProxyUsbIpUrbQueue(PUSBPROXYDEV pProxyDev, PVUSBURB pUrb)
894{
895 LogFlowFunc(("pUrb=%p\n", pUrb));
896
897 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
898
899 /* Allocate a USB/IP Urb. */
900 PUSBPROXYURBUSBIP pUrbUsbIp = usbProxyUsbIpUrbAlloc(pProxyDevUsbIp);
901 if (!pUrbUsbIp)
902 return VERR_NO_MEMORY;
903
904 pUrbUsbIp->u32SeqNumUrb = usbProxyUsbIpSeqNumGet(pProxyDevUsbIp);
905
906 UsbIpReqSubmit ReqSubmit;
907 ReqSubmit.Hdr.u32ReqRet = USBIP_CMD_SUBMIT;
908 ReqSubmit.Hdr.u32SeqNum = pUrbUsbIp->u32SeqNumUrb;
909 ReqSubmit.Hdr.u32DevId = pProxyDevUsbIp->u32DevId;
910 ReqSubmit.Hdr.u32Endpoint = pUrb->EndPt;
911 ReqSubmit.Hdr.u32Direction = pUrb->enmDir == VUSBDIRECTION_IN ? USBIP_DIR_IN : USBIP_DIR_OUT;
912 ReqSubmit.u32XferFlags = 0;
913 if (pUrb->enmDir == VUSBDIRECTION_IN && pUrb->fShortNotOk)
914 ReqSubmit.u32XferFlags |= USBIP_XFER_FLAGS_SHORT_NOT_OK;
915
916 ReqSubmit.u32TransferBufferLength = pUrb->cbData;
917 ReqSubmit.u32StartFrame = 0;
918 ReqSubmit.u32NumIsocPkts = 0;
919 ReqSubmit.u32Interval = 0;
920
921 switch (pUrb->enmType)
922 {
923 case VUSBXFERTYPE_MSG:
924 memcpy(&ReqSubmit.Setup, &pUrb->abData, sizeof(ReqSubmit.Setup));
925 LogFlowFunc(("Message (Control) URB\n"));
926 break;
927 case VUSBXFERTYPE_ISOC:
928 ReqSubmit.u32XferFlags |= USBIP_XFER_FLAGS_ISO_ASAP;
929 ReqSubmit.u32NumIsocPkts = pUrb->cIsocPkts;
930#if 0
931 for (unsigned i = 0; i < pUrb->cIsocPkts; i++)
932 {
933 pUrbLnx->KUrb.iso_frame_desc[i].length = pUrb->aIsocPkts[i].cb;
934 pUrbLnx->KUrb.iso_frame_desc[i].actual_length = 0;
935 pUrbLnx->KUrb.iso_frame_desc[i].status = 0x7fff;
936 }
937#else /** @todo: Implement isochronous support */
938 usbProxyUsbIpUrbFree(pProxyDevUsbIp, pUrbUsbIp);
939 return VERR_NOT_SUPPORTED;
940#endif
941 break;
942 case VUSBXFERTYPE_BULK:
943 case VUSBXFERTYPE_INTR:
944 break;
945 default:
946 usbProxyUsbIpUrbFree(pProxyDevUsbIp, pUrbUsbIp);
947 return VERR_INVALID_PARAMETER; /** @todo: better status code. */
948 }
949 usbProxyUsbIpReqSubmitH2N(&ReqSubmit);
950
951 /* Send the command. */
952 RTSGBUF SgBufReq;
953 RTSGSEG aSegReq[2];
954 aSegReq[0].pvSeg = &ReqSubmit;
955 aSegReq[0].cbSeg = sizeof(ReqSubmit);
956 aSegReq[1].pvSeg = &pUrb->abData[0];
957 aSegReq[1].cbSeg = pUrb->cbData;
958 RTSgBufInit(&SgBufReq, &aSegReq[0], RT_ELEMENTS(aSegReq));
959
960 int rc = RTTcpSgWrite(pProxyDevUsbIp->hSocket, &SgBufReq);
961 if (RT_SUCCESS(rc))
962 {
963 /* Link the URB into the list of in flight URBs. */
964 pUrb->Dev.pvPrivate = pUrbUsbIp;
965 pUrbUsbIp->pVUsbUrb = pUrb;
966 usbProxyUsbIpLinkUrb(pProxyDevUsbIp, &pProxyDevUsbIp->ListUrbsInFlight, pUrbUsbIp);
967 }
968 else
969 usbProxyUsbIpUrbFree(pProxyDevUsbIp, pUrbUsbIp);
970
971 return rc;
972}
973
974static DECLCALLBACK(PVUSBURB) usbProxyUsbIpUrbReap(PUSBPROXYDEV pProxyDev, RTMSINTERVAL cMillies)
975{
976 LogFlowFunc(("pProxyDev=%s\n", pProxyDev->pUsbIns->pszName));
977
978 PUSBPROXYDEVUSBIP pDev = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
979 PUSBPROXYURBUSBIP pUrbUsbIp = NULL;
980 PVUSBURB pUrb = NULL;
981
982 /* Any URBs pending delivery? */
983 if (!RTListIsEmpty(&pDev->ListUrbsLanded))
984 {
985 pUrbUsbIp = RTListGetFirst(&pDev->ListUrbsLanded, USBPROXYURBUSBIP, NodeList);
986 if (pUrbUsbIp)
987 {
988 /* unlink from the pending delivery list */
989 usbProxyUsbIpUnlinkUrb(pDev, pUrbUsbIp);
990 }
991 }
992
993 if (!pUrbUsbIp)
994 {
995 uint32_t uIdReady = 0;
996 uint32_t fEventsRecv = 0;
997
998 if (!ASMAtomicXchgBool(&pDev->fWokenUp, false))
999 {
1000 int rc = RTPoll(pDev->hPollSet, cMillies, &fEventsRecv, &uIdReady);
1001 Assert(RT_SUCCESS(rc) || rc == VERR_TIMEOUT);
1002 }
1003
1004 UsbIpRet Reply;
1005 }
1006
1007 if (pUrbUsbIp)
1008 {
1009
1010 }
1011
1012 return pUrb;
1013}
1014
1015static DECLCALLBACK(int) usbProxyUsbIpUrbCancel(PUSBPROXYDEV pProxyDev, PVUSBURB pUrb)
1016{
1017 LogFlowFunc(("pUrb=%p\n", pUrb));
1018
1019 PUSBPROXYDEVUSBIP pProxyDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1020 PUSBPROXYURBUSBIP pUrbUsbIp = (PUSBPROXYURBUSBIP)pUrb->Dev.pvPrivate;
1021 UsbIpReqUnlink ReqUnlink;
1022
1023 uint32_t u32SeqNum = usbProxyUsbIpSeqNumGet(pProxyDevUsbIp);
1024 ReqUnlink.Hdr.u32ReqRet = USBIP_CMD_UNLINK;
1025 ReqUnlink.Hdr.u32SeqNum = u32SeqNum;
1026 ReqUnlink.Hdr.u32DevId = pProxyDevUsbIp->u32DevId;
1027 ReqUnlink.Hdr.u32Direction = USBIP_DIR_OUT;
1028 ReqUnlink.Hdr.u32Endpoint = pUrb->EndPt;
1029 ReqUnlink.u32SeqNum = pUrbUsbIp->u32SeqNumUrb;
1030
1031 int rc = RTTcpWrite(pProxyDevUsbIp->hSocket, &ReqUnlink, sizeof(ReqUnlink));
1032 /* Wait for the reply. */
1033 return rc;
1034}
1035
1036static DECLCALLBACK(int) usbProxyUsbIpWakeup(PUSBPROXYDEV pProxyDev)
1037{
1038 LogFlowFunc(("pProxyDev=%s\n", pProxyDev->pUsbIns->pszName));
1039
1040 PUSBPROXYDEVUSBIP pDevUsbIp = USBPROXYDEV_2_DATA(pProxyDev, PUSBPROXYDEVUSBIP);
1041 int rc = VINF_SUCCESS;
1042
1043 if (!ASMAtomicXchgBool(&pDevUsbIp->fWokenUp, true))
1044 {
1045 size_t cbWritten = 0;
1046
1047 rc = RTPipeWrite(pDevUsbIp->hPipeW, "", 1, &cbWritten);
1048 Assert(RT_SUCCESS(rc) || cbWritten == 0);
1049 }
1050
1051 return rc;
1052}
1053
1054/**
1055 * The USB/IP USB Proxy Backend operations.
1056 */
1057extern const USBPROXYBACK g_USBProxyDeviceUsbIp =
1058{
1059 /* pszName */
1060 "usbip",
1061 /* cbBackend */
1062 sizeof(USBPROXYDEVUSBIP),
1063 usbProxyUsbIpOpen,
1064 NULL,
1065 usbProxyUsbIpClose,
1066 usbProxyUsbIpReset,
1067 usbProxyUsbIpSetConfig,
1068 usbProxyUsbIpClaimInterface,
1069 usbProxyUsbIpReleaseInterface,
1070 usbProxyUsbIpSetInterface,
1071 usbProxyUsbIpClearHaltedEp,
1072 usbProxyUsbIpUrbQueue,
1073 usbProxyUsbIpUrbCancel,
1074 usbProxyUsbIpUrbReap,
1075 usbProxyUsbIpWakeup,
1076 0
1077};
1078
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